mirror of
https://github.com/isar/libmdbx.git
synced 2025-01-10 17:24:14 +08:00
mdbx: backport - refine mdbx_env_copy() internals (required for next patch).
Change-Id: I9e8f0dc87398564524a5ec98eda2cb9bde100909
This commit is contained in:
parent
affd28654c
commit
86e63f0b6b
267
src/mdbx.c
267
src/mdbx.c
@ -1,4 +1,4 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright 2015-2018 Leonid Yuriev <leo@yuriev.ru>
|
* Copyright 2015-2018 Leonid Yuriev <leo@yuriev.ru>
|
||||||
* and other libmdbx authors: please see AUTHORS file.
|
* and other libmdbx authors: please see AUTHORS file.
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
@ -10576,21 +10576,51 @@ done:
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Copy environment with compaction. */
|
/* Copy environment with compaction. */
|
||||||
static int __cold mdbx_env_compact(MDBX_env *env, mdbx_filehandle_t fd) {
|
static int __cold mdbx_env_compact(MDBX_env *env, MDBX_txn *read_txn,
|
||||||
MDBX_txn *txn = NULL;
|
mdbx_filehandle_t fd, uint8_t *buffer) {
|
||||||
mdbx_thread_t thr;
|
MDBX_page *const meta = mdbx_init_metas(env, buffer);
|
||||||
mdbx_copy ctx;
|
/* copy canary sequenses if present */
|
||||||
memset(&ctx, 0, sizeof(ctx));
|
if (read_txn->mt_canary.v) {
|
||||||
|
meta->mp_meta.mm_canary = read_txn->mt_canary;
|
||||||
|
meta->mp_meta.mm_canary.v = mdbx_meta_txnid_stable(env, &meta->mp_meta);
|
||||||
|
}
|
||||||
|
|
||||||
int rc = mdbx_condmutex_init(&ctx.mc_condmutex);
|
/* Set metapage 1 with current main DB */
|
||||||
|
pgno_t new_root, root = read_txn->mt_dbs[MAIN_DBI].md_root;
|
||||||
|
if ((new_root = root) == P_INVALID) {
|
||||||
|
/* When the DB is empty, handle it specially to
|
||||||
|
* fix any breakage like page leaks from ITS#8174. */
|
||||||
|
meta->mp_meta.mm_dbs[MAIN_DBI].md_flags =
|
||||||
|
read_txn->mt_dbs[MAIN_DBI].md_flags;
|
||||||
|
} else {
|
||||||
|
/* Count free pages + freeDB pages. Subtract from last_pg
|
||||||
|
* to find the new last_pg, which also becomes the new root. */
|
||||||
|
pgno_t freecount = 0;
|
||||||
|
MDBX_cursor mc;
|
||||||
|
MDBX_val key, data;
|
||||||
|
|
||||||
|
int rc = mdbx_cursor_init(&mc, read_txn, FREE_DBI, NULL);
|
||||||
if (unlikely(rc != MDBX_SUCCESS))
|
if (unlikely(rc != MDBX_SUCCESS))
|
||||||
return rc;
|
return rc;
|
||||||
|
while ((rc = mdbx_cursor_get(&mc, &key, &data, MDBX_NEXT)) == 0)
|
||||||
|
freecount += *(pgno_t *)data.iov_base;
|
||||||
|
if (unlikely(rc != MDBX_NOTFOUND))
|
||||||
|
return rc;
|
||||||
|
|
||||||
const size_t buffer_size = pgno2bytes(env, NUM_METAS) + MDBX_WBUF * 2;
|
freecount += read_txn->mt_dbs[FREE_DBI].md_branch_pages +
|
||||||
uint8_t *buffer = NULL;
|
read_txn->mt_dbs[FREE_DBI].md_leaf_pages +
|
||||||
rc = mdbx_memalign_alloc(env->me_os_psize, buffer_size, (void **)&buffer);
|
read_txn->mt_dbs[FREE_DBI].md_overflow_pages;
|
||||||
|
|
||||||
|
new_root = read_txn->mt_next_pgno - 1 - freecount;
|
||||||
|
meta->mp_meta.mm_geo.next = new_root + 1;
|
||||||
|
meta->mp_meta.mm_dbs[MAIN_DBI] = read_txn->mt_dbs[MAIN_DBI];
|
||||||
|
meta->mp_meta.mm_dbs[MAIN_DBI].md_root = new_root;
|
||||||
|
|
||||||
|
mdbx_copy ctx;
|
||||||
|
memset(&ctx, 0, sizeof(ctx));
|
||||||
|
rc = mdbx_condmutex_init(&ctx.mc_condmutex);
|
||||||
if (unlikely(rc != MDBX_SUCCESS))
|
if (unlikely(rc != MDBX_SUCCESS))
|
||||||
goto done;
|
return rc;
|
||||||
|
|
||||||
ctx.mc_wbuf[0] = buffer + pgno2bytes(env, NUM_METAS);
|
ctx.mc_wbuf[0] = buffer + pgno2bytes(env, NUM_METAS);
|
||||||
memset(ctx.mc_wbuf[0], 0, MDBX_WBUF * 2);
|
memset(ctx.mc_wbuf[0], 0, MDBX_WBUF * 2);
|
||||||
@ -10598,162 +10628,163 @@ static int __cold mdbx_env_compact(MDBX_env *env, mdbx_filehandle_t fd) {
|
|||||||
ctx.mc_next_pgno = NUM_METAS;
|
ctx.mc_next_pgno = NUM_METAS;
|
||||||
ctx.mc_env = env;
|
ctx.mc_env = env;
|
||||||
ctx.mc_fd = fd;
|
ctx.mc_fd = fd;
|
||||||
rc = mdbx_thread_create(&thr, mdbx_env_copythr, &ctx);
|
ctx.mc_txn = read_txn;
|
||||||
if (unlikely(rc != MDBX_SUCCESS))
|
|
||||||
goto done;
|
|
||||||
|
|
||||||
rc = mdbx_txn_begin(env, NULL, MDBX_RDONLY, &txn);
|
mdbx_thread_t thread;
|
||||||
if (unlikely(rc != MDBX_SUCCESS))
|
int thread_err = mdbx_thread_create(&thread, mdbx_env_copythr, &ctx);
|
||||||
goto finish;
|
if (likely(thread_err == MDBX_SUCCESS)) {
|
||||||
|
rc = mdbx_env_cwalk(&ctx, &root, 0);
|
||||||
MDBX_page *const meta = mdbx_init_metas(env, buffer);
|
mdbx_env_cthr_toggle(&ctx, 1 | MDBX_EOF);
|
||||||
|
thread_err = mdbx_thread_join(thread);
|
||||||
/* Set metapage 1 with current main DB */
|
mdbx_condmutex_destroy(&ctx.mc_condmutex);
|
||||||
pgno_t new_root, root = txn->mt_dbs[MAIN_DBI].md_root;
|
}
|
||||||
if ((new_root = root) != P_INVALID) {
|
if (unlikely(thread_err != MDBX_SUCCESS))
|
||||||
/* Count free pages + freeDB pages. Subtract from last_pg
|
return thread_err;
|
||||||
* to find the new last_pg, which also becomes the new root. */
|
|
||||||
pgno_t freecount = 0;
|
|
||||||
MDBX_cursor mc;
|
|
||||||
MDBX_val key, data;
|
|
||||||
|
|
||||||
rc = mdbx_cursor_init(&mc, txn, FREE_DBI, NULL);
|
|
||||||
if (unlikely(rc != MDBX_SUCCESS))
|
if (unlikely(rc != MDBX_SUCCESS))
|
||||||
return rc;
|
return rc;
|
||||||
while ((rc = mdbx_cursor_get(&mc, &key, &data, MDBX_NEXT)) == 0)
|
if (unlikely(ctx.mc_error != MDBX_SUCCESS))
|
||||||
freecount += *(pgno_t *)data.iov_base;
|
return ctx.mc_error;
|
||||||
if (unlikely(rc != MDBX_NOTFOUND))
|
|
||||||
goto finish;
|
|
||||||
|
|
||||||
freecount += txn->mt_dbs[FREE_DBI].md_branch_pages +
|
|
||||||
txn->mt_dbs[FREE_DBI].md_leaf_pages +
|
|
||||||
txn->mt_dbs[FREE_DBI].md_overflow_pages;
|
|
||||||
|
|
||||||
new_root = txn->mt_next_pgno - 1 - freecount;
|
|
||||||
meta->mp_meta.mm_geo.next = meta->mp_meta.mm_geo.now = new_root + 1;
|
|
||||||
meta->mp_meta.mm_dbs[MAIN_DBI] = txn->mt_dbs[MAIN_DBI];
|
|
||||||
meta->mp_meta.mm_dbs[MAIN_DBI].md_root = new_root;
|
|
||||||
} else {
|
|
||||||
/* When the DB is empty, handle it specially to
|
|
||||||
* fix any breakage like page leaks from ITS#8174. */
|
|
||||||
meta->mp_meta.mm_dbs[MAIN_DBI].md_flags = txn->mt_dbs[MAIN_DBI].md_flags;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* copy canary sequenses if present */
|
|
||||||
if (txn->mt_canary.v) {
|
|
||||||
meta->mp_meta.mm_canary = txn->mt_canary;
|
|
||||||
meta->mp_meta.mm_canary.v = mdbx_meta_txnid_stable(env, &meta->mp_meta);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* update signature */
|
|
||||||
meta->mp_meta.mm_datasync_sign = mdbx_meta_sign(&meta->mp_meta);
|
|
||||||
|
|
||||||
ctx.mc_wlen[0] = pgno2bytes(env, NUM_METAS);
|
|
||||||
ctx.mc_txn = txn;
|
|
||||||
rc = mdbx_env_cwalk(&ctx, &root, 0);
|
|
||||||
if (rc == MDBX_SUCCESS && root != new_root) {
|
|
||||||
if (root > new_root) {
|
if (root > new_root) {
|
||||||
mdbx_error("post-compactification root %" PRIaPGNO
|
mdbx_error("post-compactification root %" PRIaPGNO
|
||||||
" GT expected %" PRIaPGNO " (source DB corrupted)",
|
" GT expected %" PRIaPGNO " (source DB corrupted)",
|
||||||
root, new_root);
|
root, new_root);
|
||||||
rc = MDBX_CORRUPTED; /* page leak or corrupt DB */
|
return MDBX_CORRUPTED; /* page leak or corrupt DB */
|
||||||
} else {
|
}
|
||||||
mdbx_error("post-compactification root %" PRIaPGNO
|
if (root < new_root) {
|
||||||
|
mdbx_notice("post-compactification root %" PRIaPGNO
|
||||||
" LT expected %" PRIaPGNO " (page leak(s) in source DB)",
|
" LT expected %" PRIaPGNO " (page leak(s) in source DB)",
|
||||||
root, new_root);
|
root, new_root);
|
||||||
/* fixup and rewrite metas */
|
/* fixup meta */
|
||||||
meta->mp_meta.mm_dbs[MAIN_DBI].md_root = root;
|
meta->mp_meta.mm_dbs[MAIN_DBI].md_root = root;
|
||||||
meta->mp_meta.mm_geo.next = meta->mp_meta.mm_geo.now = root + 1;
|
meta->mp_meta.mm_geo.next = root + 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* update signature */
|
||||||
meta->mp_meta.mm_datasync_sign = mdbx_meta_sign(&meta->mp_meta);
|
meta->mp_meta.mm_datasync_sign = mdbx_meta_sign(&meta->mp_meta);
|
||||||
rc = mdbx_pwrite(fd, buffer, pgno2bytes(env, NUM_METAS), 0);
|
return MDBX_SUCCESS;
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
finish:
|
|
||||||
if (rc != MDBX_SUCCESS)
|
|
||||||
ctx.mc_error = rc;
|
|
||||||
mdbx_env_cthr_toggle(&ctx, 1 | MDBX_EOF);
|
|
||||||
rc = mdbx_thread_join(thr);
|
|
||||||
mdbx_txn_abort(txn);
|
|
||||||
|
|
||||||
done:
|
|
||||||
mdbx_memalign_free(buffer);
|
|
||||||
mdbx_condmutex_destroy(&ctx.mc_condmutex);
|
|
||||||
return rc ? rc : ctx.mc_error;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Copy environment as-is. */
|
/* Copy environment as-is. */
|
||||||
static int __cold mdbx_env_copy_asis(MDBX_env *env, mdbx_filehandle_t fd) {
|
static int __cold mdbx_env_copy_asis(MDBX_env *env, MDBX_txn *read_txn,
|
||||||
MDBX_txn *txn = NULL;
|
mdbx_filehandle_t fd, uint8_t *buffer) {
|
||||||
|
/* We must start the actual read txn after blocking writers */
|
||||||
/* Do the lock/unlock of the reader mutex before starting the
|
int rc = mdbx_txn_end(read_txn, MDBX_END_RESET_TMP);
|
||||||
* write txn. Otherwise other read txns could block writers. */
|
|
||||||
int rc = mdbx_txn_begin(env, NULL, MDBX_RDONLY, &txn);
|
|
||||||
if (unlikely(rc != MDBX_SUCCESS))
|
if (unlikely(rc != MDBX_SUCCESS))
|
||||||
return rc;
|
return rc;
|
||||||
|
|
||||||
/* We must start the actual read txn after blocking writers */
|
|
||||||
rc = mdbx_txn_end(txn, MDBX_END_RESET_TMP);
|
|
||||||
if (unlikely(rc != MDBX_SUCCESS))
|
|
||||||
goto bailout; /* FIXME: or just return? */
|
|
||||||
|
|
||||||
/* Temporarily block writers until we snapshot the meta pages */
|
/* Temporarily block writers until we snapshot the meta pages */
|
||||||
rc = mdbx_txn_lock(env, false);
|
rc = mdbx_txn_lock(env, false);
|
||||||
if (unlikely(rc != MDBX_SUCCESS))
|
if (unlikely(rc != MDBX_SUCCESS))
|
||||||
goto bailout;
|
return rc;
|
||||||
|
|
||||||
rc = mdbx_txn_renew0(txn, MDBX_RDONLY);
|
rc = mdbx_txn_renew0(read_txn, MDBX_RDONLY);
|
||||||
if (unlikely(rc != MDBX_SUCCESS)) {
|
if (unlikely(rc != MDBX_SUCCESS)) {
|
||||||
mdbx_txn_unlock(env);
|
mdbx_txn_unlock(env);
|
||||||
goto bailout;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
rc = mdbx_write(fd, env->me_map, pgno2bytes(env, NUM_METAS));
|
/* Make a snapshot of meta-pages,
|
||||||
MDBX_meta *const head = mdbx_meta_head(env);
|
* but writing ones after the data was flushed */
|
||||||
|
memcpy(buffer, env->me_map, pgno2bytes(env, NUM_METAS));
|
||||||
|
MDBX_meta *const headcopy = /* LY: get pointer to the spanshot copy */
|
||||||
|
(MDBX_meta *)(buffer + ((uint8_t *)mdbx_meta_head(env) - env->me_map));
|
||||||
const uint64_t size =
|
const uint64_t size =
|
||||||
mdbx_roundup2(pgno2bytes(env, head->mm_geo.now), env->me_os_psize);
|
mdbx_roundup2(pgno2bytes(env, headcopy->mm_geo.now), env->me_os_psize);
|
||||||
mdbx_txn_unlock(env);
|
mdbx_txn_unlock(env);
|
||||||
|
|
||||||
if (likely(rc == MDBX_SUCCESS))
|
/* Update signature to steady */
|
||||||
rc = mdbx_write(fd, env->me_map + pgno2bytes(env, NUM_METAS),
|
headcopy->mm_datasync_sign = mdbx_meta_sign(headcopy);
|
||||||
pgno2bytes(env, txn->mt_next_pgno - NUM_METAS));
|
|
||||||
|
/* Copy the data */
|
||||||
|
rc = mdbx_pwrite(fd, env->me_map + pgno2bytes(env, NUM_METAS),
|
||||||
|
pgno2bytes(env, read_txn->mt_next_pgno - NUM_METAS),
|
||||||
|
pgno2bytes(env, NUM_METAS));
|
||||||
|
|
||||||
if (likely(rc == MDBX_SUCCESS))
|
if (likely(rc == MDBX_SUCCESS))
|
||||||
rc = mdbx_ftruncate(fd, size);
|
rc = mdbx_ftruncate(fd, size);
|
||||||
|
|
||||||
bailout:
|
|
||||||
mdbx_txn_abort(txn);
|
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
int __cold mdbx_env_copy2fd(MDBX_env *env, mdbx_filehandle_t fd,
|
int __cold mdbx_env_copy2fd(MDBX_env *env, mdbx_filehandle_t fd,
|
||||||
unsigned flags) {
|
unsigned flags) {
|
||||||
if (flags & MDBX_CP_COMPACT)
|
if (unlikely(!env))
|
||||||
return mdbx_env_compact(env, fd);
|
return MDBX_EINVAL;
|
||||||
|
|
||||||
return mdbx_env_copy_asis(env, fd);
|
if (unlikely(env->me_signature != MDBX_ME_SIGNATURE))
|
||||||
|
return MDBX_EBADSIGN;
|
||||||
|
|
||||||
|
int rc = mdbx_fseek(fd, 0);
|
||||||
|
if (unlikely(rc != MDBX_SUCCESS))
|
||||||
|
return rc;
|
||||||
|
|
||||||
|
const size_t buffer_size = pgno2bytes(env, NUM_METAS) +
|
||||||
|
((flags & MDBX_CP_COMPACT) ? MDBX_WBUF * 2 : 0);
|
||||||
|
uint8_t *buffer = NULL;
|
||||||
|
rc = mdbx_memalign_alloc(env->me_os_psize, buffer_size, (void **)&buffer);
|
||||||
|
if (unlikely(rc != MDBX_SUCCESS))
|
||||||
|
return rc;
|
||||||
|
|
||||||
|
MDBX_txn *read_txn = NULL;
|
||||||
|
/* Do the lock/unlock of the reader mutex before starting the
|
||||||
|
* write txn. Otherwise other read txns could block writers. */
|
||||||
|
rc = mdbx_txn_begin(env, NULL, MDBX_RDONLY, &read_txn);
|
||||||
|
if (unlikely(rc != MDBX_SUCCESS)) {
|
||||||
|
mdbx_memalign_free(buffer);
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Firstly write a stub to meta-pages.
|
||||||
|
* Now we sure to incomplete copy will not be used. */
|
||||||
|
memset(buffer, -1, pgno2bytes(env, NUM_METAS));
|
||||||
|
rc = mdbx_write(fd, buffer, pgno2bytes(env, NUM_METAS));
|
||||||
|
if (likely(rc == MDBX_SUCCESS)) {
|
||||||
|
memset(buffer, 0, pgno2bytes(env, NUM_METAS));
|
||||||
|
rc = (flags & MDBX_CP_COMPACT)
|
||||||
|
? mdbx_env_compact(env, read_txn, fd, buffer)
|
||||||
|
: mdbx_env_copy_asis(env, read_txn, fd, buffer);
|
||||||
|
}
|
||||||
|
mdbx_txn_abort(read_txn);
|
||||||
|
|
||||||
|
if (likely(rc == MDBX_SUCCESS))
|
||||||
|
rc = mdbx_filesync(fd, true);
|
||||||
|
|
||||||
|
/* Write actual meta */
|
||||||
|
if (likely(rc == MDBX_SUCCESS))
|
||||||
|
rc = mdbx_pwrite(fd, buffer, pgno2bytes(env, NUM_METAS), 0);
|
||||||
|
|
||||||
|
mdbx_memalign_free(buffer);
|
||||||
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
int __cold mdbx_env_copy(MDBX_env *env, const char *path, unsigned flags) {
|
int __cold mdbx_env_copy(MDBX_env *env, const char *dest_path, unsigned flags) {
|
||||||
char *lck_pathname;
|
if (unlikely(!env || !dest_path))
|
||||||
|
return MDBX_EINVAL;
|
||||||
|
|
||||||
|
if (unlikely(env->me_signature != MDBX_ME_SIGNATURE))
|
||||||
|
return MDBX_EBADSIGN;
|
||||||
|
|
||||||
|
char *dxb_pathname;
|
||||||
mdbx_filehandle_t newfd = INVALID_HANDLE_VALUE;
|
mdbx_filehandle_t newfd = INVALID_HANDLE_VALUE;
|
||||||
|
|
||||||
if (env->me_flags & MDBX_NOSUBDIR) {
|
if (env->me_flags & MDBX_NOSUBDIR) {
|
||||||
lck_pathname = (char *)path;
|
dxb_pathname = (char *)dest_path;
|
||||||
} else {
|
} else {
|
||||||
size_t len = strlen(path);
|
size_t len = strlen(dest_path);
|
||||||
len += sizeof(MDBX_DATANAME);
|
len += sizeof(MDBX_DATANAME);
|
||||||
lck_pathname = malloc(len);
|
dxb_pathname = malloc(len);
|
||||||
if (!lck_pathname)
|
if (!dxb_pathname)
|
||||||
return MDBX_ENOMEM;
|
return MDBX_ENOMEM;
|
||||||
sprintf(lck_pathname, "%s" MDBX_DATANAME, path);
|
sprintf(dxb_pathname, "%s" MDBX_DATANAME, dest_path);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* The destination path must exist, but the destination file must not.
|
/* The destination path must exist, but the destination file must not.
|
||||||
* We don't want the OS to cache the writes, since the source data is
|
* We don't want the OS to cache the writes, since the source data is
|
||||||
* already in the OS cache. */
|
* already in the OS cache. */
|
||||||
int rc =
|
int rc =
|
||||||
mdbx_openfile(lck_pathname, O_WRONLY | O_CREAT | O_EXCL, 0666, &newfd);
|
mdbx_openfile(dxb_pathname, O_WRONLY | O_CREAT | O_EXCL, 0640, &newfd);
|
||||||
if (rc == MDBX_SUCCESS) {
|
if (rc == MDBX_SUCCESS) {
|
||||||
if (env->me_psize >= env->me_os_psize) {
|
if (env->me_psize >= env->me_os_psize) {
|
||||||
#ifdef F_NOCACHE /* __APPLE__ */
|
#ifdef F_NOCACHE /* __APPLE__ */
|
||||||
@ -10767,15 +10798,17 @@ int __cold mdbx_env_copy(MDBX_env *env, const char *path, unsigned flags) {
|
|||||||
rc = mdbx_env_copy2fd(env, newfd, flags);
|
rc = mdbx_env_copy2fd(env, newfd, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!(env->me_flags & MDBX_NOSUBDIR))
|
|
||||||
free(lck_pathname);
|
|
||||||
|
|
||||||
if (newfd != INVALID_HANDLE_VALUE) {
|
if (newfd != INVALID_HANDLE_VALUE) {
|
||||||
int err = mdbx_closefile(newfd);
|
int err = mdbx_closefile(newfd);
|
||||||
if (rc == MDBX_SUCCESS && err != rc)
|
if (rc == MDBX_SUCCESS && err != rc)
|
||||||
rc = err;
|
rc = err;
|
||||||
|
if (rc != MDBX_SUCCESS)
|
||||||
|
(void)mdbx_removefile(dxb_pathname);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (dxb_pathname != dest_path)
|
||||||
|
free(dxb_pathname);
|
||||||
|
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
20
src/osal.c
20
src/osal.c
@ -399,6 +399,13 @@ int mdbx_fastmutex_release(mdbx_fastmutex_t *fastmutex) {
|
|||||||
|
|
||||||
/*----------------------------------------------------------------------------*/
|
/*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
int mdbx_removefile(const char *pathname) {
|
||||||
|
#if defined(_WIN32) || defined(_WIN64)
|
||||||
|
return DeleteFileA(pathname) ? MDBX_SUCCESS : GetLastError();
|
||||||
|
#else
|
||||||
|
return unlink(pathname) ? errno : MDBX_SUCCESS;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
int mdbx_openfile(const char *pathname, int flags, mode_t mode,
|
int mdbx_openfile(const char *pathname, int flags, mode_t mode,
|
||||||
mdbx_filehandle_t *fd) {
|
mdbx_filehandle_t *fd) {
|
||||||
*fd = INVALID_HANDLE_VALUE;
|
*fd = INVALID_HANDLE_VALUE;
|
||||||
@ -698,6 +705,19 @@ int mdbx_ftruncate(mdbx_filehandle_t fd, uint64_t length) {
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int mdbx_fseek(mdbx_filehandle_t fd, uint64_t pos) {
|
||||||
|
#if defined(_WIN32) || defined(_WIN64)
|
||||||
|
LARGE_INTEGER li;
|
||||||
|
li.QuadPart = pos;
|
||||||
|
return SetFilePointerEx(fd, li, NULL, FILE_BEGIN) ? MDBX_SUCCESS
|
||||||
|
: GetLastError();
|
||||||
|
#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
|
||||||
|
}
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------*/
|
/*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
int mdbx_thread_create(mdbx_thread_t *thread,
|
int mdbx_thread_create(mdbx_thread_t *thread,
|
||||||
|
@ -473,10 +473,12 @@ int mdbx_thread_join(mdbx_thread_t thread);
|
|||||||
int mdbx_filesync(mdbx_filehandle_t fd, bool fullsync);
|
int mdbx_filesync(mdbx_filehandle_t fd, bool fullsync);
|
||||||
int mdbx_filesize_sync(mdbx_filehandle_t fd);
|
int mdbx_filesize_sync(mdbx_filehandle_t fd);
|
||||||
int mdbx_ftruncate(mdbx_filehandle_t fd, uint64_t length);
|
int mdbx_ftruncate(mdbx_filehandle_t fd, uint64_t length);
|
||||||
|
int mdbx_fseek(mdbx_filehandle_t fd, uint64_t pos);
|
||||||
int mdbx_filesize(mdbx_filehandle_t fd, uint64_t *length);
|
int mdbx_filesize(mdbx_filehandle_t fd, uint64_t *length);
|
||||||
int mdbx_openfile(const char *pathname, int flags, mode_t mode,
|
int mdbx_openfile(const char *pathname, int flags, mode_t mode,
|
||||||
mdbx_filehandle_t *fd);
|
mdbx_filehandle_t *fd);
|
||||||
int mdbx_closefile(mdbx_filehandle_t fd);
|
int mdbx_closefile(mdbx_filehandle_t fd);
|
||||||
|
int mdbx_removefile(const char *pathname);
|
||||||
|
|
||||||
typedef struct mdbx_mmap_param {
|
typedef struct mdbx_mmap_param {
|
||||||
union {
|
union {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user