mirror of
https://github.com/isar/libmdbx.git
synced 2025-01-08 07:54:13 +08:00
mdbx: rollback weak checkpoint or MDB_CORRUPTED.
This commit is contained in:
parent
bfa3e864b6
commit
015a242c89
39
src/mdbx.c
39
src/mdbx.c
@ -1398,8 +1398,9 @@ static __inline MDB_meta *mdbx_env_meta_flipflop(const MDB_env *env,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static __inline int mdbx_meta_lt(const MDB_meta *a, const MDB_meta *b) {
|
static __inline int mdbx_meta_lt(const MDB_meta *a, const MDB_meta *b) {
|
||||||
return (META_IS_STEADY(a) == META_IS_STEADY(b)) ? a->mm_txnid < b->mm_txnid
|
if (META_IS_STEADY(a) == META_IS_STEADY(b))
|
||||||
: META_IS_STEADY(b);
|
return a->mm_txnid < b->mm_txnid;
|
||||||
|
return META_IS_STEADY(b);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Find oldest txnid still referenced. */
|
/** Find oldest txnid still referenced. */
|
||||||
@ -3815,6 +3816,39 @@ static int __cold mdbx_setup_dxb(MDB_env *env, MDB_meta *meta, int lck_rc) {
|
|||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
|
|
||||||
|
MDB_meta *const head = mdbx_meta_head(env);
|
||||||
|
if (head->mm_txnid != meta->mm_txnid) {
|
||||||
|
mdbx_trace("head->mm_txnid (%" PRIuPTR ") != (%" PRIuPTR
|
||||||
|
") meta->mm_txnid",
|
||||||
|
head->mm_txnid, meta->mm_txnid);
|
||||||
|
if (lck_rc == /* lck exclusive */ MDBX_RESULT_TRUE) {
|
||||||
|
assert(META_IS_STEADY(meta) && !META_IS_STEADY(head));
|
||||||
|
if (env->me_flags & MDB_RDONLY) {
|
||||||
|
mdbx_trace("exclusive, but read-only, unable recovery/rollback");
|
||||||
|
return MDB_CORRUPTED /* LY: could not recovery/rollback */;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* LY: rollback weak checkpoint */
|
||||||
|
MDB_meta rollback = *head;
|
||||||
|
rollback.mm_txnid = 0;
|
||||||
|
if (rollback.mm_txnid == meta->mm_txnid)
|
||||||
|
rollback = *meta;
|
||||||
|
err = mdbx_pwrite(env->me_fd, &rollback, sizeof(MDB_meta),
|
||||||
|
(uint8_t *)head - (uint8_t *)env->me_map);
|
||||||
|
if (err)
|
||||||
|
return err;
|
||||||
|
} else if (!env->me_lck) {
|
||||||
|
/* LY: without-lck (read-only) mode, so it is imposible that other
|
||||||
|
* process made weak checkpoint. */
|
||||||
|
mdbx_trace("without-lck, unable recovery/rollback");
|
||||||
|
return MDB_CORRUPTED;
|
||||||
|
} else {
|
||||||
|
/* LY: assume just have a collision with other running process,
|
||||||
|
* or someone make a weak checkpoint */
|
||||||
|
mdbx_trace("assume collision or online weak checkpoint");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
mdbx_env_setup_limits(env, env->me_psize);
|
mdbx_env_setup_limits(env, env->me_psize);
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
@ -3872,6 +3906,7 @@ static int __cold mdbx_setup_lck(MDB_env *env, char *lck_pathname, int mode) {
|
|||||||
err = mdbx_mmap(&addr, size, true, env->me_lfd);
|
err = mdbx_mmap(&addr, size, true, env->me_lfd);
|
||||||
if (unlikely(err != MDB_SUCCESS))
|
if (unlikely(err != MDB_SUCCESS))
|
||||||
return err;
|
return err;
|
||||||
|
assert(addr != nullptr);
|
||||||
env->me_lck = addr;
|
env->me_lck = addr;
|
||||||
|
|
||||||
#ifdef MADV_NOHUGEPAGE
|
#ifdef MADV_NOHUGEPAGE
|
||||||
|
Loading…
x
Reference in New Issue
Block a user