2022-09-25 12:47:31 +03:00
|
|
|
|
/* https://en.wikipedia.org/wiki/Operating_system_abstraction_layer */
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
|
|
|
|
/*
|
2022-01-15 18:50:22 +03:00
|
|
|
|
* Copyright 2015-2022 Leonid Yuriev <leo@yuriev.ru>
|
2017-03-16 18:09:27 +03:00
|
|
|
|
* and other libmdbx authors: please see AUTHORS file.
|
|
|
|
|
* All rights reserved.
|
|
|
|
|
*
|
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
|
* modification, are permitted only as authorized by the OpenLDAP
|
|
|
|
|
* Public License.
|
|
|
|
|
*
|
|
|
|
|
* A copy of this license is available in the file LICENSE in the
|
|
|
|
|
* top-level directory of the distribution or, alternatively, at
|
|
|
|
|
* <http://www.OpenLDAP.org/license.html>.
|
|
|
|
|
*/
|
|
|
|
|
|
2019-09-10 02:19:35 +03:00
|
|
|
|
#include "internals.h"
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2017-09-05 20:58:03 +03:00
|
|
|
|
|
2019-08-29 23:48:51 +03:00
|
|
|
|
#include <winioctl.h>
|
|
|
|
|
|
2022-10-09 00:16:40 +03:00
|
|
|
|
#if !MDBX_WITHOUT_MSVC_CRT && defined(_DEBUG)
|
|
|
|
|
#include <crtdbg.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
2017-03-30 18:54:57 +03:00
|
|
|
|
static int waitstatus2errcode(DWORD result) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
switch (result) {
|
|
|
|
|
case WAIT_OBJECT_0:
|
2017-05-24 01:42:10 +03:00
|
|
|
|
return MDBX_SUCCESS;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
case WAIT_FAILED:
|
2021-05-08 10:59:15 +03:00
|
|
|
|
return (int)GetLastError();
|
2017-03-16 18:09:27 +03:00
|
|
|
|
case WAIT_ABANDONED:
|
|
|
|
|
return ERROR_ABANDONED_WAIT_0;
|
|
|
|
|
case WAIT_IO_COMPLETION:
|
|
|
|
|
return ERROR_USER_APC;
|
|
|
|
|
case WAIT_TIMEOUT:
|
|
|
|
|
return ERROR_TIMEOUT;
|
|
|
|
|
default:
|
|
|
|
|
return ERROR_UNHANDLED_ERROR;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-06-06 21:20:16 +03:00
|
|
|
|
|
|
|
|
|
/* Map a result from an NTAPI call to WIN32 error code. */
|
|
|
|
|
static int ntstatus2errcode(NTSTATUS status) {
|
|
|
|
|
DWORD dummy;
|
2017-07-02 09:07:57 +03:00
|
|
|
|
OVERLAPPED ov;
|
|
|
|
|
memset(&ov, 0, sizeof(ov));
|
|
|
|
|
ov.Internal = status;
|
2017-06-06 21:20:16 +03:00
|
|
|
|
return GetOverlappedResult(NULL, &ov, &dummy, FALSE) ? MDBX_SUCCESS
|
2021-05-08 10:59:15 +03:00
|
|
|
|
: (int)GetLastError();
|
2017-06-06 21:20:16 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* We use native NT APIs to setup the memory map, so that we can
|
|
|
|
|
* let the DB file grow incrementally instead of always preallocating
|
|
|
|
|
* the full size. These APIs are defined in <wdm.h> and <ntifs.h>
|
|
|
|
|
* but those headers are meant for driver-level development and
|
|
|
|
|
* conflict with the regular user-level headers, so we explicitly
|
|
|
|
|
* declare them here. Using these APIs also means we must link to
|
|
|
|
|
* ntdll.dll, which is not linked by default in user code. */
|
2017-06-21 01:34:56 +03:00
|
|
|
|
|
|
|
|
|
extern NTSTATUS NTAPI NtCreateSection(
|
|
|
|
|
OUT PHANDLE SectionHandle, IN ACCESS_MASK DesiredAccess,
|
|
|
|
|
IN OPTIONAL POBJECT_ATTRIBUTES ObjectAttributes,
|
|
|
|
|
IN OPTIONAL PLARGE_INTEGER MaximumSize, IN ULONG SectionPageProtection,
|
|
|
|
|
IN ULONG AllocationAttributes, IN OPTIONAL HANDLE FileHandle);
|
|
|
|
|
|
2017-12-25 18:31:59 +03:00
|
|
|
|
typedef struct _SECTION_BASIC_INFORMATION {
|
|
|
|
|
ULONG Unknown;
|
|
|
|
|
ULONG SectionAttributes;
|
|
|
|
|
LARGE_INTEGER SectionSize;
|
|
|
|
|
} SECTION_BASIC_INFORMATION, *PSECTION_BASIC_INFORMATION;
|
|
|
|
|
|
2017-06-21 01:34:56 +03:00
|
|
|
|
extern NTSTATUS NTAPI NtMapViewOfSection(
|
|
|
|
|
IN HANDLE SectionHandle, IN HANDLE ProcessHandle, IN OUT PVOID *BaseAddress,
|
|
|
|
|
IN ULONG_PTR ZeroBits, IN SIZE_T CommitSize,
|
|
|
|
|
IN OUT OPTIONAL PLARGE_INTEGER SectionOffset, IN OUT PSIZE_T ViewSize,
|
|
|
|
|
IN SECTION_INHERIT InheritDisposition, IN ULONG AllocationType,
|
|
|
|
|
IN ULONG Win32Protect);
|
|
|
|
|
|
|
|
|
|
extern NTSTATUS NTAPI NtUnmapViewOfSection(IN HANDLE ProcessHandle,
|
|
|
|
|
IN OPTIONAL PVOID BaseAddress);
|
|
|
|
|
|
|
|
|
|
extern NTSTATUS NTAPI NtClose(HANDLE Handle);
|
2017-06-06 21:20:16 +03:00
|
|
|
|
|
2017-06-21 01:34:56 +03:00
|
|
|
|
extern NTSTATUS NTAPI NtAllocateVirtualMemory(
|
2018-01-14 18:36:25 +03:00
|
|
|
|
IN HANDLE ProcessHandle, IN OUT PVOID *BaseAddress, IN ULONG_PTR ZeroBits,
|
|
|
|
|
IN OUT PSIZE_T RegionSize, IN ULONG AllocationType, IN ULONG Protect);
|
2017-06-21 01:34:56 +03:00
|
|
|
|
|
|
|
|
|
extern NTSTATUS NTAPI NtFreeVirtualMemory(IN HANDLE ProcessHandle,
|
|
|
|
|
IN PVOID *BaseAddress,
|
2018-01-14 18:36:25 +03:00
|
|
|
|
IN OUT PSIZE_T RegionSize,
|
2017-06-21 01:34:56 +03:00
|
|
|
|
IN ULONG FreeType);
|
2017-06-06 21:20:16 +03:00
|
|
|
|
|
2018-06-12 22:56:26 +03:00
|
|
|
|
#ifndef WOF_CURRENT_VERSION
|
|
|
|
|
typedef struct _WOF_EXTERNAL_INFO {
|
|
|
|
|
DWORD Version;
|
|
|
|
|
DWORD Provider;
|
|
|
|
|
} WOF_EXTERNAL_INFO, *PWOF_EXTERNAL_INFO;
|
|
|
|
|
#endif /* WOF_CURRENT_VERSION */
|
|
|
|
|
|
|
|
|
|
#ifndef WIM_PROVIDER_CURRENT_VERSION
|
|
|
|
|
#define WIM_PROVIDER_HASH_SIZE 20
|
|
|
|
|
|
|
|
|
|
typedef struct _WIM_PROVIDER_EXTERNAL_INFO {
|
|
|
|
|
DWORD Version;
|
|
|
|
|
DWORD Flags;
|
|
|
|
|
LARGE_INTEGER DataSourceId;
|
|
|
|
|
BYTE ResourceHash[WIM_PROVIDER_HASH_SIZE];
|
|
|
|
|
} WIM_PROVIDER_EXTERNAL_INFO, *PWIM_PROVIDER_EXTERNAL_INFO;
|
|
|
|
|
#endif /* WIM_PROVIDER_CURRENT_VERSION */
|
|
|
|
|
|
2017-07-21 15:49:05 +03:00
|
|
|
|
#ifndef FILE_PROVIDER_CURRENT_VERSION
|
2017-07-11 19:05:40 +03:00
|
|
|
|
typedef struct _FILE_PROVIDER_EXTERNAL_INFO_V1 {
|
|
|
|
|
ULONG Version;
|
|
|
|
|
ULONG Algorithm;
|
|
|
|
|
ULONG Flags;
|
|
|
|
|
} FILE_PROVIDER_EXTERNAL_INFO_V1, *PFILE_PROVIDER_EXTERNAL_INFO_V1;
|
2018-06-12 22:56:26 +03:00
|
|
|
|
#endif /* FILE_PROVIDER_CURRENT_VERSION */
|
2017-07-11 19:05:40 +03:00
|
|
|
|
|
|
|
|
|
#ifndef STATUS_OBJECT_NOT_EXTERNALLY_BACKED
|
|
|
|
|
#define STATUS_OBJECT_NOT_EXTERNALLY_BACKED ((NTSTATUS)0xC000046DL)
|
|
|
|
|
#endif
|
|
|
|
|
#ifndef STATUS_INVALID_DEVICE_REQUEST
|
|
|
|
|
#define STATUS_INVALID_DEVICE_REQUEST ((NTSTATUS)0xC0000010L)
|
|
|
|
|
#endif
|
2020-11-24 15:41:08 +03:00
|
|
|
|
#ifndef STATUS_NOT_SUPPORTED
|
|
|
|
|
#define STATUS_NOT_SUPPORTED ((NTSTATUS)0xC00000BBL)
|
|
|
|
|
#endif
|
2017-07-11 19:05:40 +03:00
|
|
|
|
|
2018-10-06 19:28:16 +03:00
|
|
|
|
#ifndef FILE_DEVICE_FILE_SYSTEM
|
|
|
|
|
#define FILE_DEVICE_FILE_SYSTEM 0x00000009
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifndef FSCTL_GET_EXTERNAL_BACKING
|
|
|
|
|
#define FSCTL_GET_EXTERNAL_BACKING \
|
|
|
|
|
CTL_CODE(FILE_DEVICE_FILE_SYSTEM, 196, METHOD_BUFFERED, FILE_ANY_ACCESS)
|
|
|
|
|
#endif
|
|
|
|
|
|
2021-05-08 10:12:41 +03:00
|
|
|
|
#ifndef ERROR_NOT_CAPABLE
|
|
|
|
|
#define ERROR_NOT_CAPABLE 775L
|
|
|
|
|
#endif
|
|
|
|
|
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#endif /* _WIN32 || _WIN64 */
|
|
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
2022-06-01 16:18:27 +03:00
|
|
|
|
#if defined(__ANDROID_API__)
|
|
|
|
|
__extern_C void __assert2(const char *file, int line, const char *function,
|
|
|
|
|
const char *msg) __noreturn;
|
|
|
|
|
#define __assert_fail(assertion, file, line, function) \
|
|
|
|
|
__assert2(file, line, function, assertion)
|
|
|
|
|
|
|
|
|
|
#elif defined(__UCLIBC__)
|
2020-09-27 12:50:13 +03:00
|
|
|
|
__extern_C void __assert(const char *, const char *, unsigned int, const char *)
|
|
|
|
|
#ifdef __THROW
|
|
|
|
|
__THROW
|
|
|
|
|
#else
|
|
|
|
|
__nothrow
|
|
|
|
|
#endif /* __THROW */
|
2020-10-03 11:35:26 +03:00
|
|
|
|
MDBX_NORETURN;
|
2020-09-27 12:50:13 +03:00
|
|
|
|
#define __assert_fail(assertion, file, line, function) \
|
|
|
|
|
__assert(assertion, file, line, function)
|
|
|
|
|
|
|
|
|
|
#elif _POSIX_C_SOURCE > 200212 && \
|
2019-03-04 14:41:29 +03:00
|
|
|
|
/* workaround for avoid musl libc wrong prototype */ ( \
|
|
|
|
|
defined(__GLIBC__) || defined(__GNU_LIBRARY__))
|
|
|
|
|
/* Prototype should match libc runtime. ISO POSIX (2003) & LSB 1.x-3.x */
|
2019-08-25 03:05:58 +03:00
|
|
|
|
__extern_C void __assert_fail(const char *assertion, const char *file,
|
|
|
|
|
unsigned line, const char *function)
|
|
|
|
|
#ifdef __THROW
|
|
|
|
|
__THROW
|
|
|
|
|
#else
|
|
|
|
|
__nothrow
|
|
|
|
|
#endif /* __THROW */
|
2020-09-14 16:40:46 +03:00
|
|
|
|
MDBX_NORETURN;
|
2019-08-25 03:05:58 +03:00
|
|
|
|
|
2019-08-13 02:07:10 +03:00
|
|
|
|
#elif defined(__APPLE__) || defined(__MACH__)
|
2019-08-25 03:05:58 +03:00
|
|
|
|
__extern_C void __assert_rtn(const char *function, const char *file, int line,
|
|
|
|
|
const char *assertion) /* __nothrow */
|
|
|
|
|
#ifdef __dead2
|
|
|
|
|
__dead2
|
|
|
|
|
#else
|
2020-09-14 16:40:46 +03:00
|
|
|
|
MDBX_NORETURN
|
2019-08-25 03:05:58 +03:00
|
|
|
|
#endif /* __dead2 */
|
|
|
|
|
#ifdef __disable_tail_calls
|
|
|
|
|
__disable_tail_calls
|
|
|
|
|
#endif /* __disable_tail_calls */
|
|
|
|
|
;
|
|
|
|
|
|
2019-08-13 02:07:10 +03:00
|
|
|
|
#define __assert_fail(assertion, file, line, function) \
|
|
|
|
|
__assert_rtn(function, file, line, assertion)
|
2019-11-09 10:50:43 +03:00
|
|
|
|
#elif defined(__sun) || defined(__SVR4) || defined(__svr4__)
|
|
|
|
|
__extern_C void __assert_c99(const char *assection, const char *file, int line,
|
2020-09-14 16:40:46 +03:00
|
|
|
|
const char *function) MDBX_NORETURN;
|
2019-11-09 10:50:43 +03:00
|
|
|
|
#define __assert_fail(assertion, file, line, function) \
|
|
|
|
|
__assert_c99(assertion, file, line, function)
|
2019-11-04 17:19:12 +03:00
|
|
|
|
#elif defined(__OpenBSD__)
|
|
|
|
|
__extern_C __dead void __assert2(const char *file, int line,
|
|
|
|
|
const char *function,
|
|
|
|
|
const char *assertion) /* __nothrow */;
|
|
|
|
|
#define __assert_fail(assertion, file, line, function) \
|
|
|
|
|
__assert2(file, line, function, assertion)
|
|
|
|
|
#elif defined(__NetBSD__)
|
|
|
|
|
__extern_C __dead void __assert13(const char *file, int line,
|
|
|
|
|
const char *function,
|
|
|
|
|
const char *assertion) /* __nothrow */;
|
|
|
|
|
#define __assert_fail(assertion, file, line, function) \
|
|
|
|
|
__assert13(file, line, function, assertion)
|
|
|
|
|
#elif defined(__FreeBSD__) || defined(__BSD__) || defined(__bsdi__) || \
|
2019-08-13 02:07:10 +03:00
|
|
|
|
defined(__DragonFly__)
|
2019-08-25 03:05:58 +03:00
|
|
|
|
__extern_C void __assert(const char *function, const char *file, int line,
|
|
|
|
|
const char *assertion) /* __nothrow */
|
|
|
|
|
#ifdef __dead2
|
|
|
|
|
__dead2
|
|
|
|
|
#else
|
2020-09-14 16:40:46 +03:00
|
|
|
|
MDBX_NORETURN
|
2019-08-25 03:05:58 +03:00
|
|
|
|
#endif /* __dead2 */
|
|
|
|
|
#ifdef __disable_tail_calls
|
|
|
|
|
__disable_tail_calls
|
|
|
|
|
#endif /* __disable_tail_calls */
|
|
|
|
|
;
|
2019-07-13 20:07:24 +03:00
|
|
|
|
#define __assert_fail(assertion, file, line, function) \
|
|
|
|
|
__assert(function, file, line, assertion)
|
|
|
|
|
|
|
|
|
|
#endif /* __assert_fail */
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
2021-07-03 00:41:08 +03:00
|
|
|
|
__cold void mdbx_assert_fail(const MDBX_env *env, const char *msg,
|
2022-06-01 18:55:18 +03:00
|
|
|
|
const char *func, unsigned line) {
|
2017-05-24 01:42:10 +03:00
|
|
|
|
#if MDBX_DEBUG
|
2022-09-30 19:40:18 +03:00
|
|
|
|
if (env && env->me_assert_func)
|
2017-03-16 18:09:27 +03:00
|
|
|
|
env->me_assert_func(env, msg, func, line);
|
|
|
|
|
#else
|
|
|
|
|
(void)env;
|
2022-09-30 19:40:18 +03:00
|
|
|
|
assert_fail(msg, func, line);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MDBX_NORETURN __cold void assert_fail(const char *msg, const char *func,
|
|
|
|
|
unsigned line) {
|
2017-05-24 01:42:10 +03:00
|
|
|
|
#endif /* MDBX_DEBUG */
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
if (debug_logger)
|
|
|
|
|
debug_log(MDBX_LOG_FATAL, func, line, "assert: %s\n", msg);
|
2018-10-14 01:11:14 +03:00
|
|
|
|
else {
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
char *message = nullptr;
|
2022-08-11 01:03:15 +03:00
|
|
|
|
const int num = osal_asprintf(&message, "\r\nMDBX-ASSERTION: %s, %s:%u",
|
2018-10-21 15:42:26 +03:00
|
|
|
|
msg, func ? func : "unknown", line);
|
2018-10-14 01:11:14 +03:00
|
|
|
|
if (num < 1 || !message)
|
|
|
|
|
message = "<troubles with assertion-message preparation>";
|
|
|
|
|
OutputDebugStringA(message);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
2018-10-14 01:11:14 +03:00
|
|
|
|
__assert_fail(msg, "mdbx", line, func);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2022-09-28 14:06:55 +03:00
|
|
|
|
while (1) {
|
2018-10-14 01:11:14 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2022-10-09 00:16:40 +03:00
|
|
|
|
#if !MDBX_WITHOUT_MSVC_CRT && defined(_DEBUG)
|
|
|
|
|
_CrtDbgReport(_CRT_ASSERT, func ? func : "unknown", line, "libmdbx",
|
|
|
|
|
"assertion failed: %s", msg);
|
|
|
|
|
#else
|
2022-09-28 14:06:55 +03:00
|
|
|
|
if (IsDebuggerPresent())
|
|
|
|
|
DebugBreak();
|
2022-10-09 00:16:40 +03:00
|
|
|
|
#endif
|
|
|
|
|
FatalExit(STATUS_ASSERTION_FAILURE);
|
2018-10-14 01:11:14 +03:00
|
|
|
|
#else
|
2022-09-28 14:06:55 +03:00
|
|
|
|
abort();
|
2018-10-14 01:11:14 +03:00
|
|
|
|
#endif
|
2022-09-28 14:06:55 +03:00
|
|
|
|
}
|
2017-03-16 18:09:27 +03:00
|
|
|
|
}
|
|
|
|
|
|
2020-08-22 20:19:46 +03:00
|
|
|
|
__cold void mdbx_panic(const char *fmt, ...) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
va_list ap;
|
|
|
|
|
va_start(ap, fmt);
|
2018-10-14 01:11:14 +03:00
|
|
|
|
|
|
|
|
|
char *message = nullptr;
|
2022-08-11 01:03:15 +03:00
|
|
|
|
const int num = osal_vasprintf(&message, fmt, ap);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
va_end(ap);
|
2019-10-25 22:56:25 +03:00
|
|
|
|
const char *const const_message =
|
2022-09-30 19:40:18 +03:00
|
|
|
|
unlikely(num < 1 || !message)
|
|
|
|
|
? "<troubles with panic-message preparation>"
|
|
|
|
|
: message;
|
|
|
|
|
|
|
|
|
|
if (debug_logger)
|
|
|
|
|
debug_log(MDBX_LOG_FATAL, "panic", 0, "%s", const_message);
|
2018-10-14 01:11:14 +03:00
|
|
|
|
|
2022-09-28 14:06:55 +03:00
|
|
|
|
while (1) {
|
2018-10-14 01:11:14 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2022-10-09 00:16:40 +03:00
|
|
|
|
#if !MDBX_WITHOUT_MSVC_CRT && defined(_DEBUG)
|
|
|
|
|
_CrtDbgReport(_CRT_ASSERT, "mdbx.c", 0, "libmdbx", "panic: %s",
|
|
|
|
|
const_message);
|
|
|
|
|
#else
|
2022-09-28 14:06:55 +03:00
|
|
|
|
OutputDebugStringA("\r\nMDBX-PANIC: ");
|
|
|
|
|
OutputDebugStringA(const_message);
|
|
|
|
|
if (IsDebuggerPresent())
|
|
|
|
|
DebugBreak();
|
2022-10-09 00:16:40 +03:00
|
|
|
|
#endif
|
2022-09-28 14:06:55 +03:00
|
|
|
|
FatalExit(ERROR_UNHANDLED_ERROR);
|
2018-10-14 01:11:14 +03:00
|
|
|
|
#else
|
2022-09-28 14:06:55 +03:00
|
|
|
|
__assert_fail(const_message, "mdbx", 0, "panic");
|
|
|
|
|
abort();
|
2018-10-14 01:11:14 +03:00
|
|
|
|
#endif
|
2022-09-28 14:06:55 +03:00
|
|
|
|
}
|
2017-03-16 18:09:27 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
#ifndef osal_vasprintf
|
|
|
|
|
MDBX_INTERNAL_FUNC int osal_vasprintf(char **strp, const char *fmt,
|
2019-08-31 17:10:04 +03:00
|
|
|
|
va_list ap) {
|
2018-10-05 15:41:27 +03:00
|
|
|
|
va_list ones;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
va_copy(ones, ap);
|
|
|
|
|
int needed = vsnprintf(nullptr, 0, fmt, ap);
|
|
|
|
|
|
|
|
|
|
if (unlikely(needed < 0 || needed >= INT_MAX)) {
|
2017-07-26 12:23:01 +03:00
|
|
|
|
*strp = nullptr;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
va_end(ones);
|
|
|
|
|
return needed;
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
*strp = osal_malloc(needed + 1);
|
2017-07-26 12:23:01 +03:00
|
|
|
|
if (unlikely(*strp == nullptr)) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
va_end(ones);
|
2018-10-05 15:41:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2017-04-24 19:37:01 +03:00
|
|
|
|
SetLastError(MDBX_ENOMEM);
|
2018-10-05 15:41:27 +03:00
|
|
|
|
#else
|
|
|
|
|
errno = MDBX_ENOMEM;
|
|
|
|
|
#endif
|
2017-04-24 19:37:01 +03:00
|
|
|
|
return -1;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int actual = vsnprintf(*strp, needed + 1, fmt, ones);
|
|
|
|
|
va_end(ones);
|
|
|
|
|
|
|
|
|
|
assert(actual == needed);
|
|
|
|
|
if (unlikely(actual < 0)) {
|
2022-08-11 01:03:15 +03:00
|
|
|
|
osal_free(*strp);
|
2017-07-26 12:23:01 +03:00
|
|
|
|
*strp = nullptr;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
}
|
|
|
|
|
return actual;
|
|
|
|
|
}
|
2022-08-11 01:03:15 +03:00
|
|
|
|
#endif /* osal_vasprintf */
|
2018-10-05 15:41:27 +03:00
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
#ifndef osal_asprintf
|
|
|
|
|
MDBX_INTERNAL_FUNC int osal_asprintf(char **strp, const char *fmt, ...) {
|
2018-10-05 15:41:27 +03:00
|
|
|
|
va_list ap;
|
|
|
|
|
va_start(ap, fmt);
|
2022-08-11 01:03:15 +03:00
|
|
|
|
int rc = osal_vasprintf(strp, fmt, ap);
|
2018-10-05 15:41:27 +03:00
|
|
|
|
va_end(ap);
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
2022-08-11 01:03:15 +03:00
|
|
|
|
#endif /* osal_asprintf */
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
#ifndef osal_memalign_alloc
|
|
|
|
|
MDBX_INTERNAL_FUNC int osal_memalign_alloc(size_t alignment, size_t bytes,
|
2019-08-31 17:10:04 +03:00
|
|
|
|
void **result) {
|
2020-02-02 20:49:51 +03:00
|
|
|
|
assert(is_powerof2(alignment) && alignment >= sizeof(void *));
|
2018-10-14 11:14:14 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
(void)alignment;
|
|
|
|
|
*result = VirtualAlloc(NULL, bytes, MEM_COMMIT | MEM_RESERVE, PAGE_READWRITE);
|
2017-05-24 01:42:10 +03:00
|
|
|
|
return *result ? MDBX_SUCCESS : MDBX_ENOMEM /* ERROR_OUTOFMEMORY */;
|
2019-07-13 20:07:42 +03:00
|
|
|
|
#elif defined(_ISOC11_SOURCE)
|
2020-04-07 03:19:35 +03:00
|
|
|
|
*result = aligned_alloc(alignment, ceil_powerof2(bytes, alignment));
|
2017-05-24 01:42:10 +03:00
|
|
|
|
return *result ? MDBX_SUCCESS : errno;
|
2020-04-13 23:36:56 +03:00
|
|
|
|
#elif _POSIX_VERSION >= 200112L && \
|
|
|
|
|
(!defined(__ANDROID_API__) || __ANDROID_API__ >= 17)
|
2017-07-26 12:23:01 +03:00
|
|
|
|
*result = nullptr;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
return posix_memalign(result, alignment, bytes);
|
2019-07-13 20:07:42 +03:00
|
|
|
|
#elif __GLIBC_PREREQ(2, 16) || __STDC_VERSION__ >= 201112L
|
|
|
|
|
*result = memalign(alignment, bytes);
|
|
|
|
|
return *result ? MDBX_SUCCESS : errno;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
|
|
|
|
#error FIXME
|
|
|
|
|
#endif
|
|
|
|
|
}
|
2022-08-11 01:03:15 +03:00
|
|
|
|
#endif /* osal_memalign_alloc */
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
#ifndef osal_memalign_free
|
|
|
|
|
MDBX_INTERNAL_FUNC void osal_memalign_free(void *ptr) {
|
2018-10-14 11:14:14 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
VirtualFree(ptr, 0, MEM_RELEASE);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
2022-08-11 01:03:15 +03:00
|
|
|
|
osal_free(ptr);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#endif
|
|
|
|
|
}
|
2022-08-11 01:03:15 +03:00
|
|
|
|
#endif /* osal_memalign_free */
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
#ifndef osal_strdup
|
|
|
|
|
char *osal_strdup(const char *str) {
|
2018-10-14 11:24:59 +03:00
|
|
|
|
if (!str)
|
|
|
|
|
return NULL;
|
|
|
|
|
size_t bytes = strlen(str) + 1;
|
2022-08-11 01:03:15 +03:00
|
|
|
|
char *dup = osal_malloc(bytes);
|
2018-10-14 11:24:59 +03:00
|
|
|
|
if (dup)
|
|
|
|
|
memcpy(dup, str, bytes);
|
|
|
|
|
return dup;
|
|
|
|
|
}
|
2022-08-11 01:03:15 +03:00
|
|
|
|
#endif /* osal_strdup */
|
2018-10-14 11:24:59 +03:00
|
|
|
|
|
2017-03-16 18:09:27 +03:00
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_condpair_init(osal_condpair_t *condpair) {
|
2020-05-21 22:15:30 +03:00
|
|
|
|
int rc;
|
2022-08-11 01:03:15 +03:00
|
|
|
|
memset(condpair, 0, sizeof(osal_condpair_t));
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2020-05-21 22:15:30 +03:00
|
|
|
|
if ((condpair->mutex = CreateMutexW(NULL, FALSE, NULL)) == NULL) {
|
2021-05-08 10:59:15 +03:00
|
|
|
|
rc = (int)GetLastError();
|
2020-05-21 22:15:30 +03:00
|
|
|
|
goto bailout_mutex;
|
2017-05-23 18:40:21 +03:00
|
|
|
|
}
|
2020-05-21 22:15:30 +03:00
|
|
|
|
if ((condpair->event[0] = CreateEventW(NULL, FALSE, FALSE, NULL)) == NULL) {
|
2021-05-08 10:59:15 +03:00
|
|
|
|
rc = (int)GetLastError();
|
2020-05-21 22:15:30 +03:00
|
|
|
|
goto bailout_event;
|
2017-05-23 18:40:21 +03:00
|
|
|
|
}
|
2020-05-21 22:15:30 +03:00
|
|
|
|
if ((condpair->event[1] = CreateEventW(NULL, FALSE, FALSE, NULL)) != NULL)
|
|
|
|
|
return MDBX_SUCCESS;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
2021-05-08 10:59:15 +03:00
|
|
|
|
rc = (int)GetLastError();
|
2020-05-21 22:15:30 +03:00
|
|
|
|
(void)CloseHandle(condpair->event[0]);
|
|
|
|
|
bailout_event:
|
|
|
|
|
(void)CloseHandle(condpair->mutex);
|
|
|
|
|
#else
|
|
|
|
|
rc = pthread_mutex_init(&condpair->mutex, NULL);
|
|
|
|
|
if (unlikely(rc != 0))
|
|
|
|
|
goto bailout_mutex;
|
|
|
|
|
rc = pthread_cond_init(&condpair->cond[0], NULL);
|
|
|
|
|
if (unlikely(rc != 0))
|
|
|
|
|
goto bailout_cond;
|
|
|
|
|
rc = pthread_cond_init(&condpair->cond[1], NULL);
|
|
|
|
|
if (likely(rc == 0))
|
|
|
|
|
return MDBX_SUCCESS;
|
|
|
|
|
|
|
|
|
|
(void)pthread_cond_destroy(&condpair->cond[0]);
|
|
|
|
|
bailout_cond:
|
|
|
|
|
(void)pthread_mutex_destroy(&condpair->mutex);
|
|
|
|
|
#endif
|
|
|
|
|
bailout_mutex:
|
2022-08-11 01:03:15 +03:00
|
|
|
|
memset(condpair, 0, sizeof(osal_condpair_t));
|
2020-05-21 22:15:30 +03:00
|
|
|
|
return rc;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_condpair_destroy(osal_condpair_t *condpair) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2021-05-08 10:59:15 +03:00
|
|
|
|
int rc = CloseHandle(condpair->mutex) ? MDBX_SUCCESS : (int)GetLastError();
|
|
|
|
|
rc = CloseHandle(condpair->event[0]) ? rc : (int)GetLastError();
|
|
|
|
|
rc = CloseHandle(condpair->event[1]) ? rc : (int)GetLastError();
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
2020-05-21 22:15:30 +03:00
|
|
|
|
int err, rc = pthread_mutex_destroy(&condpair->mutex);
|
|
|
|
|
rc = (err = pthread_cond_destroy(&condpair->cond[0])) ? err : rc;
|
|
|
|
|
rc = (err = pthread_cond_destroy(&condpair->cond[1])) ? err : rc;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#endif
|
2022-08-11 01:03:15 +03:00
|
|
|
|
memset(condpair, 0, sizeof(osal_condpair_t));
|
2017-05-23 18:40:21 +03:00
|
|
|
|
return rc;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_condpair_lock(osal_condpair_t *condpair) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2020-05-21 22:15:30 +03:00
|
|
|
|
DWORD code = WaitForSingleObject(condpair->mutex, INFINITE);
|
2017-05-23 18:40:21 +03:00
|
|
|
|
return waitstatus2errcode(code);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
2022-08-11 01:03:15 +03:00
|
|
|
|
return osal_pthread_mutex_lock(&condpair->mutex);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_condpair_unlock(osal_condpair_t *condpair) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2021-05-08 10:59:15 +03:00
|
|
|
|
return ReleaseMutex(condpair->mutex) ? MDBX_SUCCESS : (int)GetLastError();
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
2020-05-21 22:15:30 +03:00
|
|
|
|
return pthread_mutex_unlock(&condpair->mutex);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_condpair_signal(osal_condpair_t *condpair,
|
2020-05-21 22:15:30 +03:00
|
|
|
|
bool part) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2021-05-08 10:59:15 +03:00
|
|
|
|
return SetEvent(condpair->event[part]) ? MDBX_SUCCESS : (int)GetLastError();
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
2020-05-21 22:15:30 +03:00
|
|
|
|
return pthread_cond_signal(&condpair->cond[part]);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_condpair_wait(osal_condpair_t *condpair,
|
2020-05-21 22:15:30 +03:00
|
|
|
|
bool part) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2020-05-21 22:15:30 +03:00
|
|
|
|
DWORD code = SignalObjectAndWait(condpair->mutex, condpair->event[part],
|
|
|
|
|
INFINITE, FALSE);
|
2019-12-09 10:59:57 +03:00
|
|
|
|
if (code == WAIT_OBJECT_0) {
|
2020-05-21 22:15:30 +03:00
|
|
|
|
code = WaitForSingleObject(condpair->mutex, INFINITE);
|
2019-12-09 10:59:57 +03:00
|
|
|
|
if (code == WAIT_OBJECT_0)
|
2020-05-21 22:15:30 +03:00
|
|
|
|
return MDBX_SUCCESS;
|
2019-12-09 10:59:57 +03:00
|
|
|
|
}
|
2017-03-30 18:54:57 +03:00
|
|
|
|
return waitstatus2errcode(code);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
2020-05-21 22:15:30 +03:00
|
|
|
|
return pthread_cond_wait(&condpair->cond[part], &condpair->mutex);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_fastmutex_init(osal_fastmutex_t *fastmutex) {
|
2017-05-23 21:54:06 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
InitializeCriticalSection(fastmutex);
|
2017-05-24 01:42:10 +03:00
|
|
|
|
return MDBX_SUCCESS;
|
2017-05-23 21:54:06 +03:00
|
|
|
|
#else
|
|
|
|
|
return pthread_mutex_init(fastmutex, NULL);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_fastmutex_destroy(osal_fastmutex_t *fastmutex) {
|
2017-05-23 21:54:06 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
DeleteCriticalSection(fastmutex);
|
2017-05-24 01:42:10 +03:00
|
|
|
|
return MDBX_SUCCESS;
|
2017-05-23 21:54:06 +03:00
|
|
|
|
#else
|
|
|
|
|
return pthread_mutex_destroy(fastmutex);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_fastmutex_acquire(osal_fastmutex_t *fastmutex) {
|
2017-05-23 21:54:06 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2020-11-25 10:49:39 +03:00
|
|
|
|
__try {
|
|
|
|
|
EnterCriticalSection(fastmutex);
|
|
|
|
|
} __except (
|
|
|
|
|
(GetExceptionCode() ==
|
|
|
|
|
0xC0000194 /* STATUS_POSSIBLE_DEADLOCK / EXCEPTION_POSSIBLE_DEADLOCK */)
|
|
|
|
|
? EXCEPTION_EXECUTE_HANDLER
|
|
|
|
|
: EXCEPTION_CONTINUE_SEARCH) {
|
|
|
|
|
return ERROR_POSSIBLE_DEADLOCK;
|
|
|
|
|
}
|
2017-05-24 01:42:10 +03:00
|
|
|
|
return MDBX_SUCCESS;
|
2017-05-23 21:54:06 +03:00
|
|
|
|
#else
|
2022-08-11 01:03:15 +03:00
|
|
|
|
return osal_pthread_mutex_lock(fastmutex);
|
2017-05-23 21:54:06 +03:00
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_fastmutex_release(osal_fastmutex_t *fastmutex) {
|
2017-05-23 21:54:06 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
LeaveCriticalSection(fastmutex);
|
2017-05-24 01:42:10 +03:00
|
|
|
|
return MDBX_SUCCESS;
|
2017-05-23 21:54:06 +03:00
|
|
|
|
#else
|
|
|
|
|
return pthread_mutex_unlock(fastmutex);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
2022-08-09 16:12:24 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
|
|
|
|
|
#ifndef WC_ERR_INVALID_CHARS
|
|
|
|
|
static const DWORD WC_ERR_INVALID_CHARS =
|
|
|
|
|
(6 /* Windows Vista */ <= /* MajorVersion */ LOBYTE(LOWORD(GetVersion())))
|
|
|
|
|
? 0x00000080
|
|
|
|
|
: 0;
|
|
|
|
|
#endif /* WC_ERR_INVALID_CHARS */
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC size_t osal_mb2w(wchar_t *dst, size_t dst_n, const char *src,
|
2022-08-09 16:12:24 +03:00
|
|
|
|
size_t src_n) {
|
|
|
|
|
return MultiByteToWideChar(CP_THREAD_ACP, MB_ERR_INVALID_CHARS, src,
|
|
|
|
|
(int)src_n, dst, (int)dst_n);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif /* Windows */
|
|
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
2022-09-25 12:47:31 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
#define ior_alignment_mask (ior->pagesize - 1)
|
|
|
|
|
#define OSAL_IOV_MAX (4096 / sizeof(ior_sgv_element))
|
|
|
|
|
|
|
|
|
|
static void ior_put_event(osal_ioring_t *ior, HANDLE event) {
|
|
|
|
|
assert(event && event != INVALID_HANDLE_VALUE && event != ior);
|
|
|
|
|
assert(ior->event_stack < ior->allocated);
|
|
|
|
|
ior->event_pool[ior->event_stack] = event;
|
|
|
|
|
ior->event_stack += 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static HANDLE ior_get_event(osal_ioring_t *ior) {
|
|
|
|
|
assert(ior->event_stack <= ior->allocated);
|
|
|
|
|
if (ior->event_stack > 0) {
|
|
|
|
|
ior->event_stack -= 1;
|
|
|
|
|
assert(ior->event_pool[ior->event_stack] != 0);
|
|
|
|
|
return ior->event_pool[ior->event_stack];
|
|
|
|
|
}
|
|
|
|
|
return CreateEventW(nullptr, true, false, nullptr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void WINAPI ior_wocr(DWORD err, DWORD bytes, OVERLAPPED *ov) {
|
|
|
|
|
osal_ioring_t *ior = ov->hEvent;
|
|
|
|
|
ov->Internal = err;
|
|
|
|
|
ov->InternalHigh = bytes;
|
|
|
|
|
if (++ior->async_completed >= ior->async_waiting)
|
|
|
|
|
SetEvent(ior->async_done);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#elif MDBX_HAVE_PWRITEV
|
|
|
|
|
#if defined(_SC_IOV_MAX)
|
|
|
|
|
static size_t osal_iov_max;
|
|
|
|
|
#define OSAL_IOV_MAX osal_iov_max
|
|
|
|
|
#else
|
|
|
|
|
#define OSAL_IOV_MAX IOV_MAX
|
|
|
|
|
#endif
|
|
|
|
|
#else
|
|
|
|
|
#undef OSAL_IOV_MAX
|
|
|
|
|
#endif /* OSAL_IOV_MAX */
|
|
|
|
|
|
|
|
|
|
MDBX_INTERNAL_FUNC int osal_ioring_create(osal_ioring_t *ior,
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2022-09-29 16:18:10 +03:00
|
|
|
|
uint8_t flags,
|
2022-09-25 12:47:31 +03:00
|
|
|
|
#endif /* Windows */
|
|
|
|
|
mdbx_filehandle_t fd) {
|
|
|
|
|
memset(ior, 0, sizeof(osal_ioring_t));
|
|
|
|
|
ior->fd = fd;
|
|
|
|
|
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
ior->flags = flags;
|
|
|
|
|
const unsigned pagesize = (unsigned)osal_syspagesize();
|
|
|
|
|
ior->pagesize = pagesize;
|
|
|
|
|
ior->pagesize_ln2 = (uint8_t)log2n_powerof2(pagesize);
|
|
|
|
|
ior->async_done = ior_get_event(ior);
|
|
|
|
|
if (!ior->async_done)
|
|
|
|
|
return GetLastError();
|
|
|
|
|
#endif /* !Windows */
|
|
|
|
|
|
|
|
|
|
#if MDBX_HAVE_PWRITEV && defined(_SC_IOV_MAX)
|
|
|
|
|
if (!osal_iov_max)
|
|
|
|
|
osal_iov_max = sysconf(_SC_IOV_MAX);
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
ior->boundary = (char *)(ior->pool + ior->allocated);
|
|
|
|
|
return MDBX_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static __inline size_t ior_offset(const ior_item_t *item) {
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
return item->ov.Offset | (size_t)((sizeof(size_t) > sizeof(item->ov.Offset))
|
|
|
|
|
? (uint64_t)item->ov.OffsetHigh << 32
|
|
|
|
|
: 0);
|
|
|
|
|
#else
|
|
|
|
|
return item->offset;
|
|
|
|
|
#endif /* !Windows */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static __inline ior_item_t *ior_next(ior_item_t *item, size_t sgvcnt) {
|
|
|
|
|
#if defined(ior_sgv_element)
|
|
|
|
|
assert(sgvcnt > 0);
|
|
|
|
|
return (ior_item_t *)((char *)item + sizeof(ior_item_t) -
|
|
|
|
|
sizeof(ior_sgv_element) +
|
|
|
|
|
sizeof(ior_sgv_element) * sgvcnt);
|
|
|
|
|
#else
|
|
|
|
|
assert(sgvcnt == 1);
|
|
|
|
|
(void)sgvcnt;
|
|
|
|
|
return item + 1;
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MDBX_INTERNAL_FUNC int osal_ioring_add(osal_ioring_t *ior, const size_t offset,
|
|
|
|
|
void *data, const size_t bytes) {
|
|
|
|
|
|
|
|
|
|
assert(bytes && data);
|
|
|
|
|
assert(bytes % MIN_PAGESIZE == 0 && bytes <= MAX_WRITE);
|
|
|
|
|
assert(offset % MIN_PAGESIZE == 0 && offset + (uint64_t)bytes <= MAX_MAPSIZE);
|
|
|
|
|
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
const unsigned segments = (unsigned)(bytes >> ior->pagesize_ln2);
|
|
|
|
|
const bool use_gather =
|
2022-09-27 02:37:28 +03:00
|
|
|
|
(ior->flags & IOR_DIRECT) && ior->slots_left >= segments;
|
2022-09-25 12:47:31 +03:00
|
|
|
|
#endif /* Windows */
|
|
|
|
|
|
|
|
|
|
ior_item_t *item = ior->pool;
|
|
|
|
|
if (likely(ior->last)) {
|
|
|
|
|
item = ior->last;
|
|
|
|
|
if (unlikely(ior_offset(item) + ior_last_bytes(ior, item) == offset) &&
|
|
|
|
|
likely(ior_last_bytes(ior, item) + bytes <= MAX_WRITE)) {
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
if (use_gather &&
|
|
|
|
|
((bytes | (uintptr_t)data | ior->last_bytes |
|
|
|
|
|
(uintptr_t)(uint64_t)item->sgv[0].Buffer) &
|
|
|
|
|
ior_alignment_mask) == 0 &&
|
|
|
|
|
ior->last_sgvcnt + segments < OSAL_IOV_MAX) {
|
|
|
|
|
assert((item->single.iov_len & 1) == 0);
|
|
|
|
|
assert(item->sgv[ior->last_sgvcnt].Buffer == 0);
|
|
|
|
|
ior->last_bytes += bytes;
|
|
|
|
|
size_t i = 0;
|
|
|
|
|
do {
|
|
|
|
|
item->sgv[ior->last_sgvcnt + i].Buffer = PtrToPtr64(data);
|
|
|
|
|
data = (char *)data + ior->pagesize;
|
|
|
|
|
} while (++i < segments);
|
|
|
|
|
ior->slots_left -= segments;
|
|
|
|
|
item->sgv[ior->last_sgvcnt += segments].Buffer = 0;
|
|
|
|
|
assert((item->single.iov_len & 1) == 0);
|
|
|
|
|
return MDBX_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
const void *end =
|
|
|
|
|
(char *)(item->single.iov_base) + item->single.iov_len - 1;
|
|
|
|
|
if (unlikely(end == data)) {
|
|
|
|
|
assert((item->single.iov_len & 1) != 0);
|
|
|
|
|
item->single.iov_len += bytes;
|
|
|
|
|
return MDBX_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
#elif MDBX_HAVE_PWRITEV
|
|
|
|
|
assert((int)item->sgvcnt > 0);
|
|
|
|
|
const void *end = (char *)(item->sgv[item->sgvcnt - 1].iov_base) +
|
|
|
|
|
item->sgv[item->sgvcnt - 1].iov_len;
|
|
|
|
|
if (unlikely(end == data)) {
|
|
|
|
|
item->sgv[item->sgvcnt - 1].iov_len += bytes;
|
|
|
|
|
ior->last_bytes += bytes;
|
|
|
|
|
return MDBX_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
if (likely(item->sgvcnt < OSAL_IOV_MAX)) {
|
|
|
|
|
if (unlikely(ior->slots_left < 1))
|
|
|
|
|
return MDBX_RESULT_TRUE;
|
|
|
|
|
item->sgv[item->sgvcnt].iov_base = data;
|
|
|
|
|
item->sgv[item->sgvcnt].iov_len = bytes;
|
|
|
|
|
ior->last_bytes += bytes;
|
|
|
|
|
item->sgvcnt += 1;
|
|
|
|
|
ior->slots_left -= 1;
|
|
|
|
|
return MDBX_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
#else
|
|
|
|
|
const void *end = (char *)(item->single.iov_base) + item->single.iov_len;
|
|
|
|
|
if (unlikely(end == data)) {
|
|
|
|
|
item->single.iov_len += bytes;
|
|
|
|
|
return MDBX_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
item = ior_next(item, ior_last_sgvcnt(ior, item));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (unlikely(ior->slots_left < 1))
|
|
|
|
|
return MDBX_RESULT_TRUE;
|
|
|
|
|
|
|
|
|
|
unsigned slots_used = 1;
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
item->ov.Internal = item->ov.InternalHigh = 0;
|
|
|
|
|
item->ov.Offset = (DWORD)offset;
|
|
|
|
|
item->ov.OffsetHigh = HIGH_DWORD(offset);
|
|
|
|
|
item->ov.hEvent = 0;
|
|
|
|
|
if (!use_gather || ((bytes | (uintptr_t)(data)) & ior_alignment_mask) != 0 ||
|
|
|
|
|
segments > OSAL_IOV_MAX) {
|
|
|
|
|
/* WriteFile() */
|
|
|
|
|
item->single.iov_base = data;
|
|
|
|
|
item->single.iov_len = bytes + 1;
|
|
|
|
|
assert((item->single.iov_len & 1) != 0);
|
|
|
|
|
} else {
|
|
|
|
|
/* WriteFileGather() */
|
|
|
|
|
item->sgv[0].Buffer = PtrToPtr64(data);
|
|
|
|
|
for (size_t i = 1; i < segments; ++i) {
|
|
|
|
|
data = (char *)data + ior->pagesize;
|
|
|
|
|
item->sgv[slots_used].Buffer = PtrToPtr64(data);
|
|
|
|
|
}
|
|
|
|
|
item->sgv[slots_used].Buffer = 0;
|
|
|
|
|
assert((item->single.iov_len & 1) == 0);
|
|
|
|
|
slots_used = segments;
|
|
|
|
|
}
|
|
|
|
|
ior->last_bytes = bytes;
|
|
|
|
|
ior_last_sgvcnt(ior, item) = slots_used;
|
|
|
|
|
#elif MDBX_HAVE_PWRITEV
|
|
|
|
|
item->offset = offset;
|
|
|
|
|
item->sgv[0].iov_base = data;
|
|
|
|
|
item->sgv[0].iov_len = bytes;
|
|
|
|
|
ior->last_bytes = bytes;
|
|
|
|
|
ior_last_sgvcnt(ior, item) = slots_used;
|
|
|
|
|
#else
|
|
|
|
|
item->offset = offset;
|
|
|
|
|
item->single.iov_base = data;
|
|
|
|
|
item->single.iov_len = bytes;
|
|
|
|
|
#endif /* !Windows */
|
|
|
|
|
ior->slots_left -= slots_used;
|
|
|
|
|
ior->last = item;
|
|
|
|
|
return MDBX_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MDBX_INTERNAL_FUNC void osal_ioring_walk(
|
|
|
|
|
osal_ioring_t *ior, iov_ctx_t *ctx,
|
|
|
|
|
void (*callback)(iov_ctx_t *ctx, size_t offset, void *data, size_t bytes)) {
|
|
|
|
|
for (ior_item_t *item = ior->pool; item <= ior->last;) {
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
size_t offset = ior_offset(item);
|
|
|
|
|
char *data = item->single.iov_base;
|
|
|
|
|
size_t bytes = item->single.iov_len - 1;
|
|
|
|
|
size_t i = 1;
|
|
|
|
|
if (bytes & 1) {
|
|
|
|
|
data = Ptr64ToPtr(item->sgv[0].Buffer);
|
|
|
|
|
bytes = ior->pagesize;
|
|
|
|
|
while (item->sgv[i].Buffer) {
|
|
|
|
|
if (data + ior->pagesize != item->sgv[i].Buffer) {
|
|
|
|
|
callback(ctx, offset, data, bytes);
|
|
|
|
|
offset += bytes;
|
|
|
|
|
data = Ptr64ToPtr(item->sgv[i].Buffer);
|
|
|
|
|
bytes = 0;
|
|
|
|
|
}
|
|
|
|
|
bytes += ior->pagesize;
|
|
|
|
|
++i;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
assert(bytes < MAX_WRITE);
|
|
|
|
|
callback(ctx, offset, data, bytes);
|
|
|
|
|
#elif MDBX_HAVE_PWRITEV
|
|
|
|
|
assert(item->sgvcnt > 0);
|
|
|
|
|
size_t offset = item->offset;
|
|
|
|
|
size_t i = 0;
|
|
|
|
|
do {
|
|
|
|
|
callback(ctx, offset, item->sgv[i].iov_base, item->sgv[i].iov_len);
|
|
|
|
|
offset += item->sgv[i].iov_len;
|
|
|
|
|
} while (++i != item->sgvcnt);
|
|
|
|
|
#else
|
|
|
|
|
const size_t i = 1;
|
|
|
|
|
callback(ctx, item->offset, item->single.iov_base, item->single.iov_len);
|
|
|
|
|
#endif
|
|
|
|
|
item = ior_next(item, i);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MDBX_INTERNAL_FUNC osal_ioring_write_result_t
|
|
|
|
|
osal_ioring_write(osal_ioring_t *ior) {
|
|
|
|
|
osal_ioring_write_result_t r = {MDBX_SUCCESS, 0};
|
|
|
|
|
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
HANDLE *const end_wait_for =
|
|
|
|
|
ior->event_pool + ior->allocated +
|
|
|
|
|
/* был выделен один дополнительный элемент для async_done */ 1;
|
|
|
|
|
HANDLE *wait_for = end_wait_for;
|
|
|
|
|
LONG async_started = 0;
|
|
|
|
|
for (ior_item_t *item = ior->pool; item <= ior->last;) {
|
|
|
|
|
item->ov.Internal = STATUS_PENDING;
|
|
|
|
|
size_t i = 1, bytes = item->single.iov_len - 1;
|
|
|
|
|
r.wops += 1;
|
|
|
|
|
if (bytes & 1) {
|
|
|
|
|
bytes = ior->pagesize;
|
|
|
|
|
while (item->sgv[i].Buffer) {
|
|
|
|
|
bytes += ior->pagesize;
|
|
|
|
|
++i;
|
|
|
|
|
}
|
|
|
|
|
assert(bytes < MAX_WRITE);
|
|
|
|
|
item->ov.hEvent = ior_get_event(ior);
|
|
|
|
|
if (unlikely(!item->ov.hEvent)) {
|
|
|
|
|
bailout_geterr:
|
|
|
|
|
r.err = GetLastError();
|
|
|
|
|
bailout_rc:
|
|
|
|
|
assert(r.err != MDBX_SUCCESS);
|
|
|
|
|
CancelIo(ior->fd);
|
|
|
|
|
return r;
|
|
|
|
|
}
|
|
|
|
|
if (WriteFileGather(ior->fd, item->sgv, (DWORD)bytes, nullptr,
|
|
|
|
|
&item->ov)) {
|
|
|
|
|
assert(item->ov.Internal == 0 &&
|
|
|
|
|
WaitForSingleObject(item->ov.hEvent, 0) == WAIT_OBJECT_0);
|
|
|
|
|
ior_put_event(ior, item->ov.hEvent);
|
|
|
|
|
item->ov.hEvent = 0;
|
|
|
|
|
} else {
|
|
|
|
|
r.err = (int)GetLastError();
|
|
|
|
|
if (unlikely(r.err != ERROR_IO_PENDING)) {
|
|
|
|
|
ERROR("%s: fd %p, item %p (%zu), pgno %u, bytes %zu, offset %" PRId64
|
|
|
|
|
", err %d",
|
|
|
|
|
"WriteFileGather", ior->fd, item, item - ior->pool,
|
|
|
|
|
((MDBX_page *)item->single.iov_base)->mp_pgno, bytes,
|
|
|
|
|
item->ov.Offset + ((uint64_t)item->ov.OffsetHigh << 32), r.err);
|
|
|
|
|
goto bailout_rc;
|
|
|
|
|
}
|
|
|
|
|
assert(wait_for > ior->event_pool + ior->event_stack);
|
|
|
|
|
*--wait_for = item->ov.hEvent;
|
|
|
|
|
}
|
|
|
|
|
} else if (ior->flags & IOR_OVERLAPPED) {
|
|
|
|
|
assert(bytes < MAX_WRITE);
|
|
|
|
|
retry:
|
|
|
|
|
item->ov.hEvent = ior;
|
|
|
|
|
if (WriteFileEx(ior->fd, item->single.iov_base, (DWORD)bytes, &item->ov,
|
|
|
|
|
ior_wocr)) {
|
|
|
|
|
async_started += 1;
|
|
|
|
|
} else {
|
|
|
|
|
r.err = (int)GetLastError();
|
|
|
|
|
switch (r.err) {
|
|
|
|
|
default:
|
|
|
|
|
ERROR("%s: fd %p, item %p (%zu), pgno %u, bytes %zu, offset %" PRId64
|
|
|
|
|
", err %d",
|
|
|
|
|
"WriteFileEx", ior->fd, item, item - ior->pool,
|
|
|
|
|
((MDBX_page *)item->single.iov_base)->mp_pgno, bytes,
|
|
|
|
|
item->ov.Offset + ((uint64_t)item->ov.OffsetHigh << 32), r.err);
|
|
|
|
|
goto bailout_rc;
|
|
|
|
|
case ERROR_NOT_FOUND:
|
|
|
|
|
case ERROR_USER_MAPPED_FILE:
|
|
|
|
|
case ERROR_LOCK_VIOLATION:
|
|
|
|
|
WARNING(
|
|
|
|
|
"%s: fd %p, item %p (%zu), pgno %u, bytes %zu, offset %" PRId64
|
|
|
|
|
", err %d",
|
|
|
|
|
"WriteFileEx", ior->fd, item, item - ior->pool,
|
|
|
|
|
((MDBX_page *)item->single.iov_base)->mp_pgno, bytes,
|
|
|
|
|
item->ov.Offset + ((uint64_t)item->ov.OffsetHigh << 32), r.err);
|
|
|
|
|
SleepEx(0, true);
|
|
|
|
|
goto retry;
|
|
|
|
|
case ERROR_INVALID_USER_BUFFER:
|
|
|
|
|
case ERROR_NOT_ENOUGH_MEMORY:
|
|
|
|
|
if (SleepEx(0, true) == WAIT_IO_COMPLETION)
|
|
|
|
|
goto retry;
|
|
|
|
|
goto bailout_rc;
|
|
|
|
|
case ERROR_IO_PENDING:
|
|
|
|
|
async_started += 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
assert(bytes < MAX_WRITE);
|
|
|
|
|
DWORD written = 0;
|
|
|
|
|
if (!WriteFile(ior->fd, item->single.iov_base, (DWORD)bytes, &written,
|
|
|
|
|
&item->ov)) {
|
|
|
|
|
r.err = (int)GetLastError();
|
|
|
|
|
ERROR("%s: fd %p, item %p (%zu), pgno %u, bytes %zu, offset %" PRId64
|
|
|
|
|
", err %d",
|
|
|
|
|
"WriteFile", ior->fd, item, item - ior->pool,
|
|
|
|
|
((MDBX_page *)item->single.iov_base)->mp_pgno, bytes,
|
|
|
|
|
item->ov.Offset + ((uint64_t)item->ov.OffsetHigh << 32), r.err);
|
|
|
|
|
goto bailout_rc;
|
|
|
|
|
} else if (unlikely(written != bytes)) {
|
|
|
|
|
r.err = ERROR_WRITE_FAULT;
|
|
|
|
|
goto bailout_rc;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
item = ior_next(item, i);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
assert(ior->async_waiting > ior->async_completed &&
|
|
|
|
|
ior->async_waiting == INT_MAX);
|
|
|
|
|
ior->async_waiting = async_started;
|
|
|
|
|
if (async_started > ior->async_completed && end_wait_for == wait_for) {
|
|
|
|
|
assert(wait_for > ior->event_pool + ior->event_stack);
|
|
|
|
|
*--wait_for = ior->async_done;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const size_t pending_count = end_wait_for - wait_for;
|
|
|
|
|
if (pending_count) {
|
|
|
|
|
/* Ждем до MAXIMUM_WAIT_OBJECTS (64) последних хендлов, а после избирательно
|
|
|
|
|
* ждем посредством GetOverlappedResult(), если какие-то более ранние
|
|
|
|
|
* элементы еще не завершены. В целом, так получается меньше системных
|
|
|
|
|
* вызовов, т.е. меньше накладных расходов. Однако, не факт что эта экономия
|
|
|
|
|
* не будет перекрыта неэффективностью реализации
|
|
|
|
|
* WaitForMultipleObjectsEx(), но тогда это проблемы на стороне M$. */
|
|
|
|
|
DWORD madness;
|
|
|
|
|
do
|
|
|
|
|
madness = WaitForMultipleObjectsEx((pending_count < MAXIMUM_WAIT_OBJECTS)
|
|
|
|
|
? (DWORD)pending_count
|
|
|
|
|
: MAXIMUM_WAIT_OBJECTS,
|
|
|
|
|
wait_for, true,
|
|
|
|
|
/* сутки */ 86400000ul, true);
|
|
|
|
|
while (madness == WAIT_IO_COMPLETION);
|
|
|
|
|
STATIC_ASSERT(WAIT_OBJECT_0 == 0);
|
|
|
|
|
if (/* madness >= WAIT_OBJECT_0 && */
|
|
|
|
|
madness < WAIT_OBJECT_0 + MAXIMUM_WAIT_OBJECTS)
|
|
|
|
|
r.err = MDBX_SUCCESS;
|
|
|
|
|
else if (madness >= WAIT_ABANDONED_0 &&
|
|
|
|
|
madness < WAIT_ABANDONED_0 + MAXIMUM_WAIT_OBJECTS) {
|
|
|
|
|
r.err = ERROR_ABANDONED_WAIT_0;
|
|
|
|
|
goto bailout_rc;
|
|
|
|
|
} else if (madness == WAIT_TIMEOUT) {
|
|
|
|
|
r.err = WAIT_TIMEOUT;
|
|
|
|
|
goto bailout_rc;
|
|
|
|
|
} else {
|
|
|
|
|
r.err = /* madness == WAIT_FAILED */ MDBX_PROBLEM;
|
|
|
|
|
goto bailout_rc;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
assert(ior->async_waiting == ior->async_completed);
|
|
|
|
|
for (ior_item_t *item = ior->pool; item <= ior->last;) {
|
|
|
|
|
size_t i = 1, bytes = item->single.iov_len - 1;
|
|
|
|
|
if (bytes & 1) {
|
|
|
|
|
bytes = ior->pagesize;
|
|
|
|
|
while (item->sgv[i].Buffer) {
|
|
|
|
|
bytes += ior->pagesize;
|
|
|
|
|
++i;
|
|
|
|
|
}
|
|
|
|
|
if (!HasOverlappedIoCompleted(&item->ov)) {
|
|
|
|
|
DWORD written = 0;
|
|
|
|
|
if (unlikely(
|
|
|
|
|
!GetOverlappedResult(ior->fd, &item->ov, &written, true))) {
|
|
|
|
|
ERROR("%s: item %p (%zu), pgno %u, bytes %zu, offset %" PRId64
|
|
|
|
|
", err %d",
|
|
|
|
|
"GetOverlappedResult", item, item - ior->pool,
|
|
|
|
|
((MDBX_page *)item->single.iov_base)->mp_pgno, bytes,
|
|
|
|
|
item->ov.Offset + ((uint64_t)item->ov.OffsetHigh << 32),
|
|
|
|
|
GetLastError());
|
|
|
|
|
goto bailout_geterr;
|
|
|
|
|
}
|
|
|
|
|
assert(MDBX_SUCCESS == item->ov.Internal);
|
|
|
|
|
assert(written == item->ov.InternalHigh);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
assert(HasOverlappedIoCompleted(&item->ov));
|
|
|
|
|
}
|
|
|
|
|
assert(item->ov.Internal != ERROR_IO_PENDING);
|
|
|
|
|
if (unlikely(item->ov.Internal != MDBX_SUCCESS)) {
|
|
|
|
|
DWORD written = 0;
|
|
|
|
|
r.err = (int)item->ov.Internal;
|
|
|
|
|
if ((r.err & 0x80000000) &&
|
|
|
|
|
GetOverlappedResult(NULL, &item->ov, &written, true))
|
|
|
|
|
r.err = (int)GetLastError();
|
|
|
|
|
ERROR("%s: item %p (%zu), pgno %u, bytes %zu, offset %" PRId64
|
|
|
|
|
", err %d",
|
|
|
|
|
"Result", item, item - ior->pool,
|
|
|
|
|
((MDBX_page *)item->single.iov_base)->mp_pgno, bytes,
|
|
|
|
|
item->ov.Offset + ((uint64_t)item->ov.OffsetHigh << 32),
|
|
|
|
|
GetLastError());
|
|
|
|
|
goto bailout_rc;
|
|
|
|
|
}
|
|
|
|
|
if (unlikely(item->ov.InternalHigh != bytes)) {
|
|
|
|
|
r.err = ERROR_WRITE_FAULT;
|
|
|
|
|
goto bailout_rc;
|
|
|
|
|
}
|
|
|
|
|
item = ior_next(item, i);
|
|
|
|
|
}
|
|
|
|
|
assert(ior->async_waiting == ior->async_completed);
|
|
|
|
|
} else {
|
|
|
|
|
assert(r.err == MDBX_SUCCESS);
|
|
|
|
|
}
|
|
|
|
|
assert(ior->async_waiting == ior->async_completed);
|
|
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
STATIC_ASSERT_MSG(sizeof(off_t) >= sizeof(size_t),
|
|
|
|
|
"libmdbx requires 64-bit file I/O on 64-bit systems");
|
|
|
|
|
for (ior_item_t *item = ior->pool; item <= ior->last;) {
|
|
|
|
|
#if MDBX_HAVE_PWRITEV
|
|
|
|
|
assert(item->sgvcnt > 0);
|
|
|
|
|
if (item->sgvcnt == 1)
|
|
|
|
|
r.err = osal_pwrite(ior->fd, item->sgv[0].iov_base, item->sgv[0].iov_len,
|
|
|
|
|
item->offset);
|
|
|
|
|
else
|
|
|
|
|
r.err = osal_pwritev(ior->fd, item->sgv, item->sgvcnt, item->offset);
|
|
|
|
|
|
|
|
|
|
// TODO: io_uring_prep_write(sqe, fd, ...);
|
|
|
|
|
|
|
|
|
|
item = ior_next(item, item->sgvcnt);
|
|
|
|
|
#else
|
|
|
|
|
r.err = osal_pwrite(ior->fd, item->single.iov_base, item->single.iov_len,
|
|
|
|
|
item->offset);
|
|
|
|
|
item = ior_next(item, 1);
|
|
|
|
|
#endif
|
|
|
|
|
r.wops += 1;
|
|
|
|
|
if (unlikely(r.err != MDBX_SUCCESS))
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// TODO: io_uring_submit(&ring)
|
|
|
|
|
// TODO: err = io_uring_wait_cqe(&ring, &cqe);
|
|
|
|
|
// TODO: io_uring_cqe_seen(&ring, cqe);
|
|
|
|
|
|
|
|
|
|
#endif /* !Windows */
|
|
|
|
|
return r;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MDBX_INTERNAL_FUNC void osal_ioring_reset(osal_ioring_t *ior) {
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
if (ior->last) {
|
|
|
|
|
for (ior_item_t *item = ior->pool; item <= ior->last;) {
|
|
|
|
|
if (!HasOverlappedIoCompleted(&item->ov))
|
|
|
|
|
CancelIoEx(ior->fd, &item->ov);
|
|
|
|
|
if (item->ov.hEvent && item->ov.hEvent != ior)
|
|
|
|
|
ior_put_event(ior, item->ov.hEvent);
|
|
|
|
|
size_t i = 1;
|
|
|
|
|
if ((item->single.iov_len & 1) == 0)
|
|
|
|
|
while (item->sgv[i].Buffer)
|
|
|
|
|
++i;
|
|
|
|
|
item = ior_next(item, i);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
ior->async_waiting = INT_MAX;
|
|
|
|
|
ior->async_completed = 0;
|
|
|
|
|
ResetEvent(ior->async_done);
|
|
|
|
|
#endif /* !Windows */
|
|
|
|
|
ior->slots_left = ior->allocated;
|
|
|
|
|
ior->last = nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void ior_cleanup(osal_ioring_t *ior, const size_t since) {
|
|
|
|
|
osal_ioring_reset(ior);
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
for (size_t i = since; i < ior->event_stack; ++i)
|
|
|
|
|
CloseHandle(ior->event_pool[i]);
|
|
|
|
|
ior->event_stack = 0;
|
|
|
|
|
#else
|
|
|
|
|
(void)since;
|
|
|
|
|
#endif /* Windows */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MDBX_INTERNAL_FUNC int osal_ioring_resize(osal_ioring_t *ior, size_t items) {
|
|
|
|
|
assert(items > 0 && items < INT_MAX / sizeof(ior_item_t));
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
if (ior->state & IOR_STATE_LOCKED)
|
|
|
|
|
return MDBX_SUCCESS;
|
|
|
|
|
const bool useSetFileIoOverlappedRange = (ior->flags & IOR_OVERLAPPED) &&
|
|
|
|
|
mdbx_SetFileIoOverlappedRange &&
|
|
|
|
|
items > 7;
|
|
|
|
|
const size_t ceiling =
|
|
|
|
|
useSetFileIoOverlappedRange
|
|
|
|
|
? ((items < 65536 / 2 / sizeof(ior_item_t)) ? 65536 : 65536 * 4)
|
|
|
|
|
: 4096;
|
|
|
|
|
const size_t bytes = ceil_powerof2(sizeof(ior_item_t) * items, ceiling);
|
|
|
|
|
items = bytes / sizeof(ior_item_t);
|
|
|
|
|
#endif /* Windows */
|
|
|
|
|
|
|
|
|
|
if (items != ior->allocated) {
|
|
|
|
|
assert(items >= osal_ioring_used(ior));
|
|
|
|
|
if (items < ior->allocated)
|
|
|
|
|
ior_cleanup(ior, items);
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
void *ptr = osal_realloc(
|
|
|
|
|
ior->event_pool,
|
|
|
|
|
(items + /* extra for waiting the async_done */ 1) * sizeof(HANDLE));
|
|
|
|
|
if (unlikely(!ptr))
|
|
|
|
|
return MDBX_ENOMEM;
|
|
|
|
|
ior->event_pool = ptr;
|
|
|
|
|
|
|
|
|
|
int err = osal_memalign_alloc(ceiling, bytes, &ptr);
|
|
|
|
|
if (unlikely(err != MDBX_SUCCESS))
|
|
|
|
|
return err;
|
|
|
|
|
if (ior->pool) {
|
|
|
|
|
memcpy(ptr, ior->pool, ior->allocated * sizeof(ior_item_t));
|
|
|
|
|
osal_memalign_free(ior->pool);
|
|
|
|
|
}
|
|
|
|
|
#else
|
|
|
|
|
void *ptr = osal_realloc(ior->pool, sizeof(ior_item_t) * items);
|
|
|
|
|
if (unlikely(!ptr))
|
|
|
|
|
return MDBX_ENOMEM;
|
|
|
|
|
#endif
|
|
|
|
|
ior->pool = ptr;
|
|
|
|
|
|
|
|
|
|
if (items > ior->allocated)
|
|
|
|
|
memset(ior->pool + ior->allocated, 0,
|
|
|
|
|
sizeof(ior_item_t) * (items - ior->allocated));
|
|
|
|
|
ior->allocated = (unsigned)items;
|
|
|
|
|
ior->boundary = (char *)(ior->pool + ior->allocated);
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
if (useSetFileIoOverlappedRange) {
|
|
|
|
|
if (mdbx_SetFileIoOverlappedRange(ior->fd, ptr, (ULONG)bytes))
|
|
|
|
|
ior->state += IOR_STATE_LOCKED;
|
|
|
|
|
else
|
|
|
|
|
return GetLastError();
|
|
|
|
|
}
|
|
|
|
|
#endif /* Windows */
|
|
|
|
|
}
|
|
|
|
|
return MDBX_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MDBX_INTERNAL_FUNC void osal_ioring_destroy(osal_ioring_t *ior) {
|
|
|
|
|
if (ior->allocated)
|
|
|
|
|
ior_cleanup(ior, 0);
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
osal_memalign_free(ior->pool);
|
|
|
|
|
osal_free(ior->event_pool);
|
|
|
|
|
CloseHandle(ior->async_done);
|
|
|
|
|
#else
|
|
|
|
|
osal_free(ior->pool);
|
|
|
|
|
#endif
|
|
|
|
|
memset(ior, -1, sizeof(osal_ioring_t));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_removefile(const pathchar_t *pathname) {
|
2018-06-30 14:20:55 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2022-08-09 18:27:43 +03:00
|
|
|
|
return DeleteFileW(pathname) ? MDBX_SUCCESS : (int)GetLastError();
|
2018-06-30 14:20:55 +03:00
|
|
|
|
#else
|
|
|
|
|
return unlink(pathname) ? errno : MDBX_SUCCESS;
|
|
|
|
|
#endif
|
|
|
|
|
}
|
2019-08-31 17:10:04 +03:00
|
|
|
|
|
2020-12-05 22:03:18 +03:00
|
|
|
|
#if !(defined(_WIN32) || defined(_WIN64))
|
|
|
|
|
static bool is_valid_fd(int fd) { return !(isatty(fd) < 0 && errno == EBADF); }
|
|
|
|
|
#endif /*! Windows */
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_removedirectory(const pathchar_t *pathname) {
|
2020-10-09 22:43:14 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2022-08-09 18:27:43 +03:00
|
|
|
|
return RemoveDirectoryW(pathname) ? MDBX_SUCCESS : (int)GetLastError();
|
2020-10-09 22:43:14 +03:00
|
|
|
|
#else
|
|
|
|
|
return rmdir(pathname) ? errno : MDBX_SUCCESS;
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_openfile(const enum osal_openfile_purpose purpose,
|
2022-08-09 18:27:43 +03:00
|
|
|
|
const MDBX_env *env,
|
|
|
|
|
const pathchar_t *pathname,
|
2019-12-15 00:13:43 +03:00
|
|
|
|
mdbx_filehandle_t *fd,
|
2020-08-22 20:19:46 +03:00
|
|
|
|
mdbx_mode_t unix_mode_bits) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
*fd = INVALID_HANDLE_VALUE;
|
2019-12-15 00:13:43 +03:00
|
|
|
|
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2019-12-15 00:13:43 +03:00
|
|
|
|
DWORD CreationDisposition = unix_mode_bits ? OPEN_ALWAYS : OPEN_EXISTING;
|
|
|
|
|
DWORD FlagsAndAttributes =
|
|
|
|
|
FILE_FLAG_POSIX_SEMANTICS | FILE_ATTRIBUTE_NOT_CONTENT_INDEXED;
|
|
|
|
|
DWORD DesiredAccess = FILE_READ_ATTRIBUTES;
|
|
|
|
|
DWORD ShareMode = (env->me_flags & MDBX_EXCLUSIVE)
|
|
|
|
|
? 0
|
|
|
|
|
: (FILE_SHARE_READ | FILE_SHARE_WRITE);
|
|
|
|
|
|
|
|
|
|
switch (purpose) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
default:
|
|
|
|
|
return ERROR_INVALID_PARAMETER;
|
2019-12-15 00:13:43 +03:00
|
|
|
|
case MDBX_OPEN_LCK:
|
|
|
|
|
CreationDisposition = OPEN_ALWAYS;
|
|
|
|
|
DesiredAccess |= GENERIC_READ | GENERIC_WRITE;
|
|
|
|
|
FlagsAndAttributes |= FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_TEMPORARY;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
break;
|
2019-12-15 00:13:43 +03:00
|
|
|
|
case MDBX_OPEN_DXB_READ:
|
|
|
|
|
CreationDisposition = OPEN_EXISTING;
|
|
|
|
|
DesiredAccess |= GENERIC_READ;
|
|
|
|
|
ShareMode |= FILE_SHARE_READ;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
break;
|
2019-12-15 00:13:43 +03:00
|
|
|
|
case MDBX_OPEN_DXB_LAZY:
|
|
|
|
|
DesiredAccess |= GENERIC_READ | GENERIC_WRITE;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
break;
|
2022-09-27 02:37:28 +03:00
|
|
|
|
case MDBX_OPEN_DXB_OVERLAPPED_DIRECT:
|
|
|
|
|
FlagsAndAttributes |= FILE_FLAG_NO_BUFFERING;
|
|
|
|
|
/* fall through */
|
|
|
|
|
__fallthrough;
|
2022-09-25 12:47:31 +03:00
|
|
|
|
case MDBX_OPEN_DXB_OVERLAPPED:
|
|
|
|
|
FlagsAndAttributes |= FILE_FLAG_OVERLAPPED;
|
|
|
|
|
/* fall through */
|
|
|
|
|
__fallthrough;
|
2019-12-15 00:13:43 +03:00
|
|
|
|
case MDBX_OPEN_DXB_DSYNC:
|
2017-03-16 18:09:27 +03:00
|
|
|
|
CreationDisposition = OPEN_EXISTING;
|
2022-09-25 12:47:31 +03:00
|
|
|
|
DesiredAccess |= GENERIC_WRITE | GENERIC_READ;
|
2019-12-15 00:13:43 +03:00
|
|
|
|
FlagsAndAttributes |= FILE_FLAG_WRITE_THROUGH;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
break;
|
2019-12-15 00:13:43 +03:00
|
|
|
|
case MDBX_OPEN_COPY:
|
2017-03-16 18:09:27 +03:00
|
|
|
|
CreationDisposition = CREATE_NEW;
|
2019-12-15 00:13:43 +03:00
|
|
|
|
ShareMode = 0;
|
|
|
|
|
DesiredAccess |= GENERIC_WRITE;
|
2022-09-25 12:47:31 +03:00
|
|
|
|
if (env->me_psize >= env->me_os_psize)
|
|
|
|
|
FlagsAndAttributes |= FILE_FLAG_NO_BUFFERING;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
break;
|
2020-10-09 22:43:14 +03:00
|
|
|
|
case MDBX_OPEN_DELETE:
|
|
|
|
|
CreationDisposition = OPEN_EXISTING;
|
|
|
|
|
ShareMode |= FILE_SHARE_DELETE;
|
|
|
|
|
DesiredAccess =
|
|
|
|
|
FILE_READ_ATTRIBUTES | FILE_WRITE_ATTRIBUTES | DELETE | SYNCHRONIZE;
|
|
|
|
|
break;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-09 18:27:43 +03:00
|
|
|
|
*fd = CreateFileW(pathname, DesiredAccess, ShareMode, NULL,
|
2017-03-16 18:09:27 +03:00
|
|
|
|
CreationDisposition, FlagsAndAttributes, NULL);
|
2022-06-04 02:04:55 +03:00
|
|
|
|
if (*fd == INVALID_HANDLE_VALUE) {
|
|
|
|
|
int err = (int)GetLastError();
|
|
|
|
|
if (err == ERROR_ACCESS_DENIED && purpose == MDBX_OPEN_LCK) {
|
2022-08-09 18:27:43 +03:00
|
|
|
|
if (GetFileAttributesW(pathname) == INVALID_FILE_ATTRIBUTES &&
|
2022-06-04 02:04:55 +03:00
|
|
|
|
GetLastError() == ERROR_FILE_NOT_FOUND)
|
|
|
|
|
err = ERROR_FILE_NOT_FOUND;
|
|
|
|
|
}
|
|
|
|
|
return err;
|
|
|
|
|
}
|
2019-12-15 00:13:43 +03:00
|
|
|
|
|
|
|
|
|
BY_HANDLE_FILE_INFORMATION info;
|
|
|
|
|
if (!GetFileInformationByHandle(*fd, &info)) {
|
2021-05-08 10:59:15 +03:00
|
|
|
|
int err = (int)GetLastError();
|
2019-12-15 00:13:43 +03:00
|
|
|
|
CloseHandle(*fd);
|
|
|
|
|
*fd = INVALID_HANDLE_VALUE;
|
|
|
|
|
return err;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
}
|
2019-12-15 00:13:43 +03:00
|
|
|
|
const DWORD AttributesDiff =
|
|
|
|
|
(info.dwFileAttributes ^ FlagsAndAttributes) &
|
|
|
|
|
(FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_NOT_CONTENT_INDEXED |
|
|
|
|
|
FILE_ATTRIBUTE_TEMPORARY | FILE_ATTRIBUTE_COMPRESSED);
|
|
|
|
|
if (AttributesDiff)
|
2022-08-09 18:27:43 +03:00
|
|
|
|
(void)SetFileAttributesW(pathname, info.dwFileAttributes ^ AttributesDiff);
|
2019-12-15 00:13:43 +03:00
|
|
|
|
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
2019-12-15 00:13:43 +03:00
|
|
|
|
int flags = unix_mode_bits ? O_CREAT : 0;
|
|
|
|
|
switch (purpose) {
|
|
|
|
|
default:
|
|
|
|
|
return EINVAL;
|
|
|
|
|
case MDBX_OPEN_LCK:
|
|
|
|
|
flags |= O_RDWR;
|
|
|
|
|
break;
|
|
|
|
|
case MDBX_OPEN_DXB_READ:
|
|
|
|
|
flags = O_RDONLY;
|
|
|
|
|
break;
|
|
|
|
|
case MDBX_OPEN_DXB_LAZY:
|
|
|
|
|
flags |= O_RDWR;
|
|
|
|
|
break;
|
|
|
|
|
case MDBX_OPEN_COPY:
|
|
|
|
|
flags = O_CREAT | O_WRONLY | O_EXCL;
|
|
|
|
|
break;
|
|
|
|
|
case MDBX_OPEN_DXB_DSYNC:
|
|
|
|
|
flags |= O_WRONLY;
|
|
|
|
|
#if defined(O_DSYNC)
|
|
|
|
|
flags |= O_DSYNC;
|
|
|
|
|
#elif defined(O_SYNC)
|
|
|
|
|
flags |= O_SYNC;
|
|
|
|
|
#elif defined(O_FSYNC)
|
|
|
|
|
flags |= O_FSYNC;
|
|
|
|
|
#endif
|
|
|
|
|
break;
|
2020-10-09 22:43:14 +03:00
|
|
|
|
case MDBX_OPEN_DELETE:
|
|
|
|
|
flags = O_RDWR;
|
|
|
|
|
break;
|
2019-12-15 00:13:43 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const bool direct_nocache_for_copy =
|
|
|
|
|
env->me_psize >= env->me_os_psize && purpose == MDBX_OPEN_COPY;
|
|
|
|
|
if (direct_nocache_for_copy) {
|
|
|
|
|
#if defined(O_DIRECT)
|
|
|
|
|
flags |= O_DIRECT;
|
|
|
|
|
#endif /* O_DIRECT */
|
|
|
|
|
#if defined(O_NOCACHE)
|
|
|
|
|
flags |= O_NOCACHE;
|
|
|
|
|
#endif /* O_NOCACHE */
|
|
|
|
|
}
|
|
|
|
|
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#ifdef O_CLOEXEC
|
|
|
|
|
flags |= O_CLOEXEC;
|
2019-08-11 01:06:59 +03:00
|
|
|
|
#endif /* O_CLOEXEC */
|
2019-12-15 00:13:43 +03:00
|
|
|
|
|
2022-04-21 13:26:06 +03:00
|
|
|
|
/* Safeguard for todo4recovery://erased_by_github/libmdbx/issues/144 */
|
2020-12-05 22:03:18 +03:00
|
|
|
|
#if STDIN_FILENO == 0 && STDOUT_FILENO == 1 && STDERR_FILENO == 2
|
|
|
|
|
int stub_fd0 = -1, stub_fd1 = -1, stub_fd2 = -1;
|
|
|
|
|
static const char dev_null[] = "/dev/null";
|
2020-12-06 14:35:35 +03:00
|
|
|
|
if (!is_valid_fd(STDIN_FILENO)) {
|
2022-08-11 01:03:15 +03:00
|
|
|
|
WARNING("STD%s_FILENO/%d is invalid, open %s for temporary stub", "IN",
|
|
|
|
|
STDIN_FILENO, dev_null);
|
2020-12-05 22:03:18 +03:00
|
|
|
|
stub_fd0 = open(dev_null, O_RDONLY | O_NOCTTY);
|
2020-12-06 14:35:35 +03:00
|
|
|
|
}
|
|
|
|
|
if (!is_valid_fd(STDOUT_FILENO)) {
|
2022-08-11 01:03:15 +03:00
|
|
|
|
WARNING("STD%s_FILENO/%d is invalid, open %s for temporary stub", "OUT",
|
|
|
|
|
STDOUT_FILENO, dev_null);
|
2020-12-05 22:03:18 +03:00
|
|
|
|
stub_fd1 = open(dev_null, O_WRONLY | O_NOCTTY);
|
2020-12-06 14:35:35 +03:00
|
|
|
|
}
|
|
|
|
|
if (!is_valid_fd(STDERR_FILENO)) {
|
2022-08-11 01:03:15 +03:00
|
|
|
|
WARNING("STD%s_FILENO/%d is invalid, open %s for temporary stub", "ERR",
|
|
|
|
|
STDERR_FILENO, dev_null);
|
2020-12-05 22:03:18 +03:00
|
|
|
|
stub_fd2 = open(dev_null, O_WRONLY | O_NOCTTY);
|
2020-12-06 14:35:35 +03:00
|
|
|
|
}
|
2020-12-05 22:03:18 +03:00
|
|
|
|
#else
|
|
|
|
|
#error "Unexpected or unsupported UNIX or POSIX system"
|
|
|
|
|
#endif /* STDIN_FILENO == 0 && STDERR_FILENO == 2 */
|
|
|
|
|
|
2019-12-15 00:13:43 +03:00
|
|
|
|
*fd = open(pathname, flags, unix_mode_bits);
|
|
|
|
|
#if defined(O_DIRECT)
|
|
|
|
|
if (*fd < 0 && (flags & O_DIRECT) &&
|
|
|
|
|
(errno == EINVAL || errno == EAFNOSUPPORT)) {
|
|
|
|
|
flags &= ~(O_DIRECT | O_EXCL);
|
|
|
|
|
*fd = open(pathname, flags, unix_mode_bits);
|
|
|
|
|
}
|
|
|
|
|
#endif /* O_DIRECT */
|
2020-12-05 22:03:18 +03:00
|
|
|
|
|
2022-06-04 02:04:55 +03:00
|
|
|
|
if (*fd < 0 && errno == EACCES && purpose == MDBX_OPEN_LCK) {
|
|
|
|
|
struct stat unused;
|
|
|
|
|
if (stat(pathname, &unused) == 0 || errno != ENOENT)
|
|
|
|
|
errno = EACCES /* restore errno if file exists */;
|
|
|
|
|
}
|
|
|
|
|
|
2022-04-21 13:26:06 +03:00
|
|
|
|
/* Safeguard for todo4recovery://erased_by_github/libmdbx/issues/144 */
|
2020-12-05 22:03:18 +03:00
|
|
|
|
#if STDIN_FILENO == 0 && STDOUT_FILENO == 1 && STDERR_FILENO == 2
|
|
|
|
|
if (*fd == STDIN_FILENO) {
|
2022-08-11 01:03:15 +03:00
|
|
|
|
WARNING("Got STD%s_FILENO/%d, avoid using it by dup(fd)", "IN",
|
|
|
|
|
STDIN_FILENO);
|
2020-12-05 22:03:18 +03:00
|
|
|
|
assert(stub_fd0 == -1);
|
|
|
|
|
*fd = dup(stub_fd0 = *fd);
|
|
|
|
|
}
|
|
|
|
|
if (*fd == STDOUT_FILENO) {
|
2022-08-11 01:03:15 +03:00
|
|
|
|
WARNING("Got STD%s_FILENO/%d, avoid using it by dup(fd)", "OUT",
|
|
|
|
|
STDOUT_FILENO);
|
2020-12-05 22:03:18 +03:00
|
|
|
|
assert(stub_fd1 == -1);
|
|
|
|
|
*fd = dup(stub_fd1 = *fd);
|
|
|
|
|
}
|
|
|
|
|
if (*fd == STDERR_FILENO) {
|
2022-08-11 01:03:15 +03:00
|
|
|
|
WARNING("Got STD%s_FILENO/%d, avoid using it by dup(fd)", "ERR",
|
|
|
|
|
STDERR_FILENO);
|
2020-12-05 22:03:18 +03:00
|
|
|
|
assert(stub_fd2 == -1);
|
|
|
|
|
*fd = dup(stub_fd2 = *fd);
|
|
|
|
|
}
|
|
|
|
|
if (stub_fd0 != -1)
|
|
|
|
|
close(stub_fd0);
|
|
|
|
|
if (stub_fd1 != -1)
|
|
|
|
|
close(stub_fd1);
|
|
|
|
|
if (stub_fd2 != -1)
|
|
|
|
|
close(stub_fd2);
|
|
|
|
|
if (*fd >= STDIN_FILENO && *fd <= STDERR_FILENO) {
|
2022-08-11 01:03:15 +03:00
|
|
|
|
ERROR("Rejecting the use of a FD in the range "
|
|
|
|
|
"STDIN_FILENO/%d..STDERR_FILENO/%d to prevent database corruption",
|
|
|
|
|
STDIN_FILENO, STDERR_FILENO);
|
2020-12-05 22:03:18 +03:00
|
|
|
|
close(*fd);
|
|
|
|
|
return EBADF;
|
|
|
|
|
}
|
|
|
|
|
#else
|
|
|
|
|
#error "Unexpected or unsupported UNIX or POSIX system"
|
|
|
|
|
#endif /* STDIN_FILENO == 0 && STDERR_FILENO == 2 */
|
|
|
|
|
|
2017-03-16 18:09:27 +03:00
|
|
|
|
if (*fd < 0)
|
|
|
|
|
return errno;
|
2019-08-11 01:06:59 +03:00
|
|
|
|
|
|
|
|
|
#if defined(FD_CLOEXEC) && !defined(O_CLOEXEC)
|
2019-12-15 00:13:43 +03:00
|
|
|
|
const int fd_flags = fcntl(*fd, F_GETFD);
|
2019-08-11 01:06:59 +03:00
|
|
|
|
if (fd_flags != -1)
|
|
|
|
|
(void)fcntl(*fd, F_SETFD, fd_flags | FD_CLOEXEC);
|
|
|
|
|
#endif /* FD_CLOEXEC && !O_CLOEXEC */
|
|
|
|
|
|
2019-12-15 00:13:43 +03:00
|
|
|
|
if (direct_nocache_for_copy) {
|
|
|
|
|
#if defined(F_NOCACHE) && !defined(O_NOCACHE)
|
2019-08-13 02:07:10 +03:00
|
|
|
|
(void)fcntl(*fd, F_NOCACHE, 1);
|
|
|
|
|
#endif /* F_NOCACHE */
|
2019-08-11 01:06:59 +03:00
|
|
|
|
}
|
2019-12-15 00:13:43 +03:00
|
|
|
|
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#endif
|
2017-05-24 01:42:10 +03:00
|
|
|
|
return MDBX_SUCCESS;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_closefile(mdbx_filehandle_t fd) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2021-05-08 10:59:15 +03:00
|
|
|
|
return CloseHandle(fd) ? MDBX_SUCCESS : (int)GetLastError();
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
2020-12-05 22:03:18 +03:00
|
|
|
|
assert(fd > STDERR_FILENO);
|
2017-05-24 01:42:10 +03:00
|
|
|
|
return (close(fd) == 0) ? MDBX_SUCCESS : errno;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_pread(mdbx_filehandle_t fd, void *buf, size_t bytes,
|
2019-08-31 17:10:04 +03:00
|
|
|
|
uint64_t offset) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
if (bytes > MAX_WRITE)
|
2017-04-27 18:13:39 +03:00
|
|
|
|
return MDBX_EINVAL;
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2017-03-16 18:09:27 +03:00
|
|
|
|
OVERLAPPED ov;
|
|
|
|
|
ov.hEvent = 0;
|
|
|
|
|
ov.Offset = (DWORD)offset;
|
|
|
|
|
ov.OffsetHigh = HIGH_DWORD(offset);
|
|
|
|
|
|
2017-04-24 15:51:21 +03:00
|
|
|
|
DWORD read = 0;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
if (unlikely(!ReadFile(fd, buf, (DWORD)bytes, &read, &ov))) {
|
2021-05-08 10:59:15 +03:00
|
|
|
|
int rc = (int)GetLastError();
|
2017-05-24 01:42:10 +03:00
|
|
|
|
return (rc == MDBX_SUCCESS) ? /* paranoia */ ERROR_READ_FAULT : rc;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
}
|
|
|
|
|
#else
|
2017-05-26 17:11:48 +03:00
|
|
|
|
STATIC_ASSERT_MSG(sizeof(off_t) >= sizeof(size_t),
|
|
|
|
|
"libmdbx requires 64-bit file I/O on 64-bit systems");
|
2017-07-26 18:32:46 +03:00
|
|
|
|
intptr_t read = pread(fd, buf, bytes, offset);
|
2017-04-25 19:58:00 +03:00
|
|
|
|
if (read < 0) {
|
|
|
|
|
int rc = errno;
|
2017-05-24 01:42:10 +03:00
|
|
|
|
return (rc == MDBX_SUCCESS) ? /* paranoia */ MDBX_EIO : rc;
|
2017-04-25 19:58:00 +03:00
|
|
|
|
}
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#endif
|
2017-05-24 01:42:10 +03:00
|
|
|
|
return (bytes == (size_t)read) ? MDBX_SUCCESS : MDBX_ENODATA;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_pwrite(mdbx_filehandle_t fd, const void *buf,
|
2019-08-31 17:10:04 +03:00
|
|
|
|
size_t bytes, uint64_t offset) {
|
2019-06-28 11:15:42 +03:00
|
|
|
|
while (true) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2019-06-28 11:15:42 +03:00
|
|
|
|
OVERLAPPED ov;
|
|
|
|
|
ov.hEvent = 0;
|
|
|
|
|
ov.Offset = (DWORD)offset;
|
|
|
|
|
ov.OffsetHigh = HIGH_DWORD(offset);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
2019-06-28 11:15:42 +03:00
|
|
|
|
DWORD written;
|
2019-09-16 14:16:14 +03:00
|
|
|
|
if (unlikely(!WriteFile(
|
|
|
|
|
fd, buf, likely(bytes <= MAX_WRITE) ? (DWORD)bytes : MAX_WRITE,
|
|
|
|
|
&written, &ov)))
|
2021-05-08 10:59:15 +03:00
|
|
|
|
return (int)GetLastError();
|
2019-06-28 11:15:42 +03:00
|
|
|
|
if (likely(bytes == written))
|
|
|
|
|
return MDBX_SUCCESS;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
2017-05-26 17:11:48 +03:00
|
|
|
|
STATIC_ASSERT_MSG(sizeof(off_t) >= sizeof(size_t),
|
|
|
|
|
"libmdbx requires 64-bit file I/O on 64-bit systems");
|
2019-06-24 00:56:26 +03:00
|
|
|
|
const intptr_t written =
|
2019-09-16 14:16:14 +03:00
|
|
|
|
pwrite(fd, buf, likely(bytes <= MAX_WRITE) ? bytes : MAX_WRITE, offset);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
if (likely(bytes == (size_t)written))
|
2017-05-24 01:42:10 +03:00
|
|
|
|
return MDBX_SUCCESS;
|
2019-06-24 00:56:26 +03:00
|
|
|
|
if (written < 0) {
|
|
|
|
|
const int rc = errno;
|
|
|
|
|
if (rc != EINTR)
|
|
|
|
|
return rc;
|
2019-06-28 11:15:42 +03:00
|
|
|
|
continue;
|
2019-06-24 00:56:26 +03:00
|
|
|
|
}
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#endif
|
2019-06-28 11:15:42 +03:00
|
|
|
|
bytes -= written;
|
|
|
|
|
offset += written;
|
|
|
|
|
buf = (char *)buf + written;
|
|
|
|
|
}
|
2017-03-16 18:09:27 +03:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_write(mdbx_filehandle_t fd, const void *buf,
|
2019-09-16 14:16:14 +03:00
|
|
|
|
size_t bytes) {
|
|
|
|
|
while (true) {
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
DWORD written;
|
|
|
|
|
if (unlikely(!WriteFile(
|
|
|
|
|
fd, buf, likely(bytes <= MAX_WRITE) ? (DWORD)bytes : MAX_WRITE,
|
|
|
|
|
&written, nullptr)))
|
2021-05-08 10:59:15 +03:00
|
|
|
|
return (int)GetLastError();
|
2019-09-16 14:16:14 +03:00
|
|
|
|
if (likely(bytes == written))
|
|
|
|
|
return MDBX_SUCCESS;
|
|
|
|
|
#else
|
|
|
|
|
STATIC_ASSERT_MSG(sizeof(off_t) >= sizeof(size_t),
|
|
|
|
|
"libmdbx requires 64-bit file I/O on 64-bit systems");
|
|
|
|
|
const intptr_t written =
|
|
|
|
|
write(fd, buf, likely(bytes <= MAX_WRITE) ? bytes : MAX_WRITE);
|
|
|
|
|
if (likely(bytes == (size_t)written))
|
|
|
|
|
return MDBX_SUCCESS;
|
|
|
|
|
if (written < 0) {
|
|
|
|
|
const int rc = errno;
|
|
|
|
|
if (rc != EINTR)
|
|
|
|
|
return rc;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
bytes -= written;
|
|
|
|
|
buf = (char *)buf + written;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-09-29 16:18:10 +03:00
|
|
|
|
int osal_pwritev(mdbx_filehandle_t fd, struct iovec *iov, size_t sgvcnt,
|
2022-09-25 12:47:31 +03:00
|
|
|
|
uint64_t offset) {
|
|
|
|
|
size_t expected = 0;
|
2022-09-29 16:18:10 +03:00
|
|
|
|
for (size_t i = 0; i < sgvcnt; ++i)
|
2022-09-25 12:47:31 +03:00
|
|
|
|
expected += iov[i].iov_len;
|
|
|
|
|
#if !MDBX_HAVE_PWRITEV
|
2017-03-16 18:09:27 +03:00
|
|
|
|
size_t written = 0;
|
2022-09-29 16:18:10 +03:00
|
|
|
|
for (size_t i = 0; i < sgvcnt; ++i) {
|
2022-08-11 01:03:15 +03:00
|
|
|
|
int rc = osal_pwrite(fd, iov[i].iov_base, iov[i].iov_len, offset);
|
2017-05-24 01:42:10 +03:00
|
|
|
|
if (unlikely(rc != MDBX_SUCCESS))
|
2017-03-16 18:09:27 +03:00
|
|
|
|
return rc;
|
|
|
|
|
written += iov[i].iov_len;
|
|
|
|
|
offset += iov[i].iov_len;
|
|
|
|
|
}
|
2022-09-25 12:47:31 +03:00
|
|
|
|
return (expected == written) ? MDBX_SUCCESS
|
|
|
|
|
: MDBX_EIO /* ERROR_WRITE_FAULT */;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
|
|
|
|
int rc;
|
2017-07-26 18:32:46 +03:00
|
|
|
|
intptr_t written;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
do {
|
2017-05-26 17:11:48 +03:00
|
|
|
|
STATIC_ASSERT_MSG(sizeof(off_t) >= sizeof(size_t),
|
|
|
|
|
"libmdbx requires 64-bit file I/O on 64-bit systems");
|
2022-09-25 12:47:31 +03:00
|
|
|
|
written = pwritev(fd, iov, sgvcnt, offset);
|
|
|
|
|
if (likely(expected == (size_t)written))
|
2017-05-24 01:42:10 +03:00
|
|
|
|
return MDBX_SUCCESS;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
rc = errno;
|
|
|
|
|
} while (rc == EINTR);
|
2017-05-22 19:59:16 +03:00
|
|
|
|
return (written < 0) ? rc : MDBX_EIO /* Use which error code? */;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_fsync(mdbx_filehandle_t fd,
|
|
|
|
|
enum osal_syncmode_bits mode_bits) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2020-08-01 19:13:17 +03:00
|
|
|
|
if ((mode_bits & (MDBX_SYNC_DATA | MDBX_SYNC_IODQ)) && !FlushFileBuffers(fd))
|
2021-05-08 10:59:15 +03:00
|
|
|
|
return (int)GetLastError();
|
2020-08-01 19:13:17 +03:00
|
|
|
|
return MDBX_SUCCESS;
|
2019-03-04 13:34:38 +03:00
|
|
|
|
#else
|
2019-08-20 00:17:28 +03:00
|
|
|
|
|
2019-08-20 02:45:03 +03:00
|
|
|
|
#if defined(__APPLE__) && \
|
2019-08-20 15:04:32 +03:00
|
|
|
|
MDBX_OSX_SPEED_INSTEADOF_DURABILITY == MDBX_OSX_WANNA_DURABILITY
|
2019-08-20 00:17:28 +03:00
|
|
|
|
if (mode_bits & MDBX_SYNC_IODQ)
|
|
|
|
|
return likely(fcntl(fd, F_FULLFSYNC) != -1) ? MDBX_SUCCESS : errno;
|
|
|
|
|
#endif /* MacOS */
|
2020-08-01 19:13:17 +03:00
|
|
|
|
|
|
|
|
|
/* LY: This approach is always safe and without appreciable performance
|
|
|
|
|
* degradation, even on a kernel with fdatasync's bug.
|
|
|
|
|
*
|
|
|
|
|
* For more info about of a corresponding fdatasync() bug
|
|
|
|
|
* see http://www.spinics.net/lists/linux-ext4/msg33714.html */
|
|
|
|
|
while (1) {
|
|
|
|
|
switch (mode_bits & (MDBX_SYNC_DATA | MDBX_SYNC_SIZE)) {
|
|
|
|
|
case MDBX_SYNC_NONE:
|
|
|
|
|
return MDBX_SUCCESS /* nothing to do */;
|
2019-03-04 13:39:33 +03:00
|
|
|
|
#if defined(_POSIX_SYNCHRONIZED_IO) && _POSIX_SYNCHRONIZED_IO > 0
|
2020-08-01 19:13:17 +03:00
|
|
|
|
case MDBX_SYNC_DATA:
|
2019-03-04 13:53:05 +03:00
|
|
|
|
if (fdatasync(fd) == 0)
|
|
|
|
|
return MDBX_SUCCESS;
|
2020-08-01 19:13:17 +03:00
|
|
|
|
break /* error */;
|
|
|
|
|
#if defined(__linux__) || defined(__gnu_linux__)
|
|
|
|
|
case MDBX_SYNC_SIZE:
|
2022-09-18 13:21:38 +03:00
|
|
|
|
assert(linux_kernel_version >= 0x03060000);
|
|
|
|
|
return MDBX_SUCCESS;
|
2020-08-01 19:13:17 +03:00
|
|
|
|
#endif /* Linux */
|
|
|
|
|
#endif /* _POSIX_SYNCHRONIZED_IO > 0 */
|
|
|
|
|
default:
|
|
|
|
|
if (fsync(fd) == 0)
|
|
|
|
|
return MDBX_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int rc = errno;
|
|
|
|
|
if (rc != EINTR)
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
int osal_filesize(mdbx_filehandle_t fd, uint64_t *length) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
BY_HANDLE_FILE_INFORMATION info;
|
|
|
|
|
if (!GetFileInformationByHandle(fd, &info))
|
2021-05-08 10:59:15 +03:00
|
|
|
|
return (int)GetLastError();
|
2017-05-25 09:26:03 +03:00
|
|
|
|
*length = info.nFileSizeLow | (uint64_t)info.nFileSizeHigh << 32;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
|
|
|
|
struct stat st;
|
|
|
|
|
|
2017-05-26 17:11:48 +03:00
|
|
|
|
STATIC_ASSERT_MSG(sizeof(off_t) <= sizeof(uint64_t),
|
|
|
|
|
"libmdbx requires 64-bit file I/O on 64-bit systems");
|
2017-03-16 18:09:27 +03:00
|
|
|
|
if (fstat(fd, &st))
|
|
|
|
|
return errno;
|
|
|
|
|
|
|
|
|
|
*length = st.st_size;
|
|
|
|
|
#endif
|
2017-05-24 01:42:10 +03:00
|
|
|
|
return MDBX_SUCCESS;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_is_pipe(mdbx_filehandle_t fd) {
|
2019-09-16 14:16:14 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
switch (GetFileType(fd)) {
|
|
|
|
|
case FILE_TYPE_DISK:
|
|
|
|
|
return MDBX_RESULT_FALSE;
|
|
|
|
|
case FILE_TYPE_CHAR:
|
|
|
|
|
case FILE_TYPE_PIPE:
|
|
|
|
|
return MDBX_RESULT_TRUE;
|
|
|
|
|
default:
|
2021-05-08 10:59:15 +03:00
|
|
|
|
return (int)GetLastError();
|
2019-09-16 14:16:14 +03:00
|
|
|
|
}
|
|
|
|
|
#else
|
|
|
|
|
struct stat info;
|
|
|
|
|
if (fstat(fd, &info))
|
|
|
|
|
return errno;
|
|
|
|
|
switch (info.st_mode & S_IFMT) {
|
|
|
|
|
case S_IFBLK:
|
|
|
|
|
case S_IFREG:
|
|
|
|
|
return MDBX_RESULT_FALSE;
|
|
|
|
|
case S_IFCHR:
|
|
|
|
|
case S_IFIFO:
|
|
|
|
|
case S_IFSOCK:
|
|
|
|
|
return MDBX_RESULT_TRUE;
|
|
|
|
|
case S_IFDIR:
|
|
|
|
|
case S_IFLNK:
|
|
|
|
|
default:
|
|
|
|
|
return MDBX_INCOMPATIBLE;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_ftruncate(mdbx_filehandle_t fd, uint64_t length) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2018-12-26 19:53:03 +03:00
|
|
|
|
if (mdbx_SetFileInformationByHandle) {
|
|
|
|
|
FILE_END_OF_FILE_INFO EndOfFileInfo;
|
|
|
|
|
EndOfFileInfo.EndOfFile.QuadPart = length;
|
|
|
|
|
return mdbx_SetFileInformationByHandle(fd, FileEndOfFileInfo,
|
|
|
|
|
&EndOfFileInfo,
|
|
|
|
|
sizeof(FILE_END_OF_FILE_INFO))
|
|
|
|
|
? MDBX_SUCCESS
|
2021-05-08 10:59:15 +03:00
|
|
|
|
: (int)GetLastError();
|
2018-12-26 19:53:03 +03:00
|
|
|
|
} else {
|
|
|
|
|
LARGE_INTEGER li;
|
|
|
|
|
li.QuadPart = length;
|
|
|
|
|
return (SetFilePointerEx(fd, li, NULL, FILE_BEGIN) && SetEndOfFile(fd))
|
|
|
|
|
? MDBX_SUCCESS
|
2021-05-08 10:59:15 +03:00
|
|
|
|
: (int)GetLastError();
|
2018-12-26 19:53:03 +03:00
|
|
|
|
}
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
2017-05-26 17:11:48 +03:00
|
|
|
|
STATIC_ASSERT_MSG(sizeof(off_t) >= sizeof(size_t),
|
|
|
|
|
"libmdbx requires 64-bit file I/O on 64-bit systems");
|
2017-05-24 01:42:10 +03:00
|
|
|
|
return ftruncate(fd, length) == 0 ? MDBX_SUCCESS : errno;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_fseek(mdbx_filehandle_t fd, uint64_t pos) {
|
2018-11-04 18:57:08 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
LARGE_INTEGER li;
|
|
|
|
|
li.QuadPart = pos;
|
|
|
|
|
return SetFilePointerEx(fd, li, NULL, FILE_BEGIN) ? MDBX_SUCCESS
|
2021-05-08 10:59:15 +03:00
|
|
|
|
: (int)GetLastError();
|
2018-11-04 18:57:08 +03:00
|
|
|
|
#else
|
|
|
|
|
STATIC_ASSERT_MSG(sizeof(off_t) >= sizeof(size_t),
|
|
|
|
|
"libmdbx requires 64-bit file I/O on 64-bit systems");
|
|
|
|
|
return (lseek(fd, pos, SEEK_SET) < 0) ? errno : MDBX_SUCCESS;
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2017-03-16 18:09:27 +03:00
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
2019-08-31 17:10:04 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int
|
2022-08-11 01:03:15 +03:00
|
|
|
|
osal_thread_create(osal_thread_t *thread,
|
2019-08-31 17:10:04 +03:00
|
|
|
|
THREAD_RESULT(THREAD_CALL *start_routine)(void *),
|
|
|
|
|
void *arg) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
*thread = CreateThread(NULL, 0, start_routine, arg, 0, NULL);
|
2021-05-08 10:59:15 +03:00
|
|
|
|
return *thread ? MDBX_SUCCESS : (int)GetLastError();
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
|
|
|
|
return pthread_create(thread, NULL, start_routine, arg);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_thread_join(osal_thread_t thread) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
DWORD code = WaitForSingleObject(thread, INFINITE);
|
2017-03-30 18:54:57 +03:00
|
|
|
|
return waitstatus2errcode(code);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
|
|
|
|
void *unused_retval = &unused_retval;
|
|
|
|
|
return pthread_join(thread, &unused_retval);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
2022-09-25 12:47:31 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_msync(const osal_mmap_t *map, size_t offset,
|
2020-08-01 19:13:17 +03:00
|
|
|
|
size_t length,
|
2022-08-11 01:03:15 +03:00
|
|
|
|
enum osal_syncmode_bits mode_bits) {
|
2017-07-11 14:10:24 +03:00
|
|
|
|
uint8_t *ptr = (uint8_t *)map->address + offset;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2020-08-01 19:13:17 +03:00
|
|
|
|
if (!FlushViewOfFile(ptr, length))
|
2021-05-08 10:59:15 +03:00
|
|
|
|
return (int)GetLastError();
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
2019-10-11 23:33:03 +03:00
|
|
|
|
#if defined(__linux__) || defined(__gnu_linux__)
|
2022-09-18 13:21:38 +03:00
|
|
|
|
assert(linux_kernel_version > 0x02061300);
|
|
|
|
|
/* Since Linux 2.6.19, MS_ASYNC is in fact a no-op. The kernel properly
|
|
|
|
|
* tracks dirty pages and flushes them to storage as necessary. */
|
|
|
|
|
return MDBX_SUCCESS;
|
2019-08-19 21:31:05 +03:00
|
|
|
|
#endif /* Linux */
|
2020-08-01 19:13:17 +03:00
|
|
|
|
if (msync(ptr, length, (mode_bits & MDBX_SYNC_DATA) ? MS_SYNC : MS_ASYNC))
|
|
|
|
|
return errno;
|
|
|
|
|
mode_bits &= ~MDBX_SYNC_DATA;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#endif
|
2022-08-11 01:03:15 +03:00
|
|
|
|
return osal_fsync(map->fd, mode_bits);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_check_fs_rdonly(mdbx_filehandle_t handle,
|
2022-08-09 18:27:43 +03:00
|
|
|
|
const pathchar_t *pathname,
|
|
|
|
|
int err) {
|
2019-11-04 17:19:12 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
(void)pathname;
|
|
|
|
|
(void)err;
|
|
|
|
|
if (!mdbx_GetVolumeInformationByHandleW)
|
|
|
|
|
return MDBX_ENOSYS;
|
|
|
|
|
DWORD unused, flags;
|
|
|
|
|
if (!mdbx_GetVolumeInformationByHandleW(handle, nullptr, 0, nullptr, &unused,
|
|
|
|
|
&flags, nullptr, 0))
|
2021-05-08 10:59:15 +03:00
|
|
|
|
return (int)GetLastError();
|
2019-11-04 17:19:12 +03:00
|
|
|
|
if ((flags & FILE_READ_ONLY_VOLUME) == 0)
|
|
|
|
|
return MDBX_EACCESS;
|
|
|
|
|
#else
|
|
|
|
|
struct statvfs info;
|
|
|
|
|
if (err != MDBX_ENOFILE) {
|
2022-06-04 02:04:55 +03:00
|
|
|
|
if (statvfs(pathname, &info) == 0 && (info.f_flag & ST_RDONLY) == 0)
|
2019-11-04 17:19:12 +03:00
|
|
|
|
return err;
|
2022-06-04 02:04:55 +03:00
|
|
|
|
if (errno != MDBX_ENOFILE)
|
|
|
|
|
return errno;
|
2019-11-04 17:19:12 +03:00
|
|
|
|
}
|
|
|
|
|
if (fstatvfs(handle, &info))
|
|
|
|
|
return errno;
|
|
|
|
|
if ((info.f_flag & ST_RDONLY) == 0)
|
|
|
|
|
return (err == MDBX_ENOFILE) ? MDBX_EACCESS : err;
|
|
|
|
|
#endif /* !Windows */
|
|
|
|
|
return MDBX_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
static int osal_check_fs_local(mdbx_filehandle_t handle, int flags) {
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2020-02-18 20:29:35 +03:00
|
|
|
|
if (mdbx_RunningUnderWine() && !(flags & MDBX_EXCLUSIVE))
|
|
|
|
|
return ERROR_NOT_CAPABLE /* workaround for Wine */;
|
|
|
|
|
|
2018-06-12 22:52:57 +03:00
|
|
|
|
if (GetFileType(handle) != FILE_TYPE_DISK)
|
2017-07-11 19:05:40 +03:00
|
|
|
|
return ERROR_FILE_OFFLINE;
|
|
|
|
|
|
2018-06-12 22:56:26 +03:00
|
|
|
|
if (mdbx_GetFileInformationByHandleEx) {
|
|
|
|
|
FILE_REMOTE_PROTOCOL_INFO RemoteProtocolInfo;
|
|
|
|
|
if (mdbx_GetFileInformationByHandleEx(handle, FileRemoteProtocolInfo,
|
|
|
|
|
&RemoteProtocolInfo,
|
|
|
|
|
sizeof(RemoteProtocolInfo))) {
|
2018-06-13 21:33:49 +03:00
|
|
|
|
if ((RemoteProtocolInfo.Flags & REMOTE_PROTOCOL_INFO_FLAG_OFFLINE) &&
|
|
|
|
|
!(flags & MDBX_RDONLY))
|
2018-06-12 22:56:26 +03:00
|
|
|
|
return ERROR_FILE_OFFLINE;
|
2018-06-13 21:33:49 +03:00
|
|
|
|
if (!(RemoteProtocolInfo.Flags & REMOTE_PROTOCOL_INFO_FLAG_LOOPBACK) &&
|
|
|
|
|
!(flags & MDBX_EXCLUSIVE))
|
|
|
|
|
return ERROR_REMOTE_STORAGE_MEDIA_ERROR;
|
2018-06-12 22:56:26 +03:00
|
|
|
|
}
|
2017-07-11 19:05:40 +03:00
|
|
|
|
}
|
|
|
|
|
|
2018-06-12 22:56:26 +03:00
|
|
|
|
if (mdbx_NtFsControlFile) {
|
|
|
|
|
NTSTATUS rc;
|
|
|
|
|
struct {
|
|
|
|
|
WOF_EXTERNAL_INFO wof_info;
|
|
|
|
|
union {
|
|
|
|
|
WIM_PROVIDER_EXTERNAL_INFO wim_info;
|
|
|
|
|
FILE_PROVIDER_EXTERNAL_INFO_V1 file_info;
|
|
|
|
|
};
|
|
|
|
|
size_t reserved_for_microsoft_madness[42];
|
|
|
|
|
} GetExternalBacking_OutputBuffer;
|
|
|
|
|
IO_STATUS_BLOCK StatusBlock;
|
|
|
|
|
rc = mdbx_NtFsControlFile(handle, NULL, NULL, NULL, &StatusBlock,
|
|
|
|
|
FSCTL_GET_EXTERNAL_BACKING, NULL, 0,
|
|
|
|
|
&GetExternalBacking_OutputBuffer,
|
|
|
|
|
sizeof(GetExternalBacking_OutputBuffer));
|
2018-06-13 21:33:49 +03:00
|
|
|
|
if (NT_SUCCESS(rc)) {
|
|
|
|
|
if (!(flags & MDBX_EXCLUSIVE))
|
|
|
|
|
return ERROR_REMOTE_STORAGE_MEDIA_ERROR;
|
|
|
|
|
} else if (rc != STATUS_OBJECT_NOT_EXTERNALLY_BACKED &&
|
2020-11-24 15:41:08 +03:00
|
|
|
|
rc != STATUS_INVALID_DEVICE_REQUEST &&
|
|
|
|
|
rc != STATUS_NOT_SUPPORTED)
|
2018-06-13 21:33:49 +03:00
|
|
|
|
return ntstatus2errcode(rc);
|
2017-07-11 19:05:40 +03:00
|
|
|
|
}
|
|
|
|
|
|
2018-06-12 22:56:26 +03:00
|
|
|
|
if (mdbx_GetVolumeInformationByHandleW && mdbx_GetFinalPathNameByHandleW) {
|
2022-08-11 01:03:15 +03:00
|
|
|
|
WCHAR *PathBuffer = osal_malloc(sizeof(WCHAR) * INT16_MAX);
|
2018-10-13 19:07:45 +03:00
|
|
|
|
if (!PathBuffer)
|
|
|
|
|
return MDBX_ENOMEM;
|
|
|
|
|
|
|
|
|
|
int rc = MDBX_SUCCESS;
|
2018-06-12 22:56:26 +03:00
|
|
|
|
DWORD VolumeSerialNumber, FileSystemFlags;
|
|
|
|
|
if (!mdbx_GetVolumeInformationByHandleW(handle, PathBuffer, INT16_MAX,
|
|
|
|
|
&VolumeSerialNumber, NULL,
|
2018-10-13 19:07:45 +03:00
|
|
|
|
&FileSystemFlags, NULL, 0)) {
|
2021-05-08 10:59:15 +03:00
|
|
|
|
rc = (int)GetLastError();
|
2018-10-13 19:07:45 +03:00
|
|
|
|
goto bailout;
|
|
|
|
|
}
|
2017-07-11 19:05:40 +03:00
|
|
|
|
|
2018-06-12 22:56:26 +03:00
|
|
|
|
if ((flags & MDBX_RDONLY) == 0) {
|
2018-10-13 19:07:45 +03:00
|
|
|
|
if (FileSystemFlags &
|
|
|
|
|
(FILE_SEQUENTIAL_WRITE_ONCE | FILE_READ_ONLY_VOLUME |
|
|
|
|
|
FILE_VOLUME_IS_COMPRESSED)) {
|
|
|
|
|
rc = ERROR_REMOTE_STORAGE_MEDIA_ERROR;
|
|
|
|
|
goto bailout;
|
|
|
|
|
}
|
2017-07-11 19:05:40 +03:00
|
|
|
|
}
|
2018-06-12 22:56:26 +03:00
|
|
|
|
|
2022-06-04 02:08:24 +03:00
|
|
|
|
if (mdbx_GetFinalPathNameByHandleW(handle, PathBuffer, INT16_MAX,
|
|
|
|
|
FILE_NAME_NORMALIZED | VOLUME_NAME_NT)) {
|
|
|
|
|
if (_wcsnicmp(PathBuffer, L"\\Device\\Mup\\", 12) == 0) {
|
|
|
|
|
if (!(flags & MDBX_EXCLUSIVE)) {
|
|
|
|
|
rc = ERROR_REMOTE_STORAGE_MEDIA_ERROR;
|
|
|
|
|
goto bailout;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (F_ISSET(flags, MDBX_RDONLY | MDBX_EXCLUSIVE) &&
|
|
|
|
|
(FileSystemFlags & FILE_READ_ONLY_VOLUME)) {
|
|
|
|
|
/* without-LCK (exclusive readonly) mode for DB on a read-only volume */
|
2018-10-13 19:07:45 +03:00
|
|
|
|
goto bailout;
|
|
|
|
|
}
|
2018-06-12 22:56:26 +03:00
|
|
|
|
|
2022-06-04 02:08:24 +03:00
|
|
|
|
if (mdbx_GetFinalPathNameByHandleW(handle, PathBuffer, INT16_MAX,
|
|
|
|
|
FILE_NAME_NORMALIZED |
|
|
|
|
|
VOLUME_NAME_DOS)) {
|
2018-06-12 22:56:26 +03:00
|
|
|
|
UINT DriveType = GetDriveTypeW(PathBuffer);
|
|
|
|
|
if (DriveType == DRIVE_NO_ROOT_DIR &&
|
2018-10-14 11:25:26 +03:00
|
|
|
|
_wcsnicmp(PathBuffer, L"\\\\?\\", 4) == 0 &&
|
|
|
|
|
_wcsnicmp(PathBuffer + 5, L":\\", 2) == 0) {
|
2018-06-12 22:56:26 +03:00
|
|
|
|
PathBuffer[7] = 0;
|
|
|
|
|
DriveType = GetDriveTypeW(PathBuffer + 4);
|
|
|
|
|
}
|
|
|
|
|
switch (DriveType) {
|
|
|
|
|
case DRIVE_CDROM:
|
|
|
|
|
if (flags & MDBX_RDONLY)
|
|
|
|
|
break;
|
|
|
|
|
// fall through
|
|
|
|
|
case DRIVE_UNKNOWN:
|
|
|
|
|
case DRIVE_NO_ROOT_DIR:
|
|
|
|
|
case DRIVE_REMOTE:
|
|
|
|
|
default:
|
2018-06-13 21:33:49 +03:00
|
|
|
|
if (!(flags & MDBX_EXCLUSIVE))
|
2018-10-13 19:07:45 +03:00
|
|
|
|
rc = ERROR_REMOTE_STORAGE_MEDIA_ERROR;
|
2018-06-13 21:33:49 +03:00
|
|
|
|
// fall through
|
2018-06-12 22:56:26 +03:00
|
|
|
|
case DRIVE_REMOVABLE:
|
|
|
|
|
case DRIVE_FIXED:
|
|
|
|
|
case DRIVE_RAMDISK:
|
|
|
|
|
break;
|
|
|
|
|
}
|
2017-07-11 19:05:40 +03:00
|
|
|
|
}
|
2022-06-04 02:08:24 +03:00
|
|
|
|
|
2018-10-13 19:07:45 +03:00
|
|
|
|
bailout:
|
2022-08-11 01:03:15 +03:00
|
|
|
|
osal_free(PathBuffer);
|
2018-10-13 19:07:45 +03:00
|
|
|
|
return rc;
|
2017-07-11 19:05:40 +03:00
|
|
|
|
}
|
2020-04-13 23:36:56 +03:00
|
|
|
|
|
2018-06-12 22:52:57 +03:00
|
|
|
|
#else
|
2019-11-04 17:19:12 +03:00
|
|
|
|
|
|
|
|
|
struct statvfs statvfs_info;
|
|
|
|
|
if (fstatvfs(handle, &statvfs_info))
|
|
|
|
|
return errno;
|
|
|
|
|
#if defined(ST_LOCAL) || defined(ST_EXPORTED)
|
|
|
|
|
const unsigned long st_flags = statvfs_info.f_flag;
|
|
|
|
|
#endif /* ST_LOCAL || ST_EXPORTED */
|
|
|
|
|
|
|
|
|
|
#if defined(__NetBSD__)
|
|
|
|
|
const unsigned type = 0;
|
|
|
|
|
const char *const name = statvfs_info.f_fstypename;
|
|
|
|
|
const size_t name_len = VFS_NAMELEN;
|
2019-11-09 10:50:43 +03:00
|
|
|
|
#elif defined(_AIX) || defined(__OS400__)
|
|
|
|
|
const char *const name = statvfs_info.f_basetype;
|
|
|
|
|
const size_t name_len = sizeof(statvfs_info.f_basetype);
|
|
|
|
|
struct stat st;
|
|
|
|
|
if (fstat(handle, &st))
|
|
|
|
|
return errno;
|
|
|
|
|
const unsigned type = st.st_vfstype;
|
|
|
|
|
if ((st.st_flag & FS_REMOTE) != 0 && !(flags & MDBX_EXCLUSIVE))
|
|
|
|
|
return MDBX_EREMOTE;
|
|
|
|
|
#elif defined(FSTYPSZ) || defined(_FSTYPSZ)
|
2019-11-04 17:19:12 +03:00
|
|
|
|
const unsigned type = 0;
|
2019-11-09 10:50:43 +03:00
|
|
|
|
const char *const name = statvfs_info.f_basetype;
|
|
|
|
|
const size_t name_len = sizeof(statvfs_info.f_basetype);
|
2019-11-04 17:19:12 +03:00
|
|
|
|
#elif defined(__sun) || defined(__SVR4) || defined(__svr4__) || \
|
|
|
|
|
defined(ST_FSTYPSZ) || defined(_ST_FSTYPSZ)
|
|
|
|
|
const unsigned type = 0;
|
2019-11-09 10:50:43 +03:00
|
|
|
|
struct stat st;
|
|
|
|
|
if (fstat(handle, &st))
|
2019-11-04 17:19:12 +03:00
|
|
|
|
return errno;
|
2019-11-09 10:50:43 +03:00
|
|
|
|
const char *const name = st.st_fstype;
|
2019-11-04 17:19:12 +03:00
|
|
|
|
const size_t name_len = strlen(name);
|
|
|
|
|
#else
|
|
|
|
|
struct statfs statfs_info;
|
|
|
|
|
if (fstatfs(handle, &statfs_info))
|
|
|
|
|
return errno;
|
|
|
|
|
#if defined(__OpenBSD__)
|
|
|
|
|
const unsigned type = 0;
|
|
|
|
|
#else
|
|
|
|
|
const unsigned type = statfs_info.f_type;
|
|
|
|
|
#endif
|
|
|
|
|
#if defined(MNT_LOCAL) || defined(MNT_EXPORTED)
|
|
|
|
|
const unsigned long mnt_flags = statfs_info.f_flags;
|
|
|
|
|
#endif /* MNT_LOCAL || MNT_EXPORTED */
|
|
|
|
|
#if defined(__FreeBSD__) || defined(__NetBSD__) || defined(__OpenBSD__) || \
|
|
|
|
|
defined(__BSD__) || defined(__bsdi__) || defined(__DragonFly__) || \
|
|
|
|
|
defined(__APPLE__) || defined(__MACH__) || defined(MFSNAMELEN) || \
|
|
|
|
|
defined(MFSTYPENAMELEN) || defined(VFS_NAMELEN)
|
|
|
|
|
const char *const name = statfs_info.f_fstypename;
|
|
|
|
|
const size_t name_len = sizeof(statfs_info.f_fstypename);
|
2020-04-13 23:36:56 +03:00
|
|
|
|
#elif defined(__ANDROID_API__) && __ANDROID_API__ < 21
|
|
|
|
|
const char *const name = "";
|
|
|
|
|
const unsigned name_len = 0;
|
2019-11-04 17:19:12 +03:00
|
|
|
|
#else
|
2019-11-26 19:24:09 +03:00
|
|
|
|
|
|
|
|
|
const char *name = "";
|
|
|
|
|
unsigned name_len = 0;
|
|
|
|
|
|
|
|
|
|
struct stat st;
|
|
|
|
|
if (fstat(handle, &st))
|
|
|
|
|
return errno;
|
|
|
|
|
|
|
|
|
|
char pathbuf[PATH_MAX];
|
|
|
|
|
FILE *mounted = nullptr;
|
|
|
|
|
#if defined(__linux__) || defined(__gnu_linux__)
|
|
|
|
|
mounted = setmntent("/proc/mounts", "r");
|
|
|
|
|
#endif /* Linux */
|
|
|
|
|
if (!mounted)
|
|
|
|
|
mounted = setmntent("/etc/mtab", "r");
|
|
|
|
|
if (mounted) {
|
|
|
|
|
const struct mntent *ent;
|
2020-04-13 23:36:56 +03:00
|
|
|
|
#if defined(_BSD_SOURCE) || defined(_SVID_SOURCE) || defined(__BIONIC__) || \
|
2019-11-26 19:24:09 +03:00
|
|
|
|
(defined(_DEFAULT_SOURCE) && __GLIBC_PREREQ(2, 19))
|
|
|
|
|
struct mntent entbuf;
|
2019-12-03 15:38:02 +03:00
|
|
|
|
const bool should_copy = false;
|
2019-11-26 19:24:09 +03:00
|
|
|
|
while (nullptr !=
|
|
|
|
|
(ent = getmntent_r(mounted, &entbuf, pathbuf, sizeof(pathbuf))))
|
|
|
|
|
#else
|
2019-12-03 15:38:02 +03:00
|
|
|
|
const bool should_copy = true;
|
2020-04-13 23:35:08 +03:00
|
|
|
|
while (nullptr != (ent = getmntent(mounted)))
|
2019-11-04 17:19:12 +03:00
|
|
|
|
#endif
|
2019-11-26 19:24:09 +03:00
|
|
|
|
{
|
|
|
|
|
struct stat mnt;
|
|
|
|
|
if (!stat(ent->mnt_dir, &mnt) && mnt.st_dev == st.st_dev) {
|
2019-12-03 15:38:02 +03:00
|
|
|
|
if (should_copy) {
|
|
|
|
|
name =
|
|
|
|
|
strncpy(pathbuf, ent->mnt_fsname, name_len = sizeof(pathbuf) - 1);
|
|
|
|
|
pathbuf[name_len] = 0;
|
|
|
|
|
} else {
|
|
|
|
|
name = ent->mnt_fsname;
|
|
|
|
|
name_len = strlen(name);
|
|
|
|
|
}
|
2019-11-26 19:24:09 +03:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
endmntent(mounted);
|
|
|
|
|
}
|
2020-04-13 23:36:56 +03:00
|
|
|
|
#endif /* !xBSD && !Android/Bionic */
|
2019-11-04 17:19:12 +03:00
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
if (name_len) {
|
|
|
|
|
if (((name_len > 2 && strncasecmp("nfs", name, 3) == 0) ||
|
|
|
|
|
strncasecmp("cifs", name, name_len) == 0 ||
|
|
|
|
|
strncasecmp("ncpfs", name, name_len) == 0 ||
|
|
|
|
|
strncasecmp("smbfs", name, name_len) == 0 ||
|
2021-05-26 21:08:37 +03:00
|
|
|
|
strcasecmp("9P" /* WSL2 */, name) == 0 ||
|
2019-11-04 17:19:12 +03:00
|
|
|
|
((name_len > 3 && strncasecmp("fuse", name, 4) == 0) &&
|
|
|
|
|
strncasecmp("fuseblk", name, name_len) != 0)) &&
|
|
|
|
|
!(flags & MDBX_EXCLUSIVE))
|
|
|
|
|
return MDBX_EREMOTE;
|
|
|
|
|
if (strcasecmp("ftp", name) == 0 || strcasecmp("http", name) == 0 ||
|
2021-05-26 21:08:37 +03:00
|
|
|
|
strcasecmp("sshfs", name) == 0)
|
2019-11-04 17:19:12 +03:00
|
|
|
|
return MDBX_EREMOTE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifdef ST_LOCAL
|
|
|
|
|
if ((st_flags & ST_LOCAL) == 0 && !(flags & MDBX_EXCLUSIVE))
|
|
|
|
|
return MDBX_EREMOTE;
|
|
|
|
|
#elif defined(MNT_LOCAL)
|
|
|
|
|
if ((mnt_flags & MNT_LOCAL) == 0 && !(flags & MDBX_EXCLUSIVE))
|
|
|
|
|
return MDBX_EREMOTE;
|
|
|
|
|
#endif /* ST/MNT_LOCAL */
|
|
|
|
|
|
|
|
|
|
#ifdef ST_EXPORTED
|
2019-11-09 10:50:43 +03:00
|
|
|
|
if ((st_flags & ST_EXPORTED) != 0 && !(flags & MDBX_RDONLY))
|
2019-11-04 17:19:12 +03:00
|
|
|
|
return MDBX_EREMOTE;
|
|
|
|
|
#elif defined(MNT_EXPORTED)
|
2019-11-09 10:50:43 +03:00
|
|
|
|
if ((mnt_flags & MNT_EXPORTED) != 0 && !(flags & MDBX_RDONLY))
|
2019-11-04 17:19:12 +03:00
|
|
|
|
return MDBX_EREMOTE;
|
|
|
|
|
#endif /* ST/MNT_EXPORTED */
|
|
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
|
case 0xFF534D42 /* CIFS_MAGIC_NUMBER */:
|
|
|
|
|
case 0x6969 /* NFS_SUPER_MAGIC */:
|
|
|
|
|
case 0x564c /* NCP_SUPER_MAGIC */:
|
|
|
|
|
case 0x517B /* SMB_SUPER_MAGIC */:
|
|
|
|
|
#if defined(__digital__) || defined(__osf__) || defined(__osf)
|
|
|
|
|
case 0x0E /* Tru64 NFS */:
|
|
|
|
|
#endif
|
|
|
|
|
#ifdef ST_FST_NFS
|
|
|
|
|
case ST_FST_NFS:
|
|
|
|
|
#endif
|
|
|
|
|
if ((flags & MDBX_EXCLUSIVE) == 0)
|
|
|
|
|
return MDBX_EREMOTE;
|
|
|
|
|
case 0:
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
#endif /* Unix */
|
|
|
|
|
|
2018-06-12 22:52:57 +03:00
|
|
|
|
return MDBX_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-10 12:19:20 +03:00
|
|
|
|
static int check_mmap_limit(const size_t limit) {
|
|
|
|
|
const bool should_check =
|
|
|
|
|
#if defined(__SANITIZE_ADDRESS__)
|
|
|
|
|
true;
|
|
|
|
|
#else
|
|
|
|
|
RUNNING_ON_VALGRIND;
|
|
|
|
|
#endif /* __SANITIZE_ADDRESS__ */
|
|
|
|
|
|
|
|
|
|
if (should_check) {
|
|
|
|
|
intptr_t pagesize, total_ram_pages, avail_ram_pages;
|
|
|
|
|
int err =
|
|
|
|
|
mdbx_get_sysraminfo(&pagesize, &total_ram_pages, &avail_ram_pages);
|
|
|
|
|
if (unlikely(err != MDBX_SUCCESS))
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
|
const int log2page = log2n_powerof2(pagesize);
|
|
|
|
|
if ((limit >> (log2page + 7)) > (size_t)total_ram_pages ||
|
|
|
|
|
(limit >> (log2page + 6)) > (size_t)avail_ram_pages) {
|
2022-08-11 01:03:15 +03:00
|
|
|
|
ERROR("%s (%zu pages) is too large for available (%zu pages) or total "
|
|
|
|
|
"(%zu pages) system RAM",
|
|
|
|
|
"database upper size limit", limit >> log2page, avail_ram_pages,
|
|
|
|
|
total_ram_pages);
|
2021-07-10 12:19:20 +03:00
|
|
|
|
return MDBX_TOO_LARGE;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return MDBX_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_mmap(const int flags, osal_mmap_t *map,
|
2019-10-28 18:01:02 +03:00
|
|
|
|
const size_t size, const size_t limit,
|
2019-11-08 09:06:44 +03:00
|
|
|
|
const unsigned options) {
|
2018-06-12 22:52:57 +03:00
|
|
|
|
assert(size <= limit);
|
2019-10-28 18:01:02 +03:00
|
|
|
|
map->limit = 0;
|
2018-06-12 22:52:57 +03:00
|
|
|
|
map->current = 0;
|
|
|
|
|
map->address = nullptr;
|
2021-07-08 13:18:51 +03:00
|
|
|
|
map->filesize = 0;
|
2019-10-28 18:01:02 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
map->section = NULL;
|
|
|
|
|
#endif /* Windows */
|
2018-06-12 22:52:57 +03:00
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
int err = osal_check_fs_local(map->fd, flags);
|
2019-10-28 18:01:02 +03:00
|
|
|
|
if (unlikely(err != MDBX_SUCCESS))
|
|
|
|
|
return err;
|
2017-07-11 19:05:40 +03:00
|
|
|
|
|
2021-07-10 12:19:20 +03:00
|
|
|
|
err = check_mmap_limit(limit);
|
|
|
|
|
if (unlikely(err != MDBX_SUCCESS))
|
|
|
|
|
return err;
|
|
|
|
|
|
2019-11-08 09:06:44 +03:00
|
|
|
|
if ((flags & MDBX_RDONLY) == 0 && (options & MMAP_OPTION_TRUNCATE) != 0) {
|
2022-08-11 01:03:15 +03:00
|
|
|
|
err = osal_ftruncate(map->fd, size);
|
2019-10-28 18:01:02 +03:00
|
|
|
|
if (err != MDBX_SUCCESS)
|
|
|
|
|
return err;
|
|
|
|
|
map->filesize = size;
|
2021-07-08 13:18:51 +03:00
|
|
|
|
#if !(defined(_WIN32) || defined(_WIN64))
|
2019-10-28 18:01:02 +03:00
|
|
|
|
map->current = size;
|
2021-07-08 13:18:51 +03:00
|
|
|
|
#endif /* !Windows */
|
2019-10-28 18:01:02 +03:00
|
|
|
|
} else {
|
2022-08-11 01:03:15 +03:00
|
|
|
|
err = osal_filesize(map->fd, &map->filesize);
|
2019-10-28 18:01:02 +03:00
|
|
|
|
if (err != MDBX_SUCCESS)
|
|
|
|
|
return err;
|
2021-07-08 13:18:51 +03:00
|
|
|
|
#if !(defined(_WIN32) || defined(_WIN64))
|
|
|
|
|
map->current = (map->filesize > limit) ? limit : (size_t)map->filesize;
|
|
|
|
|
#endif /* !Windows */
|
2018-01-07 14:26:20 +03:00
|
|
|
|
}
|
|
|
|
|
|
2019-10-28 18:01:02 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2018-01-07 14:26:20 +03:00
|
|
|
|
LARGE_INTEGER SectionSize;
|
2018-01-07 14:37:38 +03:00
|
|
|
|
SectionSize.QuadPart = size;
|
2019-10-28 18:01:02 +03:00
|
|
|
|
err = NtCreateSection(
|
2017-06-21 01:34:56 +03:00
|
|
|
|
&map->section,
|
2018-06-18 21:29:12 +03:00
|
|
|
|
/* DesiredAccess */
|
2018-06-20 13:44:23 +03:00
|
|
|
|
(flags & MDBX_WRITEMAP)
|
2017-12-25 18:31:59 +03:00
|
|
|
|
? SECTION_QUERY | SECTION_MAP_READ | SECTION_EXTEND_SIZE |
|
|
|
|
|
SECTION_MAP_WRITE
|
|
|
|
|
: SECTION_QUERY | SECTION_MAP_READ | SECTION_EXTEND_SIZE,
|
2018-01-07 14:26:20 +03:00
|
|
|
|
/* ObjectAttributes */ NULL, /* MaximumSize (InitialSize) */ &SectionSize,
|
2018-06-18 21:29:12 +03:00
|
|
|
|
/* SectionPageProtection */
|
2018-06-20 13:44:23 +03:00
|
|
|
|
(flags & MDBX_RDONLY) ? PAGE_READONLY : PAGE_READWRITE,
|
2017-06-21 01:34:56 +03:00
|
|
|
|
/* AllocationAttributes */ SEC_RESERVE, map->fd);
|
2019-10-28 18:01:02 +03:00
|
|
|
|
if (!NT_SUCCESS(err))
|
|
|
|
|
return ntstatus2errcode(err);
|
2017-06-21 01:34:56 +03:00
|
|
|
|
|
2021-12-02 18:17:41 +03:00
|
|
|
|
SIZE_T ViewSize = (flags & MDBX_RDONLY) ? 0
|
|
|
|
|
: mdbx_RunningUnderWine() ? size
|
|
|
|
|
: limit;
|
2019-10-28 18:01:02 +03:00
|
|
|
|
err = NtMapViewOfSection(
|
2017-06-21 01:34:56 +03:00
|
|
|
|
map->section, GetCurrentProcess(), &map->address,
|
|
|
|
|
/* ZeroBits */ 0,
|
2018-01-07 14:26:20 +03:00
|
|
|
|
/* CommitSize */ 0,
|
2017-06-21 01:34:56 +03:00
|
|
|
|
/* SectionOffset */ NULL, &ViewSize,
|
|
|
|
|
/* InheritDisposition */ ViewUnmap,
|
|
|
|
|
/* AllocationType */ (flags & MDBX_RDONLY) ? 0 : MEM_RESERVE,
|
2018-06-18 21:29:12 +03:00
|
|
|
|
/* Win32Protect */
|
2018-06-20 13:44:23 +03:00
|
|
|
|
(flags & MDBX_WRITEMAP) ? PAGE_READWRITE : PAGE_READONLY);
|
2019-10-28 18:01:02 +03:00
|
|
|
|
if (!NT_SUCCESS(err)) {
|
2017-06-21 01:34:56 +03:00
|
|
|
|
NtClose(map->section);
|
|
|
|
|
map->section = 0;
|
2017-07-26 12:16:47 +03:00
|
|
|
|
map->address = nullptr;
|
2019-10-28 18:01:02 +03:00
|
|
|
|
return ntstatus2errcode(err);
|
2017-06-21 01:34:56 +03:00
|
|
|
|
}
|
|
|
|
|
assert(map->address != MAP_FAILED);
|
2017-07-28 13:48:10 +03:00
|
|
|
|
|
2018-01-07 14:26:20 +03:00
|
|
|
|
map->current = (size_t)SectionSize.QuadPart;
|
2019-10-28 18:01:02 +03:00
|
|
|
|
map->limit = ViewSize;
|
|
|
|
|
|
2021-02-26 04:10:56 +03:00
|
|
|
|
#else /* Windows */
|
2019-10-28 18:01:02 +03:00
|
|
|
|
|
2019-11-13 15:56:35 +03:00
|
|
|
|
#ifndef MAP_TRYFIXED
|
|
|
|
|
#define MAP_TRYFIXED 0
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifndef MAP_HASSEMAPHORE
|
|
|
|
|
#define MAP_HASSEMAPHORE 0
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifndef MAP_CONCEAL
|
|
|
|
|
#define MAP_CONCEAL 0
|
2019-11-08 09:06:44 +03:00
|
|
|
|
#endif
|
2019-11-13 15:56:35 +03:00
|
|
|
|
|
|
|
|
|
#ifndef MAP_NOSYNC
|
|
|
|
|
#define MAP_NOSYNC 0
|
2021-03-17 01:13:47 +03:00
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifndef MAP_FIXED_NOREPLACE
|
|
|
|
|
#define MAP_FIXED_NOREPLACE 0
|
2021-03-17 01:14:52 +03:00
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifndef MAP_NORESERVE
|
|
|
|
|
#define MAP_NORESERVE 0
|
2019-11-08 09:06:44 +03:00
|
|
|
|
#endif
|
2019-11-13 15:56:35 +03:00
|
|
|
|
|
|
|
|
|
map->address = mmap(
|
|
|
|
|
NULL, limit, (flags & MDBX_WRITEMAP) ? PROT_READ | PROT_WRITE : PROT_READ,
|
2021-03-17 01:14:52 +03:00
|
|
|
|
MAP_SHARED | MAP_FILE | MAP_NORESERVE |
|
2019-11-13 15:56:35 +03:00
|
|
|
|
(F_ISSET(flags, MDBX_UTTERLY_NOSYNC) ? MAP_NOSYNC : 0) |
|
|
|
|
|
((options & MMAP_OPTION_SEMAPHORE) ? MAP_HASSEMAPHORE | MAP_NOSYNC
|
|
|
|
|
: MAP_CONCEAL),
|
2019-11-08 09:06:44 +03:00
|
|
|
|
map->fd, 0);
|
2019-10-27 00:34:49 +03:00
|
|
|
|
|
|
|
|
|
if (unlikely(map->address == MAP_FAILED)) {
|
2019-10-28 18:01:02 +03:00
|
|
|
|
map->limit = 0;
|
|
|
|
|
map->current = 0;
|
2019-10-27 00:34:49 +03:00
|
|
|
|
map->address = nullptr;
|
|
|
|
|
return errno;
|
2017-07-12 21:13:17 +03:00
|
|
|
|
}
|
2019-10-28 18:01:02 +03:00
|
|
|
|
map->limit = limit;
|
2019-10-27 00:34:49 +03:00
|
|
|
|
|
2021-02-26 04:10:56 +03:00
|
|
|
|
#if MDBX_ENABLE_MADVISE
|
2019-10-27 00:34:49 +03:00
|
|
|
|
#ifdef MADV_DONTFORK
|
2019-10-28 18:01:02 +03:00
|
|
|
|
if (unlikely(madvise(map->address, map->limit, MADV_DONTFORK) != 0))
|
2019-10-27 00:34:49 +03:00
|
|
|
|
return errno;
|
2021-02-26 04:10:56 +03:00
|
|
|
|
#endif /* MADV_DONTFORK */
|
2019-10-27 00:34:49 +03:00
|
|
|
|
#ifdef MADV_NOHUGEPAGE
|
2019-10-28 18:01:02 +03:00
|
|
|
|
(void)madvise(map->address, map->limit, MADV_NOHUGEPAGE);
|
2021-02-26 04:10:56 +03:00
|
|
|
|
#endif /* MADV_NOHUGEPAGE */
|
|
|
|
|
#endif /* MDBX_ENABLE_MADVISE */
|
2019-10-27 00:34:49 +03:00
|
|
|
|
|
2021-02-26 04:10:56 +03:00
|
|
|
|
#endif /* ! Windows */
|
2019-10-28 18:01:02 +03:00
|
|
|
|
|
2019-12-29 01:19:31 +03:00
|
|
|
|
VALGRIND_MAKE_MEM_DEFINED(map->address, map->current);
|
2021-07-16 14:59:37 +03:00
|
|
|
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(map->address, map->current);
|
2019-10-28 18:01:02 +03:00
|
|
|
|
return MDBX_SUCCESS;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_munmap(osal_mmap_t *map) {
|
2019-12-29 01:19:31 +03:00
|
|
|
|
VALGRIND_MAKE_MEM_NOACCESS(map->address, map->current);
|
2020-04-10 10:23:43 -07:00
|
|
|
|
/* Unpoisoning is required for ASAN to avoid false-positive diagnostic
|
2020-09-21 23:51:47 -04:00
|
|
|
|
* when this memory will re-used by malloc or another mmapping.
|
2022-04-21 13:26:06 +03:00
|
|
|
|
* See todo4recovery://erased_by_github/libmdbx/pull/93#issuecomment-613687203
|
|
|
|
|
*/
|
2021-07-16 14:59:37 +03:00
|
|
|
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(map->address,
|
|
|
|
|
(map->filesize && map->filesize < map->limit)
|
|
|
|
|
? map->filesize
|
|
|
|
|
: map->limit);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2017-06-21 01:34:56 +03:00
|
|
|
|
if (map->section)
|
|
|
|
|
NtClose(map->section);
|
|
|
|
|
NTSTATUS rc = NtUnmapViewOfSection(GetCurrentProcess(), map->address);
|
2017-07-12 21:13:17 +03:00
|
|
|
|
if (!NT_SUCCESS(rc))
|
|
|
|
|
ntstatus2errcode(rc);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
2019-10-28 18:01:02 +03:00
|
|
|
|
if (unlikely(munmap(map->address, map->limit)))
|
2017-07-12 21:13:17 +03:00
|
|
|
|
return errno;
|
2021-02-26 04:10:56 +03:00
|
|
|
|
#endif /* ! Windows */
|
2019-10-28 18:01:02 +03:00
|
|
|
|
|
|
|
|
|
map->limit = 0;
|
|
|
|
|
map->current = 0;
|
|
|
|
|
map->address = nullptr;
|
2017-07-12 21:13:17 +03:00
|
|
|
|
return MDBX_SUCCESS;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC int osal_mresize(const int flags, osal_mmap_t *map,
|
2021-06-10 02:57:32 +03:00
|
|
|
|
size_t size, size_t limit) {
|
2018-01-07 14:37:38 +03:00
|
|
|
|
assert(size <= limit);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2019-10-28 18:01:02 +03:00
|
|
|
|
assert(size != map->current || limit != map->limit || size < map->filesize);
|
2018-01-09 21:18:33 +03:00
|
|
|
|
|
2018-01-09 15:30:36 +03:00
|
|
|
|
NTSTATUS status;
|
|
|
|
|
LARGE_INTEGER SectionSize;
|
|
|
|
|
int err, rc = MDBX_SUCCESS;
|
|
|
|
|
|
2020-11-23 10:45:46 +03:00
|
|
|
|
if (!(flags & MDBX_RDONLY) && limit == map->limit && size > map->current &&
|
|
|
|
|
/* workaround for Wine */ mdbx_NtExtendSection) {
|
2018-01-07 14:37:38 +03:00
|
|
|
|
/* growth rw-section */
|
2018-01-09 15:30:36 +03:00
|
|
|
|
SectionSize.QuadPart = size;
|
2020-02-17 22:40:38 +03:00
|
|
|
|
status = mdbx_NtExtendSection(map->section, &SectionSize);
|
2020-01-05 22:01:47 +03:00
|
|
|
|
if (!NT_SUCCESS(status))
|
|
|
|
|
return ntstatus2errcode(status);
|
|
|
|
|
map->current = size;
|
|
|
|
|
if (map->filesize < size)
|
|
|
|
|
map->filesize = size;
|
|
|
|
|
return MDBX_SUCCESS;
|
2017-06-21 01:34:56 +03:00
|
|
|
|
}
|
2018-01-07 14:37:38 +03:00
|
|
|
|
|
2019-10-28 18:01:02 +03:00
|
|
|
|
if (limit > map->limit) {
|
2021-07-10 12:19:20 +03:00
|
|
|
|
err = check_mmap_limit(limit);
|
|
|
|
|
if (unlikely(err != MDBX_SUCCESS))
|
|
|
|
|
return err;
|
|
|
|
|
|
2019-12-24 20:47:33 +03:00
|
|
|
|
/* check ability of address space for growth before unmap */
|
2019-10-28 18:01:02 +03:00
|
|
|
|
PVOID BaseAddress = (PBYTE)map->address + map->limit;
|
|
|
|
|
SIZE_T RegionSize = limit - map->limit;
|
2018-01-14 18:36:25 +03:00
|
|
|
|
status = NtAllocateVirtualMemory(GetCurrentProcess(), &BaseAddress, 0,
|
|
|
|
|
&RegionSize, MEM_RESERVE, PAGE_NOACCESS);
|
2021-05-08 10:59:15 +03:00
|
|
|
|
if (status == (NTSTATUS) /* STATUS_CONFLICTING_ADDRESSES */ 0xC0000018)
|
2020-02-28 16:08:58 +03:00
|
|
|
|
return MDBX_UNABLE_EXTEND_MAPSIZE;
|
2020-01-05 22:01:47 +03:00
|
|
|
|
if (!NT_SUCCESS(status))
|
2018-01-14 18:36:25 +03:00
|
|
|
|
return ntstatus2errcode(status);
|
|
|
|
|
|
|
|
|
|
status = NtFreeVirtualMemory(GetCurrentProcess(), &BaseAddress, &RegionSize,
|
|
|
|
|
MEM_RELEASE);
|
|
|
|
|
if (!NT_SUCCESS(status))
|
|
|
|
|
return ntstatus2errcode(status);
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-07 14:37:38 +03:00
|
|
|
|
/* Windows unable:
|
2018-06-18 21:29:12 +03:00
|
|
|
|
* - shrink a mapped file;
|
|
|
|
|
* - change size of mapped view;
|
|
|
|
|
* - extend read-only mapping;
|
|
|
|
|
* Therefore we should unmap/map entire section. */
|
2021-06-10 02:57:32 +03:00
|
|
|
|
if ((flags & MDBX_MRESIZE_MAY_UNMAP) == 0)
|
2022-03-29 00:31:37 +03:00
|
|
|
|
return MDBX_EPERM;
|
2021-06-10 02:57:32 +03:00
|
|
|
|
|
2021-07-10 16:09:02 +03:00
|
|
|
|
/* Unpoisoning is required for ASAN to avoid false-positive diagnostic
|
|
|
|
|
* when this memory will re-used by malloc or another mmapping.
|
2022-04-21 13:26:06 +03:00
|
|
|
|
* See todo4recovery://erased_by_github/libmdbx/pull/93#issuecomment-613687203
|
|
|
|
|
*/
|
2021-07-16 14:59:37 +03:00
|
|
|
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(map->address, map->limit);
|
2018-01-09 15:30:36 +03:00
|
|
|
|
status = NtUnmapViewOfSection(GetCurrentProcess(), map->address);
|
|
|
|
|
if (!NT_SUCCESS(status))
|
|
|
|
|
return ntstatus2errcode(status);
|
|
|
|
|
status = NtClose(map->section);
|
2018-01-07 14:37:38 +03:00
|
|
|
|
map->section = NULL;
|
2018-01-14 19:48:07 +03:00
|
|
|
|
PVOID ReservedAddress = NULL;
|
|
|
|
|
SIZE_T ReservedSize = limit;
|
2018-01-07 14:37:38 +03:00
|
|
|
|
|
2018-01-09 15:30:36 +03:00
|
|
|
|
if (!NT_SUCCESS(status)) {
|
|
|
|
|
bailout_ntstatus:
|
|
|
|
|
err = ntstatus2errcode(status);
|
|
|
|
|
bailout:
|
|
|
|
|
map->address = NULL;
|
2019-10-28 18:01:02 +03:00
|
|
|
|
map->current = map->limit = 0;
|
2020-01-05 22:01:47 +03:00
|
|
|
|
if (ReservedAddress) {
|
|
|
|
|
ReservedSize = 0;
|
|
|
|
|
status = NtFreeVirtualMemory(GetCurrentProcess(), &ReservedAddress,
|
|
|
|
|
&ReservedSize, MEM_RELEASE);
|
|
|
|
|
assert(NT_SUCCESS(status));
|
|
|
|
|
(void)status;
|
|
|
|
|
}
|
2018-01-09 15:30:36 +03:00
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-20 11:52:44 +03:00
|
|
|
|
retry_file_and_section:
|
2018-01-14 19:48:07 +03:00
|
|
|
|
/* resizing of the file may take a while,
|
|
|
|
|
* therefore we reserve address space to avoid occupy it by other threads */
|
|
|
|
|
ReservedAddress = map->address;
|
|
|
|
|
status = NtAllocateVirtualMemory(GetCurrentProcess(), &ReservedAddress, 0,
|
|
|
|
|
&ReservedSize, MEM_RESERVE, PAGE_NOACCESS);
|
|
|
|
|
if (!NT_SUCCESS(status)) {
|
|
|
|
|
ReservedAddress = NULL;
|
2021-05-08 10:59:15 +03:00
|
|
|
|
if (status != (NTSTATUS) /* STATUS_CONFLICTING_ADDRESSES */ 0xC0000018)
|
2018-01-14 19:48:07 +03:00
|
|
|
|
goto bailout_ntstatus /* no way to recovery */;
|
|
|
|
|
|
2021-06-10 02:57:32 +03:00
|
|
|
|
if (flags & MDBX_MRESIZE_MAY_MOVE)
|
2020-07-06 16:23:52 +03:00
|
|
|
|
/* the base address could be changed */
|
|
|
|
|
map->address = NULL;
|
2018-01-14 19:48:07 +03:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
err = osal_filesize(map->fd, &map->filesize);
|
2018-01-09 15:30:36 +03:00
|
|
|
|
if (err != MDBX_SUCCESS)
|
|
|
|
|
goto bailout;
|
|
|
|
|
|
2018-01-07 14:37:38 +03:00
|
|
|
|
if ((flags & MDBX_RDONLY) == 0 && map->filesize != size) {
|
2022-08-11 01:03:15 +03:00
|
|
|
|
err = osal_ftruncate(map->fd, size);
|
2018-01-09 15:30:36 +03:00
|
|
|
|
if (err == MDBX_SUCCESS)
|
2018-01-07 14:37:38 +03:00
|
|
|
|
map->filesize = size;
|
|
|
|
|
/* ignore error, because Windows unable shrink file
|
2018-01-14 21:02:08 +03:00
|
|
|
|
* that already mapped (by another process) */
|
2017-07-12 21:13:17 +03:00
|
|
|
|
}
|
2018-01-07 14:37:38 +03:00
|
|
|
|
|
|
|
|
|
SectionSize.QuadPart = size;
|
2018-01-09 15:30:36 +03:00
|
|
|
|
status = NtCreateSection(
|
2018-01-07 14:37:38 +03:00
|
|
|
|
&map->section,
|
2018-06-18 21:29:12 +03:00
|
|
|
|
/* DesiredAccess */
|
2018-06-20 13:44:23 +03:00
|
|
|
|
(flags & MDBX_WRITEMAP)
|
2018-01-07 14:37:38 +03:00
|
|
|
|
? SECTION_QUERY | SECTION_MAP_READ | SECTION_EXTEND_SIZE |
|
|
|
|
|
SECTION_MAP_WRITE
|
|
|
|
|
: SECTION_QUERY | SECTION_MAP_READ | SECTION_EXTEND_SIZE,
|
|
|
|
|
/* ObjectAttributes */ NULL,
|
|
|
|
|
/* MaximumSize (InitialSize) */ &SectionSize,
|
2018-06-18 21:29:12 +03:00
|
|
|
|
/* SectionPageProtection */
|
2018-06-20 13:44:23 +03:00
|
|
|
|
(flags & MDBX_RDONLY) ? PAGE_READONLY : PAGE_READWRITE,
|
2018-01-07 14:37:38 +03:00
|
|
|
|
/* AllocationAttributes */ SEC_RESERVE, map->fd);
|
|
|
|
|
|
2018-01-09 15:30:36 +03:00
|
|
|
|
if (!NT_SUCCESS(status))
|
|
|
|
|
goto bailout_ntstatus;
|
2018-01-07 14:37:38 +03:00
|
|
|
|
|
2018-01-14 19:48:07 +03:00
|
|
|
|
if (ReservedAddress) {
|
|
|
|
|
/* release reserved address space */
|
2020-01-05 22:01:47 +03:00
|
|
|
|
ReservedSize = 0;
|
2018-01-14 19:48:07 +03:00
|
|
|
|
status = NtFreeVirtualMemory(GetCurrentProcess(), &ReservedAddress,
|
|
|
|
|
&ReservedSize, MEM_RELEASE);
|
|
|
|
|
ReservedAddress = NULL;
|
|
|
|
|
if (!NT_SUCCESS(status))
|
|
|
|
|
goto bailout_ntstatus;
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-09 15:30:36 +03:00
|
|
|
|
retry_mapview:;
|
2018-01-07 14:37:38 +03:00
|
|
|
|
SIZE_T ViewSize = (flags & MDBX_RDONLY) ? size : limit;
|
2018-01-09 15:30:36 +03:00
|
|
|
|
status = NtMapViewOfSection(
|
2018-01-07 14:37:38 +03:00
|
|
|
|
map->section, GetCurrentProcess(), &map->address,
|
|
|
|
|
/* ZeroBits */ 0,
|
|
|
|
|
/* CommitSize */ 0,
|
|
|
|
|
/* SectionOffset */ NULL, &ViewSize,
|
|
|
|
|
/* InheritDisposition */ ViewUnmap,
|
|
|
|
|
/* AllocationType */ (flags & MDBX_RDONLY) ? 0 : MEM_RESERVE,
|
2018-06-18 21:29:12 +03:00
|
|
|
|
/* Win32Protect */
|
2018-06-20 13:44:23 +03:00
|
|
|
|
(flags & MDBX_WRITEMAP) ? PAGE_READWRITE : PAGE_READONLY);
|
2018-01-07 14:37:38 +03:00
|
|
|
|
|
2018-01-09 15:30:36 +03:00
|
|
|
|
if (!NT_SUCCESS(status)) {
|
2021-05-08 10:59:15 +03:00
|
|
|
|
if (status == (NTSTATUS) /* STATUS_CONFLICTING_ADDRESSES */ 0xC0000018 &&
|
2021-06-10 02:57:32 +03:00
|
|
|
|
map->address && (flags & MDBX_MRESIZE_MAY_MOVE) != 0) {
|
2018-10-20 17:17:31 +03:00
|
|
|
|
/* try remap at another base address */
|
2018-01-07 14:37:38 +03:00
|
|
|
|
map->address = NULL;
|
2018-01-09 15:30:36 +03:00
|
|
|
|
goto retry_mapview;
|
2018-01-07 14:37:38 +03:00
|
|
|
|
}
|
|
|
|
|
NtClose(map->section);
|
2018-01-09 15:30:36 +03:00
|
|
|
|
map->section = NULL;
|
|
|
|
|
|
2019-10-28 18:01:02 +03:00
|
|
|
|
if (map->address && (size != map->current || limit != map->limit)) {
|
2018-01-09 15:30:36 +03:00
|
|
|
|
/* try remap with previously size and limit,
|
2020-02-28 16:08:58 +03:00
|
|
|
|
* but will return MDBX_UNABLE_EXTEND_MAPSIZE on success */
|
2022-03-29 00:31:37 +03:00
|
|
|
|
rc = (limit > map->limit) ? MDBX_UNABLE_EXTEND_MAPSIZE : MDBX_EPERM;
|
2018-01-09 15:30:36 +03:00
|
|
|
|
size = map->current;
|
2020-11-20 11:52:44 +03:00
|
|
|
|
ReservedSize = limit = map->limit;
|
2018-01-09 15:30:36 +03:00
|
|
|
|
goto retry_file_and_section;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* no way to recovery */
|
|
|
|
|
goto bailout_ntstatus;
|
2017-07-12 21:13:17 +03:00
|
|
|
|
}
|
2018-01-07 14:37:38 +03:00
|
|
|
|
assert(map->address != MAP_FAILED);
|
|
|
|
|
|
|
|
|
|
map->current = (size_t)SectionSize.QuadPart;
|
2019-10-28 18:01:02 +03:00
|
|
|
|
map->limit = ViewSize;
|
2020-07-06 16:23:52 +03:00
|
|
|
|
|
2021-02-26 04:10:56 +03:00
|
|
|
|
#else /* Windows */
|
2019-10-28 18:01:02 +03:00
|
|
|
|
|
2021-07-08 13:18:51 +03:00
|
|
|
|
map->filesize = 0;
|
2022-08-11 01:03:15 +03:00
|
|
|
|
int rc = osal_filesize(map->fd, &map->filesize);
|
2019-10-28 18:01:02 +03:00
|
|
|
|
if (rc != MDBX_SUCCESS)
|
|
|
|
|
return rc;
|
|
|
|
|
|
|
|
|
|
if (flags & MDBX_RDONLY) {
|
2021-07-08 13:18:51 +03:00
|
|
|
|
map->current = (map->filesize > limit) ? limit : (size_t)map->filesize;
|
2019-10-28 18:01:02 +03:00
|
|
|
|
if (map->current != size)
|
2022-03-29 00:31:37 +03:00
|
|
|
|
rc = (size > map->current) ? MDBX_UNABLE_EXTEND_MAPSIZE : MDBX_EPERM;
|
2021-07-10 16:09:02 +03:00
|
|
|
|
} else {
|
|
|
|
|
if (map->filesize != size) {
|
2022-08-11 01:03:15 +03:00
|
|
|
|
rc = osal_ftruncate(map->fd, size);
|
2021-07-10 16:09:02 +03:00
|
|
|
|
if (rc != MDBX_SUCCESS)
|
|
|
|
|
return rc;
|
|
|
|
|
map->filesize = size;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (map->current > size) {
|
|
|
|
|
/* Clearing asan's bitmask for the region which released in shrinking,
|
|
|
|
|
* since:
|
|
|
|
|
* - after the shrinking we will get an exception when accessing
|
|
|
|
|
* this region and (therefore) do not need the help of ASAN.
|
|
|
|
|
* - this allows us to clear the mask only within the file size
|
|
|
|
|
* when closing the mapping. */
|
2021-07-16 14:59:37 +03:00
|
|
|
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(
|
2021-07-10 16:09:02 +03:00
|
|
|
|
(char *)map->address + size,
|
|
|
|
|
((map->current < map->limit) ? map->current : map->limit) - size);
|
|
|
|
|
}
|
2019-10-28 18:01:02 +03:00
|
|
|
|
map->current = size;
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-22 03:56:59 +03:00
|
|
|
|
if (limit == map->limit)
|
2021-07-10 17:52:17 +03:00
|
|
|
|
return rc;
|
2020-09-22 03:56:59 +03:00
|
|
|
|
|
|
|
|
|
if (limit < map->limit) {
|
|
|
|
|
/* unmap an excess at end of mapping. */
|
2021-08-16 23:45:56 +03:00
|
|
|
|
// coverity[offset_free : FALSE]
|
2020-09-22 03:56:59 +03:00
|
|
|
|
if (unlikely(munmap(map->dxb + limit, map->limit - limit)))
|
|
|
|
|
return errno;
|
|
|
|
|
map->limit = limit;
|
2021-07-10 17:52:17 +03:00
|
|
|
|
return rc;
|
2020-09-22 03:56:59 +03:00
|
|
|
|
}
|
|
|
|
|
|
2021-07-10 17:52:17 +03:00
|
|
|
|
int err = check_mmap_limit(limit);
|
|
|
|
|
if (unlikely(err != MDBX_SUCCESS))
|
|
|
|
|
return err;
|
2021-07-10 12:19:20 +03:00
|
|
|
|
|
2020-09-22 03:56:59 +03:00
|
|
|
|
assert(limit > map->limit);
|
|
|
|
|
uint8_t *ptr = MAP_FAILED;
|
|
|
|
|
|
2019-12-06 21:26:39 +03:00
|
|
|
|
#if defined(MREMAP_MAYMOVE)
|
2021-06-10 02:57:32 +03:00
|
|
|
|
ptr = mremap(map->address, map->limit, limit,
|
|
|
|
|
(flags & MDBX_MRESIZE_MAY_MOVE) ? MREMAP_MAYMOVE : 0);
|
2020-09-22 03:56:59 +03:00
|
|
|
|
if (ptr == MAP_FAILED) {
|
2021-07-10 17:52:17 +03:00
|
|
|
|
err = errno;
|
2020-09-22 03:56:59 +03:00
|
|
|
|
switch (err) {
|
|
|
|
|
default:
|
|
|
|
|
return err;
|
|
|
|
|
case EAGAIN:
|
|
|
|
|
case ENOMEM:
|
|
|
|
|
return MDBX_UNABLE_EXTEND_MAPSIZE;
|
|
|
|
|
case EFAULT /* MADV_DODUMP / MADV_DONTDUMP are mixed for mmap-range */:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif /* MREMAP_MAYMOVE */
|
|
|
|
|
|
|
|
|
|
const unsigned mmap_flags =
|
2021-03-17 01:14:52 +03:00
|
|
|
|
MAP_CONCEAL | MAP_SHARED | MAP_FILE | MAP_NORESERVE |
|
2020-09-22 03:56:59 +03:00
|
|
|
|
(F_ISSET(flags, MDBX_UTTERLY_NOSYNC) ? MAP_NOSYNC : 0);
|
|
|
|
|
const unsigned mmap_prot =
|
|
|
|
|
(flags & MDBX_WRITEMAP) ? PROT_READ | PROT_WRITE : PROT_READ;
|
|
|
|
|
|
|
|
|
|
if (ptr == MAP_FAILED) {
|
|
|
|
|
/* Try to mmap additional space beyond the end of mapping. */
|
|
|
|
|
ptr = mmap(map->dxb + map->limit, limit - map->limit, mmap_prot,
|
2021-03-17 01:13:47 +03:00
|
|
|
|
mmap_flags | MAP_FIXED_NOREPLACE, map->fd, map->limit);
|
2020-09-22 03:56:59 +03:00
|
|
|
|
if (ptr == map->dxb + map->limit)
|
|
|
|
|
ptr = map->dxb;
|
|
|
|
|
else if (ptr != MAP_FAILED) {
|
|
|
|
|
/* the desired address is busy, unmap unsuitable one */
|
|
|
|
|
if (unlikely(munmap(ptr, limit - map->limit)))
|
|
|
|
|
return errno;
|
|
|
|
|
ptr = MAP_FAILED;
|
|
|
|
|
} else {
|
2021-07-10 17:52:17 +03:00
|
|
|
|
err = errno;
|
2020-09-22 03:56:59 +03:00
|
|
|
|
switch (err) {
|
|
|
|
|
default:
|
|
|
|
|
return err;
|
2019-12-17 02:59:13 +03:00
|
|
|
|
case EAGAIN:
|
|
|
|
|
case ENOMEM:
|
2020-09-22 03:56:59 +03:00
|
|
|
|
return MDBX_UNABLE_EXTEND_MAPSIZE;
|
|
|
|
|
case EEXIST: /* address busy */
|
|
|
|
|
case EINVAL: /* kernel don't support MAP_FIXED_NOREPLACE */
|
|
|
|
|
break;
|
2019-12-17 02:59:13 +03:00
|
|
|
|
}
|
2019-07-14 00:39:17 +03:00
|
|
|
|
}
|
2020-09-22 03:56:59 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ptr == MAP_FAILED) {
|
|
|
|
|
/* unmap and map again whole region */
|
2021-06-10 02:57:32 +03:00
|
|
|
|
if ((flags & MDBX_MRESIZE_MAY_UNMAP) == 0) {
|
2020-07-06 16:23:52 +03:00
|
|
|
|
/* TODO: Perhaps here it is worth to implement suspend/resume threads
|
2020-09-22 03:56:59 +03:00
|
|
|
|
* and perform unmap/map as like for Windows. */
|
2020-07-06 16:23:52 +03:00
|
|
|
|
return MDBX_UNABLE_EXTEND_MAPSIZE;
|
2020-09-22 03:56:59 +03:00
|
|
|
|
}
|
2020-07-06 16:23:52 +03:00
|
|
|
|
|
|
|
|
|
if (unlikely(munmap(map->address, map->limit)))
|
|
|
|
|
return errno;
|
|
|
|
|
|
2021-08-16 23:45:56 +03:00
|
|
|
|
// coverity[pass_freed_arg : FALSE]
|
2021-06-10 02:57:32 +03:00
|
|
|
|
ptr = mmap(map->address, limit, mmap_prot,
|
|
|
|
|
(flags & MDBX_MRESIZE_MAY_MOVE)
|
|
|
|
|
? mmap_flags
|
|
|
|
|
: mmap_flags | (MAP_FIXED_NOREPLACE ? MAP_FIXED_NOREPLACE
|
|
|
|
|
: MAP_FIXED),
|
|
|
|
|
map->fd, 0);
|
|
|
|
|
if (MAP_FIXED_NOREPLACE != 0 && MAP_FIXED_NOREPLACE != MAP_FIXED &&
|
|
|
|
|
unlikely(ptr == MAP_FAILED) && !(flags & MDBX_MRESIZE_MAY_MOVE) &&
|
|
|
|
|
errno == /* kernel don't support MAP_FIXED_NOREPLACE */ EINVAL)
|
2021-08-16 23:45:56 +03:00
|
|
|
|
// coverity[pass_freed_arg : FALSE]
|
2021-06-10 02:57:32 +03:00
|
|
|
|
ptr = mmap(map->address, limit, mmap_prot, mmap_flags | MAP_FIXED,
|
|
|
|
|
map->fd, 0);
|
|
|
|
|
|
2020-07-06 16:23:52 +03:00
|
|
|
|
if (unlikely(ptr == MAP_FAILED)) {
|
2021-06-10 02:57:32 +03:00
|
|
|
|
/* try to restore prev mapping */
|
2021-08-16 23:45:56 +03:00
|
|
|
|
// coverity[pass_freed_arg : FALSE]
|
2021-06-10 02:57:32 +03:00
|
|
|
|
ptr = mmap(map->address, map->limit, mmap_prot,
|
|
|
|
|
(flags & MDBX_MRESIZE_MAY_MOVE)
|
|
|
|
|
? mmap_flags
|
|
|
|
|
: mmap_flags | (MAP_FIXED_NOREPLACE ? MAP_FIXED_NOREPLACE
|
|
|
|
|
: MAP_FIXED),
|
|
|
|
|
map->fd, 0);
|
|
|
|
|
if (MAP_FIXED_NOREPLACE != 0 && MAP_FIXED_NOREPLACE != MAP_FIXED &&
|
|
|
|
|
unlikely(ptr == MAP_FAILED) && !(flags & MDBX_MRESIZE_MAY_MOVE) &&
|
|
|
|
|
errno == /* kernel don't support MAP_FIXED_NOREPLACE */ EINVAL)
|
2021-08-16 23:45:56 +03:00
|
|
|
|
// coverity[pass_freed_arg : FALSE]
|
2021-06-10 02:57:32 +03:00
|
|
|
|
ptr = mmap(map->address, map->limit, mmap_prot, mmap_flags | MAP_FIXED,
|
|
|
|
|
map->fd, 0);
|
2020-07-06 16:23:52 +03:00
|
|
|
|
if (unlikely(ptr == MAP_FAILED)) {
|
|
|
|
|
VALGRIND_MAKE_MEM_NOACCESS(map->address, map->current);
|
|
|
|
|
/* Unpoisoning is required for ASAN to avoid false-positive diagnostic
|
2020-09-21 23:51:47 -04:00
|
|
|
|
* when this memory will re-used by malloc or another mmapping.
|
2022-04-21 13:26:06 +03:00
|
|
|
|
* See
|
|
|
|
|
* todo4recovery://erased_by_github/libmdbx/pull/93#issuecomment-613687203
|
2020-07-06 16:23:52 +03:00
|
|
|
|
*/
|
2021-07-16 14:59:37 +03:00
|
|
|
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(
|
|
|
|
|
map->address,
|
|
|
|
|
(map->current < map->limit) ? map->current : map->limit);
|
2020-07-06 16:23:52 +03:00
|
|
|
|
map->limit = 0;
|
|
|
|
|
map->current = 0;
|
|
|
|
|
map->address = nullptr;
|
|
|
|
|
return errno;
|
|
|
|
|
}
|
2020-09-22 03:56:59 +03:00
|
|
|
|
rc = MDBX_UNABLE_EXTEND_MAPSIZE;
|
|
|
|
|
limit = map->limit;
|
2020-07-06 16:23:52 +03:00
|
|
|
|
}
|
2020-09-22 03:56:59 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
assert(ptr && ptr != MAP_FAILED);
|
|
|
|
|
if (map->address != ptr) {
|
|
|
|
|
VALGRIND_MAKE_MEM_NOACCESS(map->address, map->current);
|
|
|
|
|
/* Unpoisoning is required for ASAN to avoid false-positive diagnostic
|
2020-09-25 01:13:11 +03:00
|
|
|
|
* when this memory will re-used by malloc or another mmapping.
|
2022-04-21 13:26:06 +03:00
|
|
|
|
* See
|
|
|
|
|
* todo4recovery://erased_by_github/libmdbx/pull/93#issuecomment-613687203
|
|
|
|
|
*/
|
2021-07-16 14:59:37 +03:00
|
|
|
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(
|
2021-07-10 16:09:02 +03:00
|
|
|
|
map->address, (map->current < map->limit) ? map->current : map->limit);
|
2020-09-22 03:56:59 +03:00
|
|
|
|
|
|
|
|
|
VALGRIND_MAKE_MEM_DEFINED(ptr, map->current);
|
2021-07-16 14:59:37 +03:00
|
|
|
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(ptr, map->current);
|
2020-09-22 03:56:59 +03:00
|
|
|
|
map->address = ptr;
|
|
|
|
|
}
|
|
|
|
|
map->limit = limit;
|
2019-10-28 18:01:02 +03:00
|
|
|
|
|
2021-02-26 04:10:56 +03:00
|
|
|
|
#if MDBX_ENABLE_MADVISE
|
2019-10-28 18:01:02 +03:00
|
|
|
|
#ifdef MADV_DONTFORK
|
2020-09-22 03:56:59 +03:00
|
|
|
|
if (unlikely(madvise(map->address, map->limit, MADV_DONTFORK) != 0))
|
|
|
|
|
return errno;
|
2019-12-06 21:26:39 +03:00
|
|
|
|
#endif /* MADV_DONTFORK */
|
2019-10-28 18:01:02 +03:00
|
|
|
|
#ifdef MADV_NOHUGEPAGE
|
2020-09-22 03:56:59 +03:00
|
|
|
|
(void)madvise(map->address, map->limit, MADV_NOHUGEPAGE);
|
2019-12-06 21:26:39 +03:00
|
|
|
|
#endif /* MADV_NOHUGEPAGE */
|
2021-02-26 04:10:56 +03:00
|
|
|
|
#endif /* MDBX_ENABLE_MADVISE */
|
2020-09-22 03:56:59 +03:00
|
|
|
|
|
|
|
|
|
#endif /* POSIX / Windows */
|
2020-07-06 16:23:52 +03:00
|
|
|
|
|
2019-10-28 18:01:02 +03:00
|
|
|
|
return rc;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
}
|
2017-04-27 15:18:33 +03:00
|
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
__cold MDBX_INTERNAL_FUNC void osal_jitter(bool tiny) {
|
2017-04-27 15:18:33 +03:00
|
|
|
|
for (;;) {
|
|
|
|
|
#if defined(_M_IX86) || defined(_M_X64) || defined(__i386__) || \
|
|
|
|
|
defined(__x86_64__)
|
|
|
|
|
const unsigned salt = 277u * (unsigned)__rdtsc();
|
2022-04-22 18:31:49 +03:00
|
|
|
|
#elif (defined(_WIN32) || defined(_WIN64)) && MDBX_WITHOUT_MSVC_CRT
|
|
|
|
|
static ULONG state;
|
|
|
|
|
const unsigned salt = (unsigned)RtlRandomEx(&state);
|
2017-04-27 15:18:33 +03:00
|
|
|
|
#else
|
|
|
|
|
const unsigned salt = rand();
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
const unsigned coin = salt % (tiny ? 29u : 43u);
|
|
|
|
|
if (coin < 43 / 3)
|
|
|
|
|
break;
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
SwitchToThread();
|
|
|
|
|
if (coin > 43 * 2 / 3)
|
|
|
|
|
Sleep(1);
|
|
|
|
|
#else
|
|
|
|
|
sched_yield();
|
|
|
|
|
if (coin > 43 * 2 / 3)
|
|
|
|
|
usleep(coin);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-08-23 13:13:20 +03:00
|
|
|
|
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
#elif defined(__APPLE__) || defined(__MACH__)
|
|
|
|
|
#include <mach/mach_time.h>
|
|
|
|
|
#elif defined(__linux__) || defined(__gnu_linux__)
|
2021-07-03 00:41:08 +03:00
|
|
|
|
__cold static clockid_t choice_monoclock(void) {
|
2019-08-23 13:13:20 +03:00
|
|
|
|
struct timespec probe;
|
|
|
|
|
#if defined(CLOCK_BOOTTIME)
|
|
|
|
|
if (clock_gettime(CLOCK_BOOTTIME, &probe) == 0)
|
|
|
|
|
return CLOCK_BOOTTIME;
|
|
|
|
|
#elif defined(CLOCK_MONOTONIC_RAW)
|
|
|
|
|
if (clock_gettime(CLOCK_MONOTONIC_RAW, &probe) == 0)
|
|
|
|
|
return CLOCK_MONOTONIC_RAW;
|
|
|
|
|
#elif defined(CLOCK_MONOTONIC_COARSE)
|
|
|
|
|
if (clock_gettime(CLOCK_MONOTONIC_COARSE, &probe) == 0)
|
|
|
|
|
return CLOCK_MONOTONIC_COARSE;
|
|
|
|
|
#endif
|
|
|
|
|
return CLOCK_MONOTONIC;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2019-10-01 13:53:52 +03:00
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
static LARGE_INTEGER performance_frequency;
|
|
|
|
|
#elif defined(__APPLE__) || defined(__MACH__)
|
|
|
|
|
static uint64_t ratio_16dot16_to_monotine;
|
|
|
|
|
#endif
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC uint64_t osal_16dot16_to_monotime(uint32_t seconds_16dot16) {
|
2019-08-23 13:13:20 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2019-10-01 13:53:52 +03:00
|
|
|
|
if (unlikely(performance_frequency.QuadPart == 0))
|
2019-08-23 13:13:20 +03:00
|
|
|
|
QueryPerformanceFrequency(&performance_frequency);
|
|
|
|
|
const uint64_t ratio = performance_frequency.QuadPart;
|
|
|
|
|
#elif defined(__APPLE__) || defined(__MACH__)
|
2019-10-01 13:53:52 +03:00
|
|
|
|
if (unlikely(ratio_16dot16_to_monotine == 0)) {
|
2019-08-23 13:13:20 +03:00
|
|
|
|
mach_timebase_info_data_t ti;
|
|
|
|
|
mach_timebase_info(&ti);
|
2019-10-01 13:53:52 +03:00
|
|
|
|
ratio_16dot16_to_monotine = UINT64_C(1000000000) * ti.denom / ti.numer;
|
2019-08-23 13:13:20 +03:00
|
|
|
|
}
|
2019-10-01 13:53:52 +03:00
|
|
|
|
const uint64_t ratio = ratio_16dot16_to_monotine;
|
2019-08-23 13:13:20 +03:00
|
|
|
|
#else
|
|
|
|
|
const uint64_t ratio = UINT64_C(1000000000);
|
|
|
|
|
#endif
|
2021-05-12 17:44:18 +03:00
|
|
|
|
const uint64_t ret = (ratio * seconds_16dot16 + 32768) >> 16;
|
|
|
|
|
return likely(ret || seconds_16dot16 == 0) ? ret : /* fix underflow */ 1;
|
2019-08-23 13:13:20 +03:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC uint32_t osal_monotime_to_16dot16(uint64_t monotime) {
|
2019-10-01 13:53:52 +03:00
|
|
|
|
static uint64_t limit;
|
|
|
|
|
if (unlikely(monotime > limit)) {
|
2022-08-07 12:10:36 +03:00
|
|
|
|
if (likely(limit != 0))
|
2019-10-01 13:53:52 +03:00
|
|
|
|
return UINT32_MAX;
|
2022-08-11 01:03:15 +03:00
|
|
|
|
limit = osal_16dot16_to_monotime(UINT32_MAX - 1);
|
2022-08-07 12:10:36 +03:00
|
|
|
|
if (unlikely(monotime > limit))
|
2019-10-01 13:53:52 +03:00
|
|
|
|
return UINT32_MAX;
|
|
|
|
|
}
|
2021-05-12 17:44:18 +03:00
|
|
|
|
const uint32_t ret =
|
2019-10-01 13:53:52 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2021-05-12 17:44:18 +03:00
|
|
|
|
(uint32_t)((monotime << 16) / performance_frequency.QuadPart);
|
2019-10-01 13:53:52 +03:00
|
|
|
|
#elif defined(__APPLE__) || defined(__MACH__)
|
2021-05-12 17:44:18 +03:00
|
|
|
|
(uint32_t)((monotime << 16) / ratio_16dot16_to_monotine);
|
2019-10-01 13:53:52 +03:00
|
|
|
|
#else
|
2021-05-12 17:44:18 +03:00
|
|
|
|
(uint32_t)(monotime * 128 / 1953125);
|
2019-10-01 13:53:52 +03:00
|
|
|
|
#endif
|
2022-10-07 14:53:35 +03:00
|
|
|
|
return ret;
|
2019-10-01 13:53:52 +03:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
MDBX_INTERNAL_FUNC uint64_t osal_monotime(void) {
|
2019-08-23 13:13:20 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
LARGE_INTEGER counter;
|
|
|
|
|
counter.QuadPart = 0;
|
|
|
|
|
QueryPerformanceCounter(&counter);
|
|
|
|
|
return counter.QuadPart;
|
|
|
|
|
#elif defined(__APPLE__) || defined(__MACH__)
|
|
|
|
|
return mach_absolute_time();
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
|
|
#if defined(__linux__) || defined(__gnu_linux__)
|
|
|
|
|
static clockid_t posix_clockid = -1;
|
|
|
|
|
if (unlikely(posix_clockid < 0))
|
2019-08-23 22:15:35 +03:00
|
|
|
|
posix_clockid = choice_monoclock();
|
2019-08-23 13:13:20 +03:00
|
|
|
|
#elif defined(CLOCK_MONOTONIC)
|
|
|
|
|
#define posix_clockid CLOCK_MONOTONIC
|
|
|
|
|
#else
|
|
|
|
|
#define posix_clockid CLOCK_REALTIME
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
struct timespec ts;
|
|
|
|
|
if (unlikely(clock_gettime(posix_clockid, &ts) != 0)) {
|
|
|
|
|
ts.tv_nsec = 0;
|
|
|
|
|
ts.tv_sec = 0;
|
|
|
|
|
}
|
|
|
|
|
return ts.tv_sec * UINT64_C(1000000000) + ts.tv_nsec;
|
|
|
|
|
#endif
|
|
|
|
|
}
|
2019-11-01 21:25:17 +03:00
|
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
|
|
static void bootid_shake(bin128_t *p) {
|
|
|
|
|
/* Bob Jenkins's PRNG: https://burtleburtle.net/bob/rand/smallprng.html */
|
|
|
|
|
const uint32_t e = p->a - (p->b << 23 | p->b >> 9);
|
|
|
|
|
p->a = p->b ^ (p->c << 16 | p->c >> 16);
|
|
|
|
|
p->b = p->c + (p->d << 11 | p->d >> 21);
|
|
|
|
|
p->c = p->d + e;
|
|
|
|
|
p->d = e + p->a;
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-07 12:10:36 +03:00
|
|
|
|
__cold static void bootid_collect(bin128_t *p, const void *s, size_t n) {
|
2019-11-01 21:25:17 +03:00
|
|
|
|
p->y += UINT64_C(64526882297375213);
|
|
|
|
|
bootid_shake(p);
|
|
|
|
|
for (size_t i = 0; i < n; ++i) {
|
|
|
|
|
bootid_shake(p);
|
|
|
|
|
p->y ^= UINT64_C(48797879452804441) * ((const uint8_t *)s)[i];
|
|
|
|
|
bootid_shake(p);
|
|
|
|
|
p->y += 14621231;
|
|
|
|
|
}
|
|
|
|
|
bootid_shake(p);
|
|
|
|
|
|
|
|
|
|
/* minor non-linear tomfoolery */
|
|
|
|
|
const unsigned z = p->x % 61;
|
|
|
|
|
p->y = p->y << z | p->y >> (64 - z);
|
|
|
|
|
bootid_shake(p);
|
|
|
|
|
bootid_shake(p);
|
|
|
|
|
const unsigned q = p->x % 59;
|
|
|
|
|
p->y = p->y << q | p->y >> (64 - q);
|
|
|
|
|
bootid_shake(p);
|
|
|
|
|
bootid_shake(p);
|
|
|
|
|
bootid_shake(p);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
|
|
|
|
|
static uint64_t windows_systemtime_ms() {
|
|
|
|
|
FILETIME ft;
|
|
|
|
|
GetSystemTimeAsFileTime(&ft);
|
|
|
|
|
return ((uint64_t)ft.dwHighDateTime << 32 | ft.dwLowDateTime) / 10000ul;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static uint64_t windows_bootime(void) {
|
|
|
|
|
unsigned confirmed = 0;
|
|
|
|
|
uint64_t boottime = 0;
|
|
|
|
|
uint64_t up0 = mdbx_GetTickCount64();
|
|
|
|
|
uint64_t st0 = windows_systemtime_ms();
|
|
|
|
|
for (uint64_t fuse = st0; up0 && st0 < fuse + 1000 * 1000u / 42;) {
|
|
|
|
|
YieldProcessor();
|
|
|
|
|
const uint64_t up1 = mdbx_GetTickCount64();
|
|
|
|
|
const uint64_t st1 = windows_systemtime_ms();
|
|
|
|
|
if (st1 > fuse && st1 == st0 && up1 == up0) {
|
|
|
|
|
uint64_t diff = st1 - up1;
|
|
|
|
|
if (boottime == diff) {
|
|
|
|
|
if (++confirmed > 4)
|
|
|
|
|
return boottime;
|
|
|
|
|
} else {
|
|
|
|
|
confirmed = 0;
|
|
|
|
|
boottime = diff;
|
|
|
|
|
}
|
|
|
|
|
fuse = st1;
|
|
|
|
|
Sleep(1);
|
|
|
|
|
}
|
|
|
|
|
st0 = st1;
|
|
|
|
|
up0 = up1;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-08 00:45:31 +03:00
|
|
|
|
static LSTATUS mdbx_RegGetValue(HKEY hKey, LPCSTR lpSubKey, LPCSTR lpValue,
|
|
|
|
|
PVOID pvData, LPDWORD pcbData) {
|
|
|
|
|
LSTATUS rc;
|
|
|
|
|
if (!mdbx_RegGetValueA) {
|
|
|
|
|
/* an old Windows 2000/XP */
|
|
|
|
|
HKEY hSubKey;
|
|
|
|
|
rc = RegOpenKeyA(hKey, lpSubKey, &hSubKey);
|
|
|
|
|
if (rc == ERROR_SUCCESS) {
|
|
|
|
|
rc = RegQueryValueExA(hSubKey, lpValue, NULL, NULL, pvData, pcbData);
|
|
|
|
|
RegCloseKey(hSubKey);
|
|
|
|
|
}
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
rc = mdbx_RegGetValueA(hKey, lpSubKey, lpValue, RRF_RT_ANY, NULL, pvData,
|
|
|
|
|
pcbData);
|
2019-11-01 21:25:17 +03:00
|
|
|
|
if (rc != ERROR_FILE_NOT_FOUND)
|
|
|
|
|
return rc;
|
|
|
|
|
|
2020-10-08 00:45:31 +03:00
|
|
|
|
rc = mdbx_RegGetValueA(hKey, lpSubKey, lpValue,
|
|
|
|
|
RRF_RT_ANY | 0x00010000 /* RRF_SUBKEY_WOW6464KEY */,
|
|
|
|
|
NULL, pvData, pcbData);
|
2019-11-01 21:25:17 +03:00
|
|
|
|
if (rc != ERROR_FILE_NOT_FOUND)
|
|
|
|
|
return rc;
|
2020-10-08 00:45:31 +03:00
|
|
|
|
return mdbx_RegGetValueA(hKey, lpSubKey, lpValue,
|
|
|
|
|
RRF_RT_ANY | 0x00020000 /* RRF_SUBKEY_WOW6432KEY */,
|
|
|
|
|
NULL, pvData, pcbData);
|
2019-11-01 21:25:17 +03:00
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2021-07-03 00:41:08 +03:00
|
|
|
|
__cold MDBX_MAYBE_UNUSED static bool
|
2021-05-11 20:14:09 +03:00
|
|
|
|
bootid_parse_uuid(bin128_t *s, const void *p, const size_t n) {
|
2019-11-01 21:25:17 +03:00
|
|
|
|
if (n > 31) {
|
|
|
|
|
unsigned bits = 0;
|
|
|
|
|
for (unsigned i = 0; i < n; ++i) /* try parse an UUID in text form */ {
|
|
|
|
|
uint8_t c = ((const uint8_t *)p)[i];
|
|
|
|
|
if (c >= '0' && c <= '9')
|
|
|
|
|
c -= '0';
|
|
|
|
|
else if (c >= 'a' && c <= 'f')
|
|
|
|
|
c -= 'a' - 10;
|
|
|
|
|
else if (c >= 'A' && c <= 'F')
|
|
|
|
|
c -= 'A' - 10;
|
|
|
|
|
else
|
|
|
|
|
continue;
|
|
|
|
|
assert(c <= 15);
|
|
|
|
|
c ^= s->y >> 60;
|
|
|
|
|
s->y = s->y << 4 | s->x >> 60;
|
|
|
|
|
s->x = s->x << 4 | c;
|
|
|
|
|
bits += 4;
|
|
|
|
|
}
|
|
|
|
|
if (bits > 42 * 3)
|
|
|
|
|
/* UUID parsed successfully */
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (n > 15) /* is enough handle it as a binary? */ {
|
|
|
|
|
if (n == sizeof(bin128_t)) {
|
|
|
|
|
bin128_t aligned;
|
|
|
|
|
memcpy(&aligned, p, sizeof(bin128_t));
|
|
|
|
|
s->x += aligned.x;
|
|
|
|
|
s->y += aligned.y;
|
|
|
|
|
} else
|
|
|
|
|
bootid_collect(s, p, n);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (n)
|
|
|
|
|
bootid_collect(s, p, n);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
__cold MDBX_INTERNAL_FUNC bin128_t osal_bootid(void) {
|
2019-11-01 21:25:17 +03:00
|
|
|
|
bin128_t bin = {{0, 0}};
|
2019-11-09 10:50:43 +03:00
|
|
|
|
bool got_machineid = false, got_boottime = false, got_bootseq = false;
|
2019-11-01 21:25:17 +03:00
|
|
|
|
|
|
|
|
|
#if defined(__linux__) || defined(__gnu_linux__)
|
|
|
|
|
{
|
|
|
|
|
const int fd =
|
|
|
|
|
open("/proc/sys/kernel/random/boot_id", O_RDONLY | O_NOFOLLOW);
|
|
|
|
|
if (fd != -1) {
|
2019-11-04 17:19:12 +03:00
|
|
|
|
struct statfs fs;
|
2019-11-01 21:25:17 +03:00
|
|
|
|
char buf[42];
|
|
|
|
|
const ssize_t len =
|
2019-11-04 17:19:12 +03:00
|
|
|
|
(fstatfs(fd, &fs) == 0 && fs.f_type == /* procfs */ 0x9FA0)
|
2019-11-01 21:25:17 +03:00
|
|
|
|
? read(fd, buf, sizeof(buf))
|
|
|
|
|
: -1;
|
2020-07-24 17:43:35 +03:00
|
|
|
|
const int err = close(fd);
|
|
|
|
|
assert(err == 0);
|
|
|
|
|
(void)err;
|
2019-11-01 21:25:17 +03:00
|
|
|
|
if (len > 0 && bootid_parse_uuid(&bin, buf, len))
|
|
|
|
|
return bin;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif /* Linux */
|
|
|
|
|
|
|
|
|
|
#if defined(__APPLE__) || defined(__MACH__)
|
|
|
|
|
{
|
|
|
|
|
char buf[42];
|
|
|
|
|
size_t len = sizeof(buf);
|
|
|
|
|
if (!sysctlbyname("kern.bootsessionuuid", buf, &len, nullptr, 0) &&
|
|
|
|
|
bootid_parse_uuid(&bin, buf, len))
|
|
|
|
|
return bin;
|
|
|
|
|
|
|
|
|
|
#if defined(__MAC_OS_X_VERSION_MIN_REQUIRED) && \
|
|
|
|
|
__MAC_OS_X_VERSION_MIN_REQUIRED > 1050
|
|
|
|
|
uuid_t uuid;
|
|
|
|
|
struct timespec wait = {0, 1000000000u / 42};
|
|
|
|
|
if (!gethostuuid(uuid, &wait) &&
|
|
|
|
|
bootid_parse_uuid(&bin, uuid, sizeof(uuid)))
|
|
|
|
|
got_machineid = true;
|
|
|
|
|
#endif /* > 10.5 */
|
|
|
|
|
|
|
|
|
|
struct timeval boottime;
|
|
|
|
|
len = sizeof(boottime);
|
|
|
|
|
if (!sysctlbyname("kern.boottime", &boottime, &len, nullptr, 0) &&
|
|
|
|
|
len == sizeof(boottime) && boottime.tv_sec)
|
2019-11-09 10:50:43 +03:00
|
|
|
|
got_boottime = true;
|
2019-11-01 21:25:17 +03:00
|
|
|
|
}
|
|
|
|
|
#endif /* Apple/Darwin */
|
|
|
|
|
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
{
|
|
|
|
|
union buf {
|
|
|
|
|
DWORD BootId;
|
|
|
|
|
DWORD BaseTime;
|
|
|
|
|
SYSTEM_TIMEOFDAY_INFORMATION SysTimeOfDayInfo;
|
|
|
|
|
struct {
|
|
|
|
|
LARGE_INTEGER BootTime;
|
|
|
|
|
LARGE_INTEGER CurrentTime;
|
|
|
|
|
LARGE_INTEGER TimeZoneBias;
|
|
|
|
|
ULONG TimeZoneId;
|
|
|
|
|
ULONG Reserved;
|
|
|
|
|
ULONGLONG BootTimeBias;
|
|
|
|
|
ULONGLONG SleepTimeBias;
|
|
|
|
|
} SysTimeOfDayInfoHacked;
|
|
|
|
|
wchar_t MachineGuid[42];
|
|
|
|
|
char DigitalProductId[248];
|
|
|
|
|
} buf;
|
|
|
|
|
|
2020-02-19 16:00:40 +03:00
|
|
|
|
static const char HKLM_MicrosoftCryptography[] =
|
|
|
|
|
"SOFTWARE\\Microsoft\\Cryptography";
|
2019-11-01 21:25:17 +03:00
|
|
|
|
DWORD len = sizeof(buf);
|
|
|
|
|
/* Windows is madness and must die */
|
|
|
|
|
if (mdbx_RegGetValue(HKEY_LOCAL_MACHINE, HKLM_MicrosoftCryptography,
|
2020-10-08 00:45:31 +03:00
|
|
|
|
"MachineGuid", &buf.MachineGuid,
|
2019-11-01 21:25:17 +03:00
|
|
|
|
&len) == ERROR_SUCCESS &&
|
2021-07-19 12:07:45 +03:00
|
|
|
|
len < sizeof(buf))
|
2019-11-01 21:25:17 +03:00
|
|
|
|
got_machineid = bootid_parse_uuid(&bin, &buf.MachineGuid, len);
|
|
|
|
|
|
|
|
|
|
if (!got_machineid) {
|
|
|
|
|
/* again, Windows is madness */
|
2020-02-19 16:00:40 +03:00
|
|
|
|
static const char HKLM_WindowsNT[] =
|
|
|
|
|
"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion";
|
|
|
|
|
static const char HKLM_WindowsNT_DPK[] =
|
|
|
|
|
"SOFTWARE\\Microsoft\\Windows "
|
|
|
|
|
"NT\\CurrentVersion\\DefaultProductKey";
|
|
|
|
|
static const char HKLM_WindowsNT_DPK2[] =
|
|
|
|
|
"SOFTWARE\\Microsoft\\Windows "
|
|
|
|
|
"NT\\CurrentVersion\\DefaultProductKey2";
|
2019-11-01 21:25:17 +03:00
|
|
|
|
|
|
|
|
|
len = sizeof(buf);
|
|
|
|
|
if (mdbx_RegGetValue(HKEY_LOCAL_MACHINE, HKLM_WindowsNT,
|
2020-10-08 00:45:31 +03:00
|
|
|
|
"DigitalProductId", &buf.DigitalProductId,
|
|
|
|
|
&len) == ERROR_SUCCESS &&
|
2019-11-01 21:25:17 +03:00
|
|
|
|
len > 42 && len < sizeof(buf)) {
|
|
|
|
|
bootid_collect(&bin, &buf.DigitalProductId, len);
|
|
|
|
|
got_machineid = true;
|
|
|
|
|
}
|
|
|
|
|
len = sizeof(buf);
|
|
|
|
|
if (mdbx_RegGetValue(HKEY_LOCAL_MACHINE, HKLM_WindowsNT_DPK,
|
2020-10-08 00:45:31 +03:00
|
|
|
|
"DigitalProductId", &buf.DigitalProductId,
|
|
|
|
|
&len) == ERROR_SUCCESS &&
|
2019-11-01 21:25:17 +03:00
|
|
|
|
len > 42 && len < sizeof(buf)) {
|
|
|
|
|
bootid_collect(&bin, &buf.DigitalProductId, len);
|
|
|
|
|
got_machineid = true;
|
|
|
|
|
}
|
|
|
|
|
len = sizeof(buf);
|
|
|
|
|
if (mdbx_RegGetValue(HKEY_LOCAL_MACHINE, HKLM_WindowsNT_DPK2,
|
2020-10-08 00:45:31 +03:00
|
|
|
|
"DigitalProductId", &buf.DigitalProductId,
|
|
|
|
|
&len) == ERROR_SUCCESS &&
|
2019-11-01 21:25:17 +03:00
|
|
|
|
len > 42 && len < sizeof(buf)) {
|
|
|
|
|
bootid_collect(&bin, &buf.DigitalProductId, len);
|
|
|
|
|
got_machineid = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-19 16:00:40 +03:00
|
|
|
|
static const char HKLM_PrefetcherParams[] =
|
|
|
|
|
"SYSTEM\\CurrentControlSet\\Control\\Session Manager\\Memory "
|
|
|
|
|
"Management\\PrefetchParameters";
|
2019-11-01 21:25:17 +03:00
|
|
|
|
len = sizeof(buf);
|
2020-02-19 16:00:40 +03:00
|
|
|
|
if (mdbx_RegGetValue(HKEY_LOCAL_MACHINE, HKLM_PrefetcherParams, "BootId",
|
2020-10-08 00:45:31 +03:00
|
|
|
|
&buf.BootId, &len) == ERROR_SUCCESS &&
|
2019-11-01 21:25:17 +03:00
|
|
|
|
len > 1 && len < sizeof(buf)) {
|
|
|
|
|
bootid_collect(&bin, &buf.BootId, len);
|
|
|
|
|
got_bootseq = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
len = sizeof(buf);
|
2020-02-19 16:00:40 +03:00
|
|
|
|
if (mdbx_RegGetValue(HKEY_LOCAL_MACHINE, HKLM_PrefetcherParams, "BaseTime",
|
2020-10-08 00:45:31 +03:00
|
|
|
|
&buf.BaseTime, &len) == ERROR_SUCCESS &&
|
2019-11-01 21:25:17 +03:00
|
|
|
|
len >= sizeof(buf.BaseTime) && buf.BaseTime) {
|
|
|
|
|
bootid_collect(&bin, &buf.BaseTime, len);
|
2019-11-09 10:50:43 +03:00
|
|
|
|
got_boottime = true;
|
2019-11-01 21:25:17 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* BootTime from SYSTEM_TIMEOFDAY_INFORMATION */
|
|
|
|
|
NTSTATUS status = NtQuerySystemInformation(
|
|
|
|
|
0x03 /* SystemTmeOfDayInformation */, &buf.SysTimeOfDayInfo,
|
|
|
|
|
sizeof(buf.SysTimeOfDayInfo), &len);
|
|
|
|
|
if (NT_SUCCESS(status) &&
|
2021-07-17 21:00:50 +03:00
|
|
|
|
len >= offsetof(union buf, SysTimeOfDayInfoHacked.BootTimeBias) +
|
|
|
|
|
sizeof(buf.SysTimeOfDayInfoHacked.BootTimeBias) &&
|
2019-11-01 21:25:17 +03:00
|
|
|
|
buf.SysTimeOfDayInfoHacked.BootTime.QuadPart) {
|
2021-07-17 21:00:50 +03:00
|
|
|
|
const uint64_t UnbiasedBootTime =
|
|
|
|
|
buf.SysTimeOfDayInfoHacked.BootTime.QuadPart -
|
|
|
|
|
buf.SysTimeOfDayInfoHacked.BootTimeBias;
|
|
|
|
|
if (UnbiasedBootTime) {
|
|
|
|
|
bootid_collect(&bin, &UnbiasedBootTime, sizeof(UnbiasedBootTime));
|
|
|
|
|
got_boottime = true;
|
|
|
|
|
}
|
2019-11-01 21:25:17 +03:00
|
|
|
|
}
|
|
|
|
|
|
2019-11-09 10:50:43 +03:00
|
|
|
|
if (!got_boottime) {
|
2019-11-01 21:25:17 +03:00
|
|
|
|
uint64_t boottime = windows_bootime();
|
|
|
|
|
if (boottime) {
|
|
|
|
|
bootid_collect(&bin, &boottime, sizeof(boottime));
|
2019-11-09 10:50:43 +03:00
|
|
|
|
got_boottime = true;
|
2019-11-01 21:25:17 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif /* Windows */
|
|
|
|
|
|
|
|
|
|
#if defined(CTL_HW) && defined(HW_UUID)
|
|
|
|
|
if (!got_machineid) {
|
|
|
|
|
static const int mib[] = {CTL_HW, HW_UUID};
|
|
|
|
|
char buf[42];
|
|
|
|
|
size_t len = sizeof(buf);
|
|
|
|
|
if (sysctl(
|
|
|
|
|
#ifdef SYSCTL_LEGACY_NONCONST_MIB
|
|
|
|
|
(int *)
|
|
|
|
|
#endif
|
|
|
|
|
mib,
|
|
|
|
|
ARRAY_LENGTH(mib), &buf, &len, NULL, 0) == 0)
|
|
|
|
|
got_machineid = bootid_parse_uuid(&bin, buf, len);
|
|
|
|
|
}
|
|
|
|
|
#endif /* CTL_HW && HW_UUID */
|
|
|
|
|
|
|
|
|
|
#if defined(CTL_KERN) && defined(KERN_HOSTUUID)
|
|
|
|
|
if (!got_machineid) {
|
|
|
|
|
static const int mib[] = {CTL_KERN, KERN_HOSTUUID};
|
|
|
|
|
char buf[42];
|
|
|
|
|
size_t len = sizeof(buf);
|
|
|
|
|
if (sysctl(
|
|
|
|
|
#ifdef SYSCTL_LEGACY_NONCONST_MIB
|
|
|
|
|
(int *)
|
|
|
|
|
#endif
|
|
|
|
|
mib,
|
|
|
|
|
ARRAY_LENGTH(mib), &buf, &len, NULL, 0) == 0)
|
|
|
|
|
got_machineid = bootid_parse_uuid(&bin, buf, len);
|
|
|
|
|
}
|
|
|
|
|
#endif /* CTL_KERN && KERN_HOSTUUID */
|
|
|
|
|
|
|
|
|
|
#if defined(__NetBSD__)
|
|
|
|
|
if (!got_machineid) {
|
|
|
|
|
char buf[42];
|
|
|
|
|
size_t len = sizeof(buf);
|
|
|
|
|
if (sysctlbyname("machdep.dmi.system-uuid", buf, &len, NULL, 0) == 0)
|
|
|
|
|
got_machineid = bootid_parse_uuid(&bin, buf, len);
|
|
|
|
|
}
|
|
|
|
|
#endif /* __NetBSD__ */
|
|
|
|
|
|
|
|
|
|
#if _XOPEN_SOURCE_EXTENDED
|
|
|
|
|
if (!got_machineid) {
|
|
|
|
|
const int hostid = gethostid();
|
|
|
|
|
if (hostid > 0) {
|
|
|
|
|
bootid_collect(&bin, &hostid, sizeof(hostid));
|
|
|
|
|
got_machineid = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif /* _XOPEN_SOURCE_EXTENDED */
|
|
|
|
|
|
|
|
|
|
if (!got_machineid) {
|
|
|
|
|
lack:
|
|
|
|
|
bin.x = bin.y = 0;
|
|
|
|
|
return bin;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
|
|
#if defined(CTL_KERN) && defined(KERN_BOOTTIME)
|
2019-11-09 10:50:43 +03:00
|
|
|
|
if (!got_boottime) {
|
2019-11-01 21:25:17 +03:00
|
|
|
|
static const int mib[] = {CTL_KERN, KERN_BOOTTIME};
|
|
|
|
|
struct timeval boottime;
|
|
|
|
|
size_t len = sizeof(boottime);
|
|
|
|
|
if (sysctl(
|
|
|
|
|
#ifdef SYSCTL_LEGACY_NONCONST_MIB
|
|
|
|
|
(int *)
|
|
|
|
|
#endif
|
|
|
|
|
mib,
|
|
|
|
|
ARRAY_LENGTH(mib), &boottime, &len, NULL, 0) == 0 &&
|
|
|
|
|
len == sizeof(boottime) && boottime.tv_sec) {
|
|
|
|
|
bootid_collect(&bin, &boottime, len);
|
2019-11-09 10:50:43 +03:00
|
|
|
|
got_boottime = true;
|
2019-11-01 21:25:17 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif /* CTL_KERN && KERN_BOOTTIME */
|
|
|
|
|
|
|
|
|
|
#if defined(__sun) || defined(__SVR4) || defined(__svr4__)
|
2019-11-09 10:50:43 +03:00
|
|
|
|
if (!got_boottime) {
|
2019-11-01 21:25:17 +03:00
|
|
|
|
kstat_ctl_t *kc = kstat_open();
|
|
|
|
|
if (kc) {
|
|
|
|
|
kstat_t *kp = kstat_lookup(kc, "unix", 0, "system_misc");
|
|
|
|
|
if (kp && kstat_read(kc, kp, 0) != -1) {
|
|
|
|
|
kstat_named_t *kn = (kstat_named_t *)kstat_data_lookup(kp, "boot_time");
|
|
|
|
|
if (kn) {
|
|
|
|
|
switch (kn->data_type) {
|
|
|
|
|
case KSTAT_DATA_INT32:
|
|
|
|
|
case KSTAT_DATA_UINT32:
|
|
|
|
|
bootid_collect(&bin, &kn->value, sizeof(int32_t));
|
|
|
|
|
got_boottime = true;
|
|
|
|
|
case KSTAT_DATA_INT64:
|
|
|
|
|
case KSTAT_DATA_UINT64:
|
|
|
|
|
bootid_collect(&bin, &kn->value, sizeof(int64_t));
|
|
|
|
|
got_boottime = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
kstat_close(kc);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif /* SunOS / Solaris */
|
|
|
|
|
|
|
|
|
|
#if _XOPEN_SOURCE_EXTENDED && defined(BOOT_TIME)
|
2019-11-09 10:50:43 +03:00
|
|
|
|
if (!got_boottime) {
|
2019-11-01 21:25:17 +03:00
|
|
|
|
setutxent();
|
|
|
|
|
const struct utmpx id = {.ut_type = BOOT_TIME};
|
|
|
|
|
const struct utmpx *entry = getutxid(&id);
|
|
|
|
|
if (entry) {
|
|
|
|
|
bootid_collect(&bin, entry, sizeof(*entry));
|
2019-11-09 10:50:43 +03:00
|
|
|
|
got_boottime = true;
|
2019-11-01 21:25:17 +03:00
|
|
|
|
while (unlikely((entry = getutxid(&id)) != nullptr)) {
|
|
|
|
|
/* have multiple reboot records, assuming we can distinguish next
|
|
|
|
|
* bootsession even if RTC is wrong or absent */
|
|
|
|
|
bootid_collect(&bin, entry, sizeof(*entry));
|
|
|
|
|
got_bootseq = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
endutxent();
|
|
|
|
|
}
|
|
|
|
|
#endif /* _XOPEN_SOURCE_EXTENDED && BOOT_TIME */
|
|
|
|
|
|
|
|
|
|
if (!got_bootseq) {
|
2019-11-09 10:50:43 +03:00
|
|
|
|
if (!got_boottime || !MDBX_TRUST_RTC)
|
2019-11-01 21:25:17 +03:00
|
|
|
|
goto lack;
|
|
|
|
|
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
FILETIME now;
|
|
|
|
|
GetSystemTimeAsFileTime(&now);
|
|
|
|
|
if (0x1CCCCCC > now.dwHighDateTime)
|
|
|
|
|
#else
|
|
|
|
|
struct timespec mono, real;
|
|
|
|
|
if (clock_gettime(CLOCK_MONOTONIC, &mono) ||
|
|
|
|
|
clock_gettime(CLOCK_REALTIME, &real) ||
|
|
|
|
|
/* wrong time, RTC is mad or absent */
|
|
|
|
|
1555555555l > real.tv_sec ||
|
|
|
|
|
/* seems no adjustment by RTC/NTP, i.e. a fake time */
|
|
|
|
|
real.tv_sec < mono.tv_sec || 1234567890l > real.tv_sec - mono.tv_sec ||
|
|
|
|
|
(real.tv_sec - mono.tv_sec) % 900u == 0)
|
|
|
|
|
#endif
|
|
|
|
|
goto lack;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return bin;
|
|
|
|
|
}
|
2021-04-07 01:45:14 +03:00
|
|
|
|
|
|
|
|
|
__cold int mdbx_get_sysraminfo(intptr_t *page_size, intptr_t *total_pages,
|
|
|
|
|
intptr_t *avail_pages) {
|
|
|
|
|
if (!page_size && !total_pages && !avail_pages)
|
|
|
|
|
return MDBX_EINVAL;
|
|
|
|
|
if (total_pages)
|
|
|
|
|
*total_pages = -1;
|
|
|
|
|
if (avail_pages)
|
|
|
|
|
*avail_pages = -1;
|
|
|
|
|
|
2022-08-11 01:03:15 +03:00
|
|
|
|
const intptr_t pagesize = osal_syspagesize();
|
2021-04-07 01:45:14 +03:00
|
|
|
|
if (page_size)
|
|
|
|
|
*page_size = pagesize;
|
|
|
|
|
if (unlikely(pagesize < MIN_PAGESIZE || !is_powerof2(pagesize)))
|
|
|
|
|
return MDBX_INCOMPATIBLE;
|
|
|
|
|
|
2021-05-11 20:14:09 +03:00
|
|
|
|
MDBX_MAYBE_UNUSED const int log2page = log2n_powerof2(pagesize);
|
2021-04-07 01:45:14 +03:00
|
|
|
|
assert(pagesize == (INT64_C(1) << log2page));
|
|
|
|
|
(void)log2page;
|
|
|
|
|
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
MEMORYSTATUSEX info;
|
|
|
|
|
memset(&info, 0, sizeof(info));
|
|
|
|
|
info.dwLength = sizeof(info);
|
|
|
|
|
if (!GlobalMemoryStatusEx(&info))
|
2021-05-08 10:59:15 +03:00
|
|
|
|
return (int)GetLastError();
|
2021-04-07 01:45:14 +03:00
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
if (total_pages) {
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
const intptr_t total_ram_pages = (intptr_t)(info.ullTotalPhys >> log2page);
|
|
|
|
|
#elif defined(_SC_PHYS_PAGES)
|
|
|
|
|
const intptr_t total_ram_pages = sysconf(_SC_PHYS_PAGES);
|
|
|
|
|
if (total_ram_pages == -1)
|
|
|
|
|
return errno;
|
|
|
|
|
#elif defined(_SC_AIX_REALMEM)
|
|
|
|
|
const intptr_t total_ram_Kb = sysconf(_SC_AIX_REALMEM);
|
|
|
|
|
if (total_ram_Kb == -1)
|
|
|
|
|
return errno;
|
|
|
|
|
const intptr_t total_ram_pages = (total_ram_Kb << 10) >> log2page;
|
|
|
|
|
#elif defined(HW_USERMEM) || defined(HW_PHYSMEM64) || defined(HW_MEMSIZE) || \
|
|
|
|
|
defined(HW_PHYSMEM)
|
|
|
|
|
size_t ram, len = sizeof(ram);
|
|
|
|
|
static const int mib[] = {
|
|
|
|
|
CTL_HW,
|
|
|
|
|
#if defined(HW_USERMEM)
|
|
|
|
|
HW_USERMEM
|
|
|
|
|
#elif defined(HW_PHYSMEM64)
|
|
|
|
|
HW_PHYSMEM64
|
|
|
|
|
#elif defined(HW_MEMSIZE)
|
|
|
|
|
HW_MEMSIZE
|
|
|
|
|
#else
|
|
|
|
|
HW_PHYSMEM
|
|
|
|
|
#endif
|
|
|
|
|
};
|
|
|
|
|
if (sysctl(
|
|
|
|
|
#ifdef SYSCTL_LEGACY_NONCONST_MIB
|
|
|
|
|
(int *)
|
|
|
|
|
#endif
|
|
|
|
|
mib,
|
|
|
|
|
ARRAY_LENGTH(mib), &ram, &len, NULL, 0) != 0)
|
|
|
|
|
return errno;
|
|
|
|
|
if (len != sizeof(ram))
|
|
|
|
|
return MDBX_ENOSYS;
|
|
|
|
|
const intptr_t total_ram_pages = (intptr_t)(ram >> log2page);
|
|
|
|
|
#else
|
|
|
|
|
#error "FIXME: Get User-accessible or physical RAM"
|
|
|
|
|
#endif
|
|
|
|
|
*total_pages = total_ram_pages;
|
|
|
|
|
if (total_ram_pages < 1)
|
|
|
|
|
return MDBX_ENOSYS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (avail_pages) {
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
const intptr_t avail_ram_pages = (intptr_t)(info.ullAvailPhys >> log2page);
|
|
|
|
|
#elif defined(_SC_AVPHYS_PAGES)
|
|
|
|
|
const intptr_t avail_ram_pages = sysconf(_SC_AVPHYS_PAGES);
|
|
|
|
|
if (avail_ram_pages == -1)
|
|
|
|
|
return errno;
|
|
|
|
|
#elif defined(__MACH__)
|
|
|
|
|
mach_msg_type_number_t count = HOST_VM_INFO_COUNT;
|
|
|
|
|
vm_statistics_data_t vmstat;
|
|
|
|
|
mach_port_t mport = mach_host_self();
|
|
|
|
|
kern_return_t kerr = host_statistics(mach_host_self(), HOST_VM_INFO,
|
|
|
|
|
(host_info_t)&vmstat, &count);
|
|
|
|
|
mach_port_deallocate(mach_task_self(), mport);
|
|
|
|
|
if (unlikely(kerr != KERN_SUCCESS))
|
|
|
|
|
return MDBX_ENOSYS;
|
|
|
|
|
const intptr_t avail_ram_pages = vmstat.free_count;
|
|
|
|
|
#elif defined(VM_TOTAL) || defined(VM_METER)
|
|
|
|
|
struct vmtotal info;
|
|
|
|
|
size_t len = sizeof(info);
|
|
|
|
|
static const int mib[] = {
|
|
|
|
|
CTL_VM,
|
|
|
|
|
#if defined(VM_TOTAL)
|
|
|
|
|
VM_TOTAL
|
|
|
|
|
#elif defined(VM_METER)
|
|
|
|
|
VM_METER
|
|
|
|
|
#endif
|
|
|
|
|
};
|
|
|
|
|
if (sysctl(
|
|
|
|
|
#ifdef SYSCTL_LEGACY_NONCONST_MIB
|
|
|
|
|
(int *)
|
|
|
|
|
#endif
|
|
|
|
|
mib,
|
|
|
|
|
ARRAY_LENGTH(mib), &info, &len, NULL, 0) != 0)
|
|
|
|
|
return errno;
|
|
|
|
|
if (len != sizeof(info))
|
|
|
|
|
return MDBX_ENOSYS;
|
|
|
|
|
const intptr_t avail_ram_pages = info.t_free;
|
|
|
|
|
#else
|
|
|
|
|
#error "FIXME: Get Available RAM"
|
|
|
|
|
#endif
|
|
|
|
|
*avail_pages = avail_ram_pages;
|
|
|
|
|
if (avail_ram_pages < 1)
|
|
|
|
|
return MDBX_ENOSYS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return MDBX_SUCCESS;
|
|
|
|
|
}
|