mirror of
https://github.com/isar/libmdbx.git
synced 2025-01-23 02:28:20 +08:00
mdbx: minor refine/speedup dpl_sort_slowpath()
.
This commit is contained in:
parent
a11c045f1e
commit
eac3d0499f
32
src/core.c
32
src/core.c
@ -2919,7 +2919,7 @@ RADIXSORT_IMPL(dpl, MDBX_dp, MDBX_DPL_EXTRACT_KEY,
|
|||||||
#define DP_SORT_CMP(first, last) ((first).pgno < (last).pgno)
|
#define DP_SORT_CMP(first, last) ((first).pgno < (last).pgno)
|
||||||
SORT_IMPL(dp_sort, false, MDBX_dp, DP_SORT_CMP)
|
SORT_IMPL(dp_sort, false, MDBX_dp, DP_SORT_CMP)
|
||||||
|
|
||||||
__hot __noinline static MDBX_dpl *mdbx_dpl_sort_slowpath(const MDBX_txn *txn) {
|
__hot __noinline static MDBX_dpl *dpl_sort_slowpath(const MDBX_txn *txn) {
|
||||||
MDBX_dpl *dl = txn->tw.dirtylist;
|
MDBX_dpl *dl = txn->tw.dirtylist;
|
||||||
assert(dl->items[0].pgno == 0 && dl->items[dl->length + 1].pgno == P_INVALID);
|
assert(dl->items[0].pgno == 0 && dl->items[dl->length + 1].pgno == P_INVALID);
|
||||||
const unsigned unsorted = dl->length - dl->sorted;
|
const unsigned unsorted = dl->length - dl->sorted;
|
||||||
@ -2940,14 +2940,18 @@ __hot __noinline static MDBX_dpl *mdbx_dpl_sort_slowpath(const MDBX_txn *txn) {
|
|||||||
memcpy(tmp, sorted_end, unsorted * sizeof(MDBX_dp));
|
memcpy(tmp, sorted_end, unsorted * sizeof(MDBX_dp));
|
||||||
dp_sort(tmp, tmp + unsorted);
|
dp_sort(tmp, tmp + unsorted);
|
||||||
/* merge two parts from end to begin */
|
/* merge two parts from end to begin */
|
||||||
MDBX_dp *w = dl->items + dl->length;
|
MDBX_dp *__restrict w = dl->items + dl->length;
|
||||||
MDBX_dp *l = dl->items + dl->sorted;
|
MDBX_dp *__restrict l = dl->items + dl->sorted;
|
||||||
MDBX_dp *r = end - 1;
|
MDBX_dp *__restrict r = end - 1;
|
||||||
do {
|
do {
|
||||||
const bool cmp = expect_with_probability(l->pgno > r->pgno, 0, .5);
|
const bool cmp = expect_with_probability(l->pgno > r->pgno, 0, .5);
|
||||||
|
#if defined(__LCC__) || __CLANG_PREREQ(13, 0) || !MDBX_HAVE_CMOV
|
||||||
|
*w = cmp ? *l-- : *r--;
|
||||||
|
#else
|
||||||
*w = cmp ? *l : *r;
|
*w = cmp ? *l : *r;
|
||||||
l -= cmp;
|
l -= cmp;
|
||||||
r += cmp - 1;
|
r += cmp - 1;
|
||||||
|
#endif
|
||||||
} while (likely(--w > l));
|
} while (likely(--w > l));
|
||||||
assert(r == tmp - 1);
|
assert(r == tmp - 1);
|
||||||
assert(dl->items[0].pgno == 0 &&
|
assert(dl->items[0].pgno == 0 &&
|
||||||
@ -2968,12 +2972,12 @@ __hot __noinline static MDBX_dpl *mdbx_dpl_sort_slowpath(const MDBX_txn *txn) {
|
|||||||
return dl;
|
return dl;
|
||||||
}
|
}
|
||||||
|
|
||||||
static __always_inline MDBX_dpl *mdbx_dpl_sort(const MDBX_txn *txn) {
|
static __always_inline MDBX_dpl *dpl_sort(const MDBX_txn *txn) {
|
||||||
MDBX_dpl *dl = txn->tw.dirtylist;
|
MDBX_dpl *dl = txn->tw.dirtylist;
|
||||||
assert(dl->length <= MDBX_PGL_LIMIT);
|
assert(dl->length <= MDBX_PGL_LIMIT);
|
||||||
assert(dl->sorted <= dl->length);
|
assert(dl->sorted <= dl->length);
|
||||||
assert(dl->items[0].pgno == 0 && dl->items[dl->length + 1].pgno == P_INVALID);
|
assert(dl->items[0].pgno == 0 && dl->items[dl->length + 1].pgno == P_INVALID);
|
||||||
return likely(dl->sorted == dl->length) ? dl : mdbx_dpl_sort_slowpath(txn);
|
return likely(dl->sorted == dl->length) ? dl : dpl_sort_slowpath(txn);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Returns the index of the first dirty-page whose pgno
|
/* Returns the index of the first dirty-page whose pgno
|
||||||
@ -2994,7 +2998,7 @@ static unsigned __hot mdbx_dpl_search(const MDBX_txn *txn, pgno_t pgno) {
|
|||||||
switch (dl->length - dl->sorted) {
|
switch (dl->length - dl->sorted) {
|
||||||
default:
|
default:
|
||||||
/* sort a whole */
|
/* sort a whole */
|
||||||
mdbx_dpl_sort_slowpath(txn);
|
dpl_sort_slowpath(txn);
|
||||||
break;
|
break;
|
||||||
case 0:
|
case 0:
|
||||||
/* whole sorted cases */
|
/* whole sorted cases */
|
||||||
@ -4023,7 +4027,7 @@ static void mdbx_refund_loose(MDBX_txn *txn) {
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
/* Dirtylist is mostly sorted, just refund loose pages at the end. */
|
/* Dirtylist is mostly sorted, just refund loose pages at the end. */
|
||||||
mdbx_dpl_sort(txn);
|
dpl_sort(txn);
|
||||||
mdbx_tassert(txn, dl->length < 2 ||
|
mdbx_tassert(txn, dl->length < 2 ||
|
||||||
dl->items[1].pgno < dl->items[dl->length].pgno);
|
dl->items[1].pgno < dl->items[dl->length].pgno);
|
||||||
mdbx_tassert(txn, dl->sorted == dl->length);
|
mdbx_tassert(txn, dl->sorted == dl->length);
|
||||||
@ -4838,7 +4842,7 @@ static int mdbx_txn_spill(MDBX_txn *const txn, MDBX_cursor *const m0,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Сортируем чтобы запись на диск была полее последовательна */
|
/* Сортируем чтобы запись на диск была полее последовательна */
|
||||||
MDBX_dpl *const dl = mdbx_dpl_sort(txn);
|
MDBX_dpl *const dl = dpl_sort(txn);
|
||||||
|
|
||||||
/* Preserve pages which may soon be dirtied again */
|
/* Preserve pages which may soon be dirtied again */
|
||||||
const unsigned unspillable = mdbx_txn_keep(txn, m0);
|
const unsigned unspillable = mdbx_txn_keep(txn, m0);
|
||||||
@ -8023,7 +8027,7 @@ int mdbx_txn_begin_ex(MDBX_env *env, MDBX_txn *parent, MDBX_txn_flags_t flags,
|
|||||||
txn->tw.dirtyroom = parent->tw.dirtyroom;
|
txn->tw.dirtyroom = parent->tw.dirtyroom;
|
||||||
txn->tw.dirtylru = parent->tw.dirtylru;
|
txn->tw.dirtylru = parent->tw.dirtylru;
|
||||||
|
|
||||||
mdbx_dpl_sort(parent);
|
dpl_sort(parent);
|
||||||
if (parent->tw.spill_pages)
|
if (parent->tw.spill_pages)
|
||||||
mdbx_spill_purge(parent);
|
mdbx_spill_purge(parent);
|
||||||
|
|
||||||
@ -8383,7 +8387,7 @@ static void mdbx_dpl_sift(MDBX_txn *const txn, MDBX_PNL pl,
|
|||||||
if (MDBX_PNL_SIZE(pl) && txn->tw.dirtylist->length) {
|
if (MDBX_PNL_SIZE(pl) && txn->tw.dirtylist->length) {
|
||||||
mdbx_tassert(txn,
|
mdbx_tassert(txn,
|
||||||
pnl_check_allocated(pl, (size_t)txn->mt_next_pgno << spilled));
|
pnl_check_allocated(pl, (size_t)txn->mt_next_pgno << spilled));
|
||||||
MDBX_dpl *dl = mdbx_dpl_sort(txn);
|
MDBX_dpl *dl = dpl_sort(txn);
|
||||||
|
|
||||||
/* Scanning in ascend order */
|
/* Scanning in ascend order */
|
||||||
const int step = MDBX_PNL_ASCENDING ? 1 : -1;
|
const int step = MDBX_PNL_ASCENDING ? 1 : -1;
|
||||||
@ -9740,7 +9744,7 @@ bailout:
|
|||||||
|
|
||||||
static int mdbx_txn_write(MDBX_txn *txn, struct mdbx_iov_ctx *ctx) {
|
static int mdbx_txn_write(MDBX_txn *txn, struct mdbx_iov_ctx *ctx) {
|
||||||
MDBX_dpl *const dl =
|
MDBX_dpl *const dl =
|
||||||
(txn->mt_flags & MDBX_WRITEMAP) ? txn->tw.dirtylist : mdbx_dpl_sort(txn);
|
(txn->mt_flags & MDBX_WRITEMAP) ? txn->tw.dirtylist : dpl_sort(txn);
|
||||||
int rc = MDBX_SUCCESS;
|
int rc = MDBX_SUCCESS;
|
||||||
unsigned r, w;
|
unsigned r, w;
|
||||||
for (w = 0, r = 1; r <= dl->length; ++r) {
|
for (w = 0, r = 1; r <= dl->length; ++r) {
|
||||||
@ -9792,10 +9796,10 @@ int mdbx_txn_commit(MDBX_txn *txn) { return __inline_mdbx_txn_commit(txn); }
|
|||||||
/* Merge child txn into parent */
|
/* Merge child txn into parent */
|
||||||
static __inline void mdbx_txn_merge(MDBX_txn *const parent, MDBX_txn *const txn,
|
static __inline void mdbx_txn_merge(MDBX_txn *const parent, MDBX_txn *const txn,
|
||||||
const unsigned parent_retired_len) {
|
const unsigned parent_retired_len) {
|
||||||
MDBX_dpl *const src = mdbx_dpl_sort(txn);
|
MDBX_dpl *const src = dpl_sort(txn);
|
||||||
|
|
||||||
/* Remove refunded pages from parent's dirty list */
|
/* Remove refunded pages from parent's dirty list */
|
||||||
MDBX_dpl *const dst = mdbx_dpl_sort(parent);
|
MDBX_dpl *const dst = dpl_sort(parent);
|
||||||
if (MDBX_ENABLE_REFUND) {
|
if (MDBX_ENABLE_REFUND) {
|
||||||
unsigned n = dst->length;
|
unsigned n = dst->length;
|
||||||
while (n && dst->items[n].pgno >= parent->mt_next_pgno) {
|
while (n && dst->items[n].pgno >= parent->mt_next_pgno) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user