2017-03-16 18:09:27 +03:00
|
|
|
/* https://en.wikipedia.org/wiki/Operating_system_abstraction_layer */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright 2015-2017 Leonid Yuriev <leo@yuriev.ru>
|
|
|
|
* and other libmdbx authors: please see AUTHORS file.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted only as authorized by the OpenLDAP
|
|
|
|
* Public License.
|
|
|
|
*
|
|
|
|
* A copy of this license is available in the file LICENSE in the
|
|
|
|
* top-level directory of the distribution or, alternatively, at
|
|
|
|
* <http://www.OpenLDAP.org/license.html>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "./bits.h"
|
|
|
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
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:
|
|
|
|
return MDB_SUCCESS;
|
|
|
|
case WAIT_FAILED:
|
2017-05-15 12:08:04 +03:00
|
|
|
return mdbx_get_errno_checked();
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif /* _WIN32 || _WIN64 */
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
#ifndef _MSC_VER
|
|
|
|
/* Prototype should match libc runtime. ISO POSIX (2003) & LSB 3.1 */
|
|
|
|
__nothrow __noreturn void __assert_fail(const char *assertion, const char *file,
|
|
|
|
unsigned line, const char *function);
|
|
|
|
#else
|
|
|
|
__extern_C __declspec(dllimport) void __cdecl _assert(char const *message,
|
|
|
|
char const *filename,
|
|
|
|
unsigned line);
|
|
|
|
#endif /* _MSC_VER */
|
|
|
|
|
|
|
|
#ifndef mdbx_assert_fail
|
|
|
|
void __cold mdbx_assert_fail(MDB_env *env, const char *msg, const char *func,
|
|
|
|
int line) {
|
|
|
|
#if MDB_DEBUG
|
|
|
|
if (env && env->me_assert_func) {
|
|
|
|
env->me_assert_func(env, msg, func, line);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
(void)env;
|
|
|
|
#endif /* MDB_DEBUG */
|
|
|
|
|
|
|
|
if (mdbx_debug_logger)
|
|
|
|
mdbx_debug_log(MDBX_DBG_ASSERT, func, line, "assert: %s\n", msg);
|
|
|
|
#ifndef _MSC_VER
|
|
|
|
__assert_fail(msg, "mdbx", line, func);
|
|
|
|
#else
|
|
|
|
_assert(msg, func, line);
|
|
|
|
#endif /* _MSC_VER */
|
|
|
|
}
|
|
|
|
#endif /* mdbx_assert_fail */
|
|
|
|
|
|
|
|
__cold void mdbx_panic(const char *fmt, ...) {
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, fmt);
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
if (IsDebuggerPresent()) {
|
|
|
|
OutputDebugString("\r\n" FIXME "\r\n");
|
|
|
|
FatalExit(ERROR_UNHANDLED_ERROR);
|
|
|
|
}
|
|
|
|
#elif _XOPEN_SOURCE >= 700 || _POSIX_C_SOURCE >= 200809L || \
|
|
|
|
(__GLIBC_PREREQ(1, 0) && !__GLIBC_PREREQ(2, 10) && defined(_GNU_SOURCE))
|
|
|
|
vdprintf(STDERR_FILENO, fmt, ap);
|
|
|
|
#else
|
|
|
|
#error FIXME
|
|
|
|
#endif
|
|
|
|
va_end(ap);
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
#ifndef mdbx_asprintf
|
|
|
|
int mdbx_asprintf(char **strp, const char *fmt, ...) {
|
|
|
|
va_list ap, ones;
|
|
|
|
|
|
|
|
va_start(ap, fmt);
|
|
|
|
va_copy(ones, ap);
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
int needed = _vscprintf(fmt, ap);
|
2017-04-04 15:17:24 +03:00
|
|
|
#elif defined(vsnprintf) || defined(_BSD_SOURCE) || _XOPEN_SOURCE >= 500 || \
|
2017-03-16 18:09:27 +03:00
|
|
|
defined(_ISOC99_SOURCE) || _POSIX_C_SOURCE >= 200112L
|
|
|
|
int needed = vsnprintf(nullptr, 0, fmt, ap);
|
|
|
|
#else
|
|
|
|
#error FIXME
|
|
|
|
#endif
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
if (unlikely(needed < 0 || needed >= INT_MAX)) {
|
|
|
|
*strp = NULL;
|
|
|
|
va_end(ones);
|
|
|
|
return needed;
|
|
|
|
}
|
|
|
|
|
|
|
|
*strp = malloc(needed + 1);
|
|
|
|
if (unlikely(*strp == NULL)) {
|
|
|
|
va_end(ones);
|
2017-04-24 19:37:01 +03:00
|
|
|
SetLastError(MDBX_ENOMEM);
|
|
|
|
return -1;
|
2017-03-16 18:09:27 +03:00
|
|
|
}
|
|
|
|
|
2017-04-04 15:17:24 +03:00
|
|
|
#if defined(vsnprintf) || defined(_BSD_SOURCE) || _XOPEN_SOURCE >= 500 || \
|
2017-03-16 18:09:27 +03:00
|
|
|
defined(_ISOC99_SOURCE) || _POSIX_C_SOURCE >= 200112L
|
|
|
|
int actual = vsnprintf(*strp, needed + 1, fmt, ones);
|
|
|
|
#else
|
|
|
|
#error FIXME
|
|
|
|
#endif
|
|
|
|
va_end(ones);
|
|
|
|
|
|
|
|
assert(actual == needed);
|
|
|
|
if (unlikely(actual < 0)) {
|
|
|
|
free(*strp);
|
|
|
|
*strp = NULL;
|
|
|
|
}
|
|
|
|
return actual;
|
|
|
|
}
|
|
|
|
#endif /* mdbx_asprintf */
|
|
|
|
|
|
|
|
#ifndef mdbx_memalign_alloc
|
|
|
|
int mdbx_memalign_alloc(size_t alignment, size_t bytes, void **result) {
|
|
|
|
#if _MSC_VER
|
|
|
|
*result = _aligned_malloc(bytes, alignment);
|
2017-05-22 19:59:16 +03:00
|
|
|
return *result ? MDB_SUCCESS : MDBX_ENOMEM /* ERROR_OUTOFMEMORY */;
|
2017-03-16 18:09:27 +03:00
|
|
|
#elif __GLIBC_PREREQ(2, 16) || __STDC_VERSION__ >= 201112L
|
|
|
|
*result = memalign(alignment, bytes);
|
|
|
|
return *result ? MDB_SUCCESS : errno;
|
|
|
|
#elif _POSIX_VERSION >= 200112L
|
|
|
|
*result = NULL;
|
|
|
|
return posix_memalign(result, alignment, bytes);
|
|
|
|
#else
|
|
|
|
#error FIXME
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif /* mdbx_memalign_alloc */
|
|
|
|
|
|
|
|
#ifndef mdbx_memalign_free
|
|
|
|
void mdbx_memalign_free(void *ptr) {
|
|
|
|
#if _MSC_VER
|
|
|
|
_aligned_free(ptr);
|
|
|
|
#else
|
|
|
|
free(ptr);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif /* mdbx_memalign_free */
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
2017-05-23 18:40:21 +03:00
|
|
|
int mdbx_condmutex_init(mdbx_condmutex_t *condmutex) {
|
2017-03-16 18:09:27 +03:00
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2017-05-23 18:40:21 +03:00
|
|
|
int rc = MDB_SUCCESS;
|
|
|
|
condmutex->event = NULL;
|
|
|
|
condmutex->mutex = CreateMutex(NULL, FALSE, NULL);
|
|
|
|
if (!condmutex->mutex)
|
|
|
|
return mdbx_get_errno_checked();
|
2017-03-16 18:09:27 +03:00
|
|
|
|
2017-05-23 18:40:21 +03:00
|
|
|
condmutex->event = CreateEvent(NULL, FALSE, FALSE, NULL);
|
|
|
|
if (!condmutex->event) {
|
|
|
|
rc = mdbx_get_errno_checked();
|
|
|
|
(void)CloseHandle(condmutex->mutex);
|
|
|
|
condmutex->mutex = NULL;
|
|
|
|
}
|
|
|
|
return rc;
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
2017-05-23 18:40:21 +03:00
|
|
|
memset(condmutex, 0, sizeof(mdbx_condmutex_t));
|
|
|
|
int rc = pthread_mutex_init(&condmutex->mutex, NULL);
|
|
|
|
if (rc == 0) {
|
|
|
|
rc = pthread_cond_init(&condmutex->cond, NULL);
|
|
|
|
if (rc != 0)
|
|
|
|
(void)pthread_mutex_destroy(&condmutex->mutex);
|
|
|
|
}
|
|
|
|
return rc;
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-05-23 18:40:21 +03:00
|
|
|
static bool is_allzeros(const void *ptr, size_t bytes) {
|
|
|
|
const uint8_t *u8 = ptr;
|
|
|
|
for (size_t i = 0; i < bytes; ++i)
|
|
|
|
if (u8[i] != 0)
|
|
|
|
return false;
|
|
|
|
return true;
|
2017-03-16 18:09:27 +03:00
|
|
|
}
|
|
|
|
|
2017-05-23 18:40:21 +03:00
|
|
|
int mdbx_condmutex_destroy(mdbx_condmutex_t *condmutex) {
|
|
|
|
int rc = MDBX_EINVAL;
|
2017-03-16 18:09:27 +03:00
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2017-05-23 18:40:21 +03:00
|
|
|
if (condmutex->event) {
|
|
|
|
rc = CloseHandle(condmutex->event) ? MDB_SUCCESS : mdbx_get_errno_checked();
|
|
|
|
if (rc == MDB_SUCCESS)
|
|
|
|
condmutex->event = NULL;
|
|
|
|
}
|
|
|
|
if (condmutex->mutex) {
|
|
|
|
rc = CloseHandle(condmutex->mutex) ? MDB_SUCCESS : mdbx_get_errno_checked();
|
|
|
|
if (rc == MDB_SUCCESS)
|
|
|
|
condmutex->mutex = NULL;
|
|
|
|
}
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
2017-05-23 18:40:21 +03:00
|
|
|
if (!is_allzeros(&condmutex->cond, sizeof(condmutex->cond))) {
|
|
|
|
rc = pthread_cond_destroy(&condmutex->cond);
|
|
|
|
if (rc == 0)
|
|
|
|
memset(&condmutex->cond, 0, sizeof(condmutex->cond));
|
|
|
|
}
|
|
|
|
if (!is_allzeros(&condmutex->mutex, sizeof(condmutex->mutex))) {
|
|
|
|
rc = pthread_mutex_destroy(&condmutex->mutex);
|
|
|
|
if (rc == 0)
|
|
|
|
memset(&condmutex->mutex, 0, sizeof(condmutex->mutex));
|
|
|
|
}
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif
|
2017-05-23 18:40:21 +03:00
|
|
|
return rc;
|
2017-03-16 18:09:27 +03:00
|
|
|
}
|
|
|
|
|
2017-05-23 18:40:21 +03:00
|
|
|
int mdbx_condmutex_lock(mdbx_condmutex_t *condmutex) {
|
2017-03-16 18:09:27 +03:00
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2017-05-23 18:40:21 +03:00
|
|
|
DWORD code = WaitForSingleObject(condmutex->mutex, INFINITE);
|
|
|
|
return waitstatus2errcode(code);
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
2017-05-23 18:40:21 +03:00
|
|
|
return pthread_mutex_lock(&condmutex->mutex);
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-05-23 18:40:21 +03:00
|
|
|
int mdbx_condmutex_unlock(mdbx_condmutex_t *condmutex) {
|
2017-03-16 18:09:27 +03:00
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2017-05-23 18:40:21 +03:00
|
|
|
return ReleaseMutex(condmutex->mutex) ? MDB_SUCCESS
|
|
|
|
: mdbx_get_errno_checked();
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
2017-05-23 18:40:21 +03:00
|
|
|
return pthread_mutex_unlock(&condmutex->mutex);
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-05-23 18:40:21 +03:00
|
|
|
int mdbx_condmutex_signal(mdbx_condmutex_t *condmutex) {
|
2017-03-16 18:09:27 +03:00
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2017-05-23 18:40:21 +03:00
|
|
|
return SetEvent(condmutex->event) ? MDB_SUCCESS : mdbx_get_errno_checked();
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
2017-05-23 18:40:21 +03:00
|
|
|
return pthread_cond_signal(&condmutex->cond);
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-05-23 18:40:21 +03:00
|
|
|
int mdbx_condmutex_wait(mdbx_condmutex_t *condmutex) {
|
2017-03-16 18:09:27 +03:00
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2017-05-23 18:40:21 +03:00
|
|
|
DWORD code =
|
|
|
|
SignalObjectAndWait(condmutex->mutex, condmutex->event, INFINITE, FALSE);
|
2017-03-16 18:09:27 +03:00
|
|
|
if (code == WAIT_OBJECT_0)
|
2017-05-23 18:40:21 +03:00
|
|
|
code = WaitForSingleObject(condmutex->mutex, INFINITE);
|
2017-03-30 18:54:57 +03:00
|
|
|
return waitstatus2errcode(code);
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
2017-05-23 18:40:21 +03:00
|
|
|
return pthread_cond_wait(&condmutex->cond, &condmutex->mutex);
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
int mdbx_openfile(const char *pathname, int flags, mode_t mode,
|
|
|
|
mdbx_filehandle_t *fd) {
|
|
|
|
*fd = INVALID_HANDLE_VALUE;
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
(void)mode;
|
|
|
|
|
|
|
|
DWORD DesiredAccess;
|
|
|
|
DWORD ShareMode = FILE_SHARE_READ | FILE_SHARE_WRITE;
|
|
|
|
DWORD FlagsAndAttributes = FILE_ATTRIBUTE_NORMAL;
|
|
|
|
switch (flags & (O_RDONLY | O_WRONLY | O_RDWR)) {
|
|
|
|
default:
|
|
|
|
return ERROR_INVALID_PARAMETER;
|
|
|
|
case O_RDONLY:
|
|
|
|
DesiredAccess = GENERIC_READ;
|
|
|
|
break;
|
|
|
|
case O_WRONLY: /* assume for mdb_env_copy() and friends output */
|
|
|
|
DesiredAccess = GENERIC_WRITE;
|
|
|
|
ShareMode = 0;
|
|
|
|
FlagsAndAttributes |= FILE_FLAG_NO_BUFFERING | FILE_FLAG_WRITE_THROUGH;
|
|
|
|
break;
|
|
|
|
case O_RDWR:
|
|
|
|
DesiredAccess = GENERIC_READ | GENERIC_WRITE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
DWORD CreationDisposition;
|
|
|
|
switch (flags & (O_EXCL | O_CREAT)) {
|
|
|
|
default:
|
|
|
|
return ERROR_INVALID_PARAMETER;
|
|
|
|
case 0:
|
|
|
|
CreationDisposition = OPEN_EXISTING;
|
|
|
|
break;
|
|
|
|
case O_EXCL | O_CREAT:
|
|
|
|
CreationDisposition = CREATE_NEW;
|
|
|
|
FlagsAndAttributes |= FILE_ATTRIBUTE_NOT_CONTENT_INDEXED;
|
|
|
|
break;
|
|
|
|
case O_CREAT:
|
|
|
|
CreationDisposition = OPEN_ALWAYS;
|
|
|
|
FlagsAndAttributes |= FILE_ATTRIBUTE_NOT_CONTENT_INDEXED;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
*fd = CreateFileA(pathname, DesiredAccess, ShareMode, NULL,
|
|
|
|
CreationDisposition, FlagsAndAttributes, NULL);
|
|
|
|
|
|
|
|
if (*fd == INVALID_HANDLE_VALUE)
|
2017-05-15 12:08:04 +03:00
|
|
|
return mdbx_get_errno_checked();
|
|
|
|
if ((flags & O_CREAT) && mdbx_get_errno_checked() != ERROR_ALREADY_EXISTS) {
|
2017-03-16 18:09:27 +03:00
|
|
|
/* set FILE_ATTRIBUTE_NOT_CONTENT_INDEXED for new file */
|
|
|
|
DWORD FileAttributes = GetFileAttributesA(pathname);
|
|
|
|
if (FileAttributes == INVALID_FILE_ATTRIBUTES ||
|
|
|
|
!SetFileAttributesA(pathname, FileAttributes |
|
|
|
|
FILE_ATTRIBUTE_NOT_CONTENT_INDEXED)) {
|
2017-05-15 12:08:04 +03:00
|
|
|
int rc = mdbx_get_errno_checked();
|
2017-03-16 18:09:27 +03:00
|
|
|
CloseHandle(*fd);
|
|
|
|
*fd = INVALID_HANDLE_VALUE;
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
|
|
|
|
#ifdef O_CLOEXEC
|
|
|
|
flags |= O_CLOEXEC;
|
|
|
|
#endif
|
|
|
|
*fd = open(pathname, flags, mode);
|
|
|
|
if (*fd < 0)
|
|
|
|
return errno;
|
|
|
|
#if defined(FD_CLOEXEC) && defined(F_GETFD)
|
|
|
|
flags = fcntl(*fd, F_GETFD);
|
|
|
|
if (flags >= 0)
|
|
|
|
(void)fcntl(*fd, F_SETFD, flags | FD_CLOEXEC);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
return MDB_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
int mdbx_closefile(mdbx_filehandle_t fd) {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2017-05-15 12:08:04 +03:00
|
|
|
return CloseHandle(fd) ? MDB_SUCCESS : mdbx_get_errno_checked();
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
|
|
|
return (close(fd) == 0) ? MDB_SUCCESS : errno;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
int mdbx_pread(mdbx_filehandle_t fd, void *buf, size_t bytes, off_t offset) {
|
|
|
|
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))) {
|
2017-05-15 12:08:04 +03:00
|
|
|
int rc = mdbx_get_errno_checked();
|
2017-04-24 15:51:21 +03:00
|
|
|
return (rc == MDB_SUCCESS) ? /* paranoia */ ERROR_READ_FAULT : rc;
|
2017-03-16 18:09:27 +03:00
|
|
|
}
|
|
|
|
#else
|
|
|
|
ssize_t read = pread(fd, buf, bytes, offset);
|
2017-04-25 19:58:00 +03:00
|
|
|
if (read < 0) {
|
|
|
|
int rc = errno;
|
2017-05-22 19:59:16 +03:00
|
|
|
return (rc == MDB_SUCCESS) ? /* paranoia */ MDBX_EIO : rc;
|
2017-04-25 19:58:00 +03:00
|
|
|
}
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif
|
2017-04-25 19:58:00 +03:00
|
|
|
return (bytes == (size_t)read) ? MDB_SUCCESS : MDBX_ENODATA;
|
2017-03-16 18:09:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
int mdbx_pwrite(mdbx_filehandle_t fd, const void *buf, size_t bytes,
|
|
|
|
off_t offset) {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
if (bytes > MAX_WRITE)
|
|
|
|
return ERROR_INVALID_PARAMETER;
|
|
|
|
|
|
|
|
OVERLAPPED ov;
|
|
|
|
ov.hEvent = 0;
|
|
|
|
ov.Offset = (DWORD)offset;
|
|
|
|
ov.OffsetHigh = HIGH_DWORD(offset);
|
|
|
|
|
|
|
|
DWORD written;
|
|
|
|
if (likely(WriteFile(fd, buf, (DWORD)bytes, &written, &ov)))
|
2017-05-22 19:59:16 +03:00
|
|
|
return (bytes == written) ? MDB_SUCCESS : MDBX_EIO /* ERROR_WRITE_FAULT */;
|
2017-05-15 12:08:04 +03:00
|
|
|
return mdbx_get_errno_checked();
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
|
|
|
int rc;
|
|
|
|
ssize_t written;
|
|
|
|
do {
|
|
|
|
written = pwrite(fd, buf, bytes, offset);
|
|
|
|
if (likely(bytes == (size_t)written))
|
|
|
|
return MDB_SUCCESS;
|
|
|
|
rc = errno;
|
|
|
|
} while (rc == EINTR);
|
2017-05-22 19:59:16 +03:00
|
|
|
return (written < 0) ? rc : MDBX_EIO /* Use which error code (ENOSPC)? */;
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
int mdbx_pwritev(mdbx_filehandle_t fd, struct iovec *iov, int iovcnt,
|
|
|
|
off_t offset, size_t expected_written) {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
size_t written = 0;
|
2017-04-15 22:44:48 +03:00
|
|
|
for (int i = 0; i < iovcnt; ++i) {
|
2017-03-16 18:09:27 +03:00
|
|
|
int rc = mdbx_pwrite(fd, iov[i].iov_base, iov[i].iov_len, offset);
|
|
|
|
if (unlikely(rc != MDB_SUCCESS))
|
|
|
|
return rc;
|
|
|
|
written += iov[i].iov_len;
|
|
|
|
offset += iov[i].iov_len;
|
|
|
|
}
|
2017-05-22 19:59:16 +03:00
|
|
|
return (expected_written == written) ? MDB_SUCCESS
|
|
|
|
: MDBX_EIO /* ERROR_WRITE_FAULT */;
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
|
|
|
int rc;
|
|
|
|
ssize_t written;
|
|
|
|
do {
|
|
|
|
written = pwritev(fd, iov, iovcnt, offset);
|
|
|
|
if (likely(expected_written == (size_t)written))
|
|
|
|
return MDB_SUCCESS;
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
|
|
|
int mdbx_write(mdbx_filehandle_t fd, const void *buf, size_t bytes) {
|
|
|
|
#ifdef SIGPIPE
|
|
|
|
sigset_t set, old;
|
|
|
|
sigemptyset(&set);
|
|
|
|
sigaddset(&set, SIGPIPE);
|
|
|
|
int rc = rc = pthread_sigmask(SIG_BLOCK, &set, &old);
|
|
|
|
if (rc != 0)
|
|
|
|
return rc;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
const char *ptr = buf;
|
|
|
|
for (;;) {
|
|
|
|
size_t chunk = (MAX_WRITE < bytes) ? MAX_WRITE : bytes;
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
DWORD written;
|
|
|
|
if (unlikely(!WriteFile(fd, ptr, (DWORD)chunk, &written, NULL)))
|
2017-05-15 12:08:04 +03:00
|
|
|
return mdbx_get_errno_checked();
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
|
|
|
ssize_t written = write(fd, ptr, chunk);
|
|
|
|
if (written < 0) {
|
|
|
|
int rc = errno;
|
|
|
|
#ifdef SIGPIPE
|
|
|
|
if (rc == EPIPE) {
|
|
|
|
/* Collect the pending SIGPIPE, otherwise at least OS X
|
|
|
|
* gives it to the process on thread-exit (ITS#8504).
|
|
|
|
*/
|
|
|
|
int tmp;
|
|
|
|
sigwait(&set, &tmp);
|
|
|
|
written = 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
pthread_sigmask(SIG_SETMASK, &old, NULL);
|
|
|
|
#endif
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
if (likely(bytes == (size_t)written)) {
|
|
|
|
#ifdef SIGPIPE
|
|
|
|
pthread_sigmask(SIG_SETMASK, &old, NULL);
|
|
|
|
#endif
|
|
|
|
return MDB_SUCCESS;
|
|
|
|
}
|
|
|
|
ptr += written;
|
|
|
|
bytes -= written;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-24 19:37:01 +03:00
|
|
|
int mdbx_filesync(mdbx_filehandle_t fd, bool fullsync) {
|
2017-03-16 18:09:27 +03:00
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2017-04-24 19:37:01 +03:00
|
|
|
(void)fullsync;
|
2017-05-15 12:08:04 +03:00
|
|
|
return FlushFileBuffers(fd) ? MDB_SUCCESS : mdbx_get_errno_checked();
|
2017-03-16 18:09:27 +03:00
|
|
|
#elif __GLIBC_PREREQ(2, 16) || _BSD_SOURCE || _XOPEN_SOURCE || \
|
|
|
|
(__GLIBC_PREREQ(2, 8) && _POSIX_C_SOURCE >= 200112L)
|
2017-04-24 19:37:01 +03:00
|
|
|
for (;;) {
|
2017-03-16 18:09:27 +03:00
|
|
|
#if _POSIX_C_SOURCE >= 199309L || _XOPEN_SOURCE >= 500 || \
|
|
|
|
defined(_POSIX_SYNCHRONIZED_IO)
|
2017-04-24 19:37:01 +03:00
|
|
|
if (!fullsync && fdatasync(fd) == 0)
|
|
|
|
return MDB_SUCCESS;
|
|
|
|
#else
|
|
|
|
(void)fullsync;
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif
|
2017-04-24 19:37:01 +03:00
|
|
|
if (fsync(fd) == 0)
|
|
|
|
return MDB_SUCCESS;
|
|
|
|
int rc = errno;
|
|
|
|
if (rc != EINTR)
|
|
|
|
return rc;
|
|
|
|
}
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
|
|
|
#error FIXME
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
int mdbx_filesize(mdbx_filehandle_t fd, off_t *length) {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
BY_HANDLE_FILE_INFORMATION info;
|
|
|
|
if (!GetFileInformationByHandle(fd, &info))
|
2017-05-15 12:08:04 +03:00
|
|
|
return mdbx_get_errno_checked();
|
2017-03-16 18:09:27 +03:00
|
|
|
*length = info.nFileSizeLow | (uint64_t)info.nFileIndexHigh << 32;
|
|
|
|
#else
|
|
|
|
struct stat st;
|
|
|
|
|
|
|
|
if (fstat(fd, &st))
|
|
|
|
return errno;
|
|
|
|
|
|
|
|
*length = st.st_size;
|
|
|
|
#endif
|
|
|
|
return MDB_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
int mdbx_ftruncate(mdbx_filehandle_t fd, off_t length) {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
LARGE_INTEGER li;
|
|
|
|
li.QuadPart = length;
|
|
|
|
return (SetFilePointerEx(fd, li, NULL, FILE_BEGIN) && SetEndOfFile(fd))
|
|
|
|
? MDB_SUCCESS
|
2017-05-15 12:08:04 +03:00
|
|
|
: mdbx_get_errno_checked();
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
|
|
|
return ftruncate(fd, length) == 0 ? MDB_SUCCESS : errno;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
int mdbx_thread_key_create(mdbx_thread_key_t *key) {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
*key = TlsAlloc();
|
2017-05-15 12:08:04 +03:00
|
|
|
return (*key != TLS_OUT_OF_INDEXES) ? MDB_SUCCESS : mdbx_get_errno_checked();
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
|
|
|
return pthread_key_create(key, mdbx_rthc_dtor);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void mdbx_thread_key_delete(mdbx_thread_key_t key) {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
mdbx_ensure(NULL, TlsFree(key));
|
|
|
|
#else
|
|
|
|
mdbx_ensure(NULL, pthread_key_delete(key) == 0);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void *mdbx_thread_rthc_get(mdbx_thread_key_t key) {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
return TlsGetValue(key);
|
|
|
|
#else
|
|
|
|
return pthread_getspecific(key);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void mdbx_thread_rthc_set(mdbx_thread_key_t key, const void *value) {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
mdbx_ensure(NULL, TlsSetValue(key, (void *)value));
|
|
|
|
#else
|
|
|
|
mdbx_ensure(NULL, pthread_setspecific(key, value) == 0);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
mdbx_tid_t mdbx_thread_self(void) {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
return GetCurrentThreadId();
|
|
|
|
#else
|
|
|
|
return pthread_self();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
int mdbx_thread_create(mdbx_thread_t *thread,
|
|
|
|
THREAD_RESULT(THREAD_CALL *start_routine)(void *),
|
|
|
|
void *arg) {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
*thread = CreateThread(NULL, 0, start_routine, arg, 0, NULL);
|
2017-05-15 12:08:04 +03:00
|
|
|
return *thread ? MDB_SUCCESS : mdbx_get_errno_checked();
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
|
|
|
return pthread_create(thread, NULL, start_routine, arg);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
int mdbx_thread_join(mdbx_thread_t thread) {
|
|
|
|
#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
|
|
|
|
}
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
int mdbx_msync(void *addr, size_t length, int async) {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
if (async)
|
|
|
|
return MDB_SUCCESS;
|
2017-05-15 12:08:04 +03:00
|
|
|
return FlushViewOfFile(addr, length) ? MDB_SUCCESS : mdbx_get_errno_checked();
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
|
|
|
return (msync(addr, length, async ? MS_ASYNC : MS_SYNC) == 0) ? MDB_SUCCESS
|
|
|
|
: errno;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
int mdbx_mremap_size(void **address, size_t old_size, size_t new_size) {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
*address = MAP_FAILED;
|
|
|
|
(void)old_size;
|
|
|
|
(void)new_size;
|
2017-04-24 19:37:01 +03:00
|
|
|
return ERROR_CALL_NOT_IMPLEMENTED;
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
|
|
|
*address = mremap(*address, old_size, new_size, 0, address);
|
|
|
|
return (*address != MAP_FAILED) ? MDB_SUCCESS : errno;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
int mdbx_mmap(void **address, size_t length, int rw, mdbx_filehandle_t fd) {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
HANDLE h = CreateFileMapping(fd, NULL, rw ? PAGE_READWRITE : PAGE_READONLY,
|
|
|
|
HIGH_DWORD(length), (DWORD)length, NULL);
|
|
|
|
if (!h)
|
2017-05-15 12:08:04 +03:00
|
|
|
return mdbx_get_errno_checked();
|
2017-03-16 18:09:27 +03:00
|
|
|
*address = MapViewOfFileEx(h, rw ? FILE_MAP_WRITE : FILE_MAP_READ, 0, 0,
|
|
|
|
length, *address);
|
2017-05-15 12:08:04 +03:00
|
|
|
int rc = (*address != MAP_FAILED) ? MDB_SUCCESS : mdbx_get_errno_checked();
|
2017-03-16 18:09:27 +03:00
|
|
|
CloseHandle(h);
|
|
|
|
return rc;
|
|
|
|
#else
|
2017-04-05 18:19:26 +03:00
|
|
|
*address = mmap(NULL, length, rw ? PROT_READ | PROT_WRITE : PROT_READ,
|
2017-03-16 18:09:27 +03:00
|
|
|
MAP_SHARED, fd, 0);
|
|
|
|
return (*address != MAP_FAILED) ? MDB_SUCCESS : errno;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
int mdbx_munmap(void *address, size_t length) {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
(void)length;
|
2017-05-15 12:08:04 +03:00
|
|
|
return UnmapViewOfFile(address) ? MDB_SUCCESS : mdbx_get_errno_checked();
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
|
|
|
return (munmap(address, length) == 0) ? MDB_SUCCESS : errno;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
int mdbx_mlock(const void *address, size_t length) {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2017-05-15 12:08:04 +03:00
|
|
|
return VirtualLock((void *)address, length) ? MDB_SUCCESS
|
|
|
|
: mdbx_get_errno_checked();
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
|
|
|
return (mlock(address, length) == 0) ? MDB_SUCCESS : errno;
|
|
|
|
#endif
|
|
|
|
}
|
2017-04-27 15:18:33 +03:00
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
__cold void mdbx_osal_jitter(bool tiny) {
|
|
|
|
for (;;) {
|
|
|
|
#if defined(_M_IX86) || defined(_M_X64) || defined(__i386__) || \
|
|
|
|
defined(__x86_64__)
|
|
|
|
const unsigned salt = 277u * (unsigned)__rdtsc();
|
|
|
|
#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
|
|
|
|
}
|
|
|
|
}
|