mdbx: internally split-out mdbx_handle_env_pathname().

Change-Id: Ibe387662d737d4d1e274ac0e126053872be503f9
This commit is contained in:
Leonid Yuriev 2020-10-09 17:51:06 +03:00
parent c4e3b95301
commit cd0c727880

View File

@ -10368,22 +10368,20 @@ __cold int mdbx_env_open_for_recovery(MDBX_env *env, const char *pathname,
0); 0);
} }
__cold int mdbx_env_open(MDBX_env *env, const char *pathname, typedef struct {
MDBX_env_flags_t flags, mdbx_mode_t mode) { void *buffer_for_free;
int rc = check_env(env); char *lck, *dxb;
if (unlikely(rc != MDBX_SUCCESS)) } MDBX_handle_env_pathname;
return rc;
__cold static int mdbx_handle_env_pathname(MDBX_handle_env_pathname *result,
const char *pathname,
MDBX_env_flags_t *flags,
const mdbx_mode_t mode) {
int rc;
memset(result, 0, sizeof(*result));
if (unlikely(!pathname)) if (unlikely(!pathname))
return MDBX_EINVAL; return MDBX_EINVAL;
if (flags & ~ENV_USABLE_FLAGS)
return MDBX_EINVAL;
if (env->me_lazy_fd != INVALID_HANDLE_VALUE ||
(env->me_flags & MDBX_ENV_ACTIVE) != 0)
return MDBX_EPERM;
#if defined(_WIN32) || defined(_WIN64) #if defined(_WIN32) || defined(_WIN64)
const size_t wlen = mbstowcs(nullptr, pathname, INT_MAX); const size_t wlen = mbstowcs(nullptr, pathname, INT_MAX);
if (wlen < 1 || wlen > /* MAX_PATH */ INT16_MAX) if (wlen < 1 || wlen > /* MAX_PATH */ INT16_MAX)
@ -10391,33 +10389,28 @@ __cold int mdbx_env_open(MDBX_env *env, const char *pathname,
wchar_t *const pathnameW = _alloca((wlen + 1) * sizeof(wchar_t)); wchar_t *const pathnameW = _alloca((wlen + 1) * sizeof(wchar_t));
if (wlen != mbstowcs(pathnameW, pathname, wlen + 1)) if (wlen != mbstowcs(pathnameW, pathname, wlen + 1))
return ERROR_INVALID_NAME; return ERROR_INVALID_NAME;
#endif /* Windows */
/* pickup previously mdbx_env_set_flags(),
* but avoid MDBX_UTTERLY_NOSYNC by disjunction */
flags = merge_sync_flags(flags, env->me_flags);
#if defined(_WIN32) || defined(_WIN64)
const DWORD dwAttrib = GetFileAttributesW(pathnameW); const DWORD dwAttrib = GetFileAttributesW(pathnameW);
if (dwAttrib == INVALID_FILE_ATTRIBUTES) { if (dwAttrib == INVALID_FILE_ATTRIBUTES) {
rc = GetLastError(); rc = GetLastError();
if (rc != MDBX_ENOFILE) if (rc != MDBX_ENOFILE)
return rc; return rc;
if (mode == 0 || (flags & MDBX_RDONLY) != 0) if (mode == 0 || (*flags & MDBX_RDONLY) != 0)
/* can't open existing */ /* can't open existing */
return rc; return rc;
/* auto-create directory if requested */ /* auto-create directory if requested */
if ((flags & MDBX_NOSUBDIR) == 0 && !CreateDirectoryW(pathnameW, nullptr)) { if ((*flags & MDBX_NOSUBDIR) == 0 &&
!CreateDirectoryW(pathnameW, nullptr)) {
rc = GetLastError(); rc = GetLastError();
if (rc != ERROR_ALREADY_EXISTS) if (rc != ERROR_ALREADY_EXISTS)
return rc; return rc;
} }
} else { } else {
/* ignore passed MDBX_NOSUBDIR flag and set it automatically */ /* ignore passed MDBX_NOSUBDIR flag and set it automatically */
flags |= MDBX_NOSUBDIR; *flags |= MDBX_NOSUBDIR;
if (dwAttrib & FILE_ATTRIBUTE_DIRECTORY) if (dwAttrib & FILE_ATTRIBUTE_DIRECTORY)
flags -= MDBX_NOSUBDIR; *flags -= MDBX_NOSUBDIR;
} }
#else #else
struct stat st; struct stat st;
@ -10425,7 +10418,7 @@ __cold int mdbx_env_open(MDBX_env *env, const char *pathname,
rc = errno; rc = errno;
if (rc != MDBX_ENOFILE) if (rc != MDBX_ENOFILE)
return rc; return rc;
if (mode == 0 || (flags & MDBX_RDONLY) != 0) if (mode == 0 || (*flags & MDBX_RDONLY) != 0)
/* can't open existing */ /* can't open existing */
return rc; return rc;
@ -10436,41 +10429,67 @@ __cold int mdbx_env_open(MDBX_env *env, const char *pathname,
/* always add read/write/search for owner */ S_IRWXU | /* always add read/write/search for owner */ S_IRWXU |
((mode & S_IRGRP) ? /* +search if readable by group */ S_IXGRP : 0) | ((mode & S_IRGRP) ? /* +search if readable by group */ S_IXGRP : 0) |
((mode & S_IROTH) ? /* +search if readable by others */ S_IXOTH : 0); ((mode & S_IROTH) ? /* +search if readable by others */ S_IXOTH : 0);
if ((flags & MDBX_NOSUBDIR) == 0 && mkdir(pathname, dir_mode)) { if ((*flags & MDBX_NOSUBDIR) == 0 && mkdir(pathname, dir_mode)) {
rc = errno; rc = errno;
if (rc != EEXIST) if (rc != EEXIST)
return rc; return rc;
} }
} else { } else {
/* ignore passed MDBX_NOSUBDIR flag and set it automatically */ /* ignore passed MDBX_NOSUBDIR flag and set it automatically */
flags |= MDBX_NOSUBDIR; *flags |= MDBX_NOSUBDIR;
if (S_ISDIR(st.st_mode)) if (S_ISDIR(st.st_mode))
flags -= MDBX_NOSUBDIR; *flags -= MDBX_NOSUBDIR;
} }
#endif #endif
size_t len_full, len = strlen(pathname); size_t len_full, len = strlen(pathname);
if (flags & MDBX_NOSUBDIR) { if (*flags & MDBX_NOSUBDIR) {
len_full = len + sizeof(MDBX_LOCK_SUFFIX) + len + 1; len_full = len + sizeof(MDBX_LOCK_SUFFIX) + len + 1;
} else { } else {
len_full = len + sizeof(MDBX_LOCKNAME) + len + sizeof(MDBX_DATANAME); len_full = len + sizeof(MDBX_LOCKNAME) + len + sizeof(MDBX_DATANAME);
} }
char *lck_pathname = mdbx_malloc(len_full);
if (!lck_pathname) result->buffer_for_free = mdbx_malloc(len_full);
if (!result->buffer_for_free)
return MDBX_ENOMEM; return MDBX_ENOMEM;
char *dxb_pathname; result->lck = result->buffer_for_free;
if (flags & MDBX_NOSUBDIR) { if (*flags & MDBX_NOSUBDIR) {
dxb_pathname = lck_pathname + len + sizeof(MDBX_LOCK_SUFFIX); result->dxb = result->lck + len + sizeof(MDBX_LOCK_SUFFIX);
sprintf(lck_pathname, "%s" MDBX_LOCK_SUFFIX, pathname); sprintf(result->lck, "%s" MDBX_LOCK_SUFFIX, pathname);
strcpy(dxb_pathname, pathname); strcpy(result->dxb, pathname);
} else { } else {
dxb_pathname = lck_pathname + len + sizeof(MDBX_LOCKNAME); result->dxb = result->lck + len + sizeof(MDBX_LOCKNAME);
sprintf(lck_pathname, "%s" MDBX_LOCKNAME, pathname); sprintf(result->lck, "%s" MDBX_LOCKNAME, pathname);
sprintf(dxb_pathname, "%s" MDBX_DATANAME, pathname); sprintf(result->dxb, "%s" MDBX_DATANAME, pathname);
} }
rc = MDBX_SUCCESS; return MDBX_SUCCESS;
}
__cold int mdbx_env_open(MDBX_env *env, const char *pathname,
MDBX_env_flags_t flags, mdbx_mode_t mode) {
int rc = check_env(env);
if (unlikely(rc != MDBX_SUCCESS))
return rc;
if (flags & ~ENV_USABLE_FLAGS)
return MDBX_EINVAL;
if (env->me_lazy_fd != INVALID_HANDLE_VALUE ||
(env->me_flags & MDBX_ENV_ACTIVE) != 0)
return MDBX_EPERM;
/* pickup previously mdbx_env_set_flags(),
* but avoid MDBX_UTTERLY_NOSYNC by disjunction */
const uint32_t saved_me_flags = env->me_flags;
flags = merge_sync_flags(flags, env->me_flags);
MDBX_handle_env_pathname env_pathname;
rc = mdbx_handle_env_pathname(&env_pathname, pathname, &flags, mode);
if (unlikely(rc != MDBX_SUCCESS))
goto bailout;
if (flags & MDBX_RDONLY) { if (flags & MDBX_RDONLY) {
/* LY: silently ignore irrelevant flags when /* LY: silently ignore irrelevant flags when
* we're only getting read access */ * we're only getting read access */
@ -10498,9 +10517,8 @@ __cold int mdbx_env_open(MDBX_env *env, const char *pathname,
rc = MDBX_ENOMEM; rc = MDBX_ENOMEM;
} }
const uint32_t saved_me_flags = env->me_flags;
env->me_flags = (flags & ~MDBX_FATAL_ERROR) | MDBX_ENV_ACTIVE; env->me_flags = (flags & ~MDBX_FATAL_ERROR) | MDBX_ENV_ACTIVE;
if (rc) if (unlikely(rc != MDBX_SUCCESS))
goto bailout; goto bailout;
env->me_path = mdbx_strdup(pathname); env->me_path = mdbx_strdup(pathname);
@ -10516,20 +10534,20 @@ __cold int mdbx_env_open(MDBX_env *env, const char *pathname,
rc = mdbx_openfile(F_ISSET(flags, MDBX_RDONLY) ? MDBX_OPEN_DXB_READ rc = mdbx_openfile(F_ISSET(flags, MDBX_RDONLY) ? MDBX_OPEN_DXB_READ
: MDBX_OPEN_DXB_LAZY, : MDBX_OPEN_DXB_LAZY,
env, dxb_pathname, &env->me_lazy_fd, mode); env, env_pathname.dxb, &env->me_lazy_fd, mode);
if (rc != MDBX_SUCCESS) if (rc != MDBX_SUCCESS)
goto bailout; goto bailout;
mdbx_assert(env, env->me_dsync_fd == INVALID_HANDLE_VALUE); mdbx_assert(env, env->me_dsync_fd == INVALID_HANDLE_VALUE);
if ((flags & (MDBX_RDONLY | MDBX_SAFE_NOSYNC | MDBX_NOMETASYNC)) == 0) { if ((flags & (MDBX_RDONLY | MDBX_SAFE_NOSYNC | MDBX_NOMETASYNC)) == 0) {
rc = mdbx_openfile(MDBX_OPEN_DXB_DSYNC, env, dxb_pathname, rc = mdbx_openfile(MDBX_OPEN_DXB_DSYNC, env, env_pathname.dxb,
&env->me_dsync_fd, 0); &env->me_dsync_fd, 0);
mdbx_ensure(env, (rc != MDBX_SUCCESS) == mdbx_ensure(env, (rc != MDBX_SUCCESS) ==
(env->me_dsync_fd == INVALID_HANDLE_VALUE)); (env->me_dsync_fd == INVALID_HANDLE_VALUE));
} }
#if MDBX_LOCKING == MDBX_LOCKING_SYSV #if MDBX_LOCKING == MDBX_LOCKING_SYSV
env->me_sysv_ipc.key = ftok(dxb_pathname, 42); env->me_sysv_ipc.key = ftok(env_pathname.dxb, 42);
if (env->me_sysv_ipc.key == -1) { if (env->me_sysv_ipc.key == -1) {
rc = errno; rc = errno;
goto bailout; goto bailout;
@ -10539,6 +10557,7 @@ __cold int mdbx_env_open(MDBX_env *env, const char *pathname,
#if !(defined(_WIN32) || defined(_WIN64)) #if !(defined(_WIN32) || defined(_WIN64))
if (mode == 0) { if (mode == 0) {
/* pickup mode for lck-file */ /* pickup mode for lck-file */
struct stat st;
if (fstat(env->me_lazy_fd, &st)) { if (fstat(env->me_lazy_fd, &st)) {
rc = errno; rc = errno;
goto bailout; goto bailout;
@ -10551,7 +10570,7 @@ __cold int mdbx_env_open(MDBX_env *env, const char *pathname,
((mode & S_IRGRP) ? /* +write if readable by group */ S_IWGRP : 0) | ((mode & S_IRGRP) ? /* +write if readable by group */ S_IWGRP : 0) |
((mode & S_IROTH) ? /* +write if readable by others */ S_IWOTH : 0); ((mode & S_IROTH) ? /* +write if readable by others */ S_IWOTH : 0);
#endif /* !Windows */ #endif /* !Windows */
const int lck_rc = mdbx_setup_lck(env, lck_pathname, mode); const int lck_rc = mdbx_setup_lck(env, env_pathname.lck, mode);
if (MDBX_IS_ERROR(lck_rc)) { if (MDBX_IS_ERROR(lck_rc)) {
rc = lck_rc; rc = lck_rc;
goto bailout; goto bailout;
@ -10686,7 +10705,7 @@ bailout:
mdbx_txn_valgrind(env, nullptr); mdbx_txn_valgrind(env, nullptr);
#endif #endif
} }
mdbx_free(lck_pathname); mdbx_free(env_pathname.buffer_for_free);
return rc; return rc;
} }