mirror of
https://github.com/isar/libmdbx.git
synced 2025-01-04 17:54:13 +08:00
mdbx: simplify/speedup scan4seq()
.
This commit is contained in:
parent
75d19b5806
commit
268b33cbf7
132
src/core.c
132
src/core.c
@ -6562,59 +6562,86 @@ __cold static int mdbx_wipe_steady(MDBX_env *env, const txnid_t last_steady) {
|
|||||||
return MDBX_SUCCESS;
|
return MDBX_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
__hot static pgno_t *scan4range(const MDBX_PNL pnl, const unsigned len,
|
__hot static pgno_t *scan4seq(pgno_t *range, const size_t len,
|
||||||
const int num) {
|
const unsigned seq) {
|
||||||
assert(num > 0 && len >= (unsigned)num && len == MDBX_PNL_SIZE(pnl));
|
assert(seq > 0 && len > seq);
|
||||||
#if MDBX_PNL_ASCENDING
|
#if MDBX_PNL_ASCENDING
|
||||||
const pgno_t *const detent = pnl + len - num;
|
assert(range[-1] == len);
|
||||||
pgno_t *scan = pnl + 1;
|
const pgno_t *const detent = range + len - seq;
|
||||||
while (likely(scan + 7 <= detent)) {
|
const ptrdiff_t offset = (ptrdiff_t)seq;
|
||||||
if (unlikely(scan[num] == *scan + num))
|
const pgno_t target = (pgno_t)offset;
|
||||||
return scan;
|
if (likely(len > seq + 3)) {
|
||||||
if (unlikely(scan[num + 1] == scan[1] + num))
|
do {
|
||||||
return scan + 1;
|
const pgno_t diff0 = range[offset + 0] - range[0];
|
||||||
if (unlikely(scan[num + 2] == scan[2] + num))
|
const pgno_t diff1 = range[offset + 1] - range[1];
|
||||||
return scan + 2;
|
const pgno_t diff2 = range[offset + 2] - range[2];
|
||||||
if (unlikely(scan[num + 3] == scan[3] + num))
|
const pgno_t diff3 = range[offset + 3] - range[3];
|
||||||
return scan + 3;
|
if (diff0 == target)
|
||||||
if (unlikely(scan[num + 4] == scan[4] + num))
|
return range + 0;
|
||||||
return scan + 4;
|
if (diff1 == target)
|
||||||
if (unlikely(scan[num + 5] == scan[5] + num))
|
return range + 1;
|
||||||
return scan + 5;
|
if (diff2 == target)
|
||||||
if (unlikely(scan[num + 6] == scan[6] + num))
|
return range + 2;
|
||||||
return scan + 6;
|
if (diff3 == target)
|
||||||
if (unlikely(scan[num + 7] == scan[7] + num))
|
return range + 3;
|
||||||
return scan + 7;
|
range += 4;
|
||||||
scan += 8;
|
} while (range + 3 < detent);
|
||||||
|
if (range == detent)
|
||||||
|
return nullptr;
|
||||||
}
|
}
|
||||||
for (; scan <= detent; ++scan)
|
do
|
||||||
if (scan[num] == *scan + num)
|
if (range[offset] - *range == target)
|
||||||
return scan;
|
return range;
|
||||||
|
while (++range < detent);
|
||||||
#else
|
#else
|
||||||
const pgno_t *const detent = pnl + num;
|
assert(range[-len] == len);
|
||||||
pgno_t *scan = pnl + len;
|
const pgno_t *const detent = range - len + seq;
|
||||||
while (likely(scan - 7 >= detent)) {
|
const ptrdiff_t offset = -(ptrdiff_t)seq;
|
||||||
if (unlikely(scan[-num] == *scan + num))
|
const pgno_t target = (pgno_t)offset;
|
||||||
return scan;
|
if (likely(len > seq + 3)) {
|
||||||
if (unlikely(scan[-num - 1] == scan[-1] + num))
|
do {
|
||||||
return scan - 1;
|
const pgno_t diff0 = range[-0] - range[offset - 0];
|
||||||
if (unlikely(scan[-num - 2] == scan[-2] + num))
|
const pgno_t diff1 = range[-1] - range[offset - 1];
|
||||||
return scan - 2;
|
const pgno_t diff2 = range[-2] - range[offset - 2];
|
||||||
if (unlikely(scan[-num - 3] == scan[-3] + num))
|
const pgno_t diff3 = range[-3] - range[offset - 3];
|
||||||
return scan - 3;
|
/* Смысл вычислений до ветвлений в том, чтобы позволить компилятору
|
||||||
if (unlikely(scan[-num - 4] == scan[-4] + num))
|
* загружать и вычислять все значения параллельно. */
|
||||||
return scan - 4;
|
if (diff0 == target)
|
||||||
if (unlikely(scan[-num - 5] == scan[-5] + num))
|
return range - 0;
|
||||||
return scan - 5;
|
if (diff1 == target)
|
||||||
if (unlikely(scan[-num - 6] == scan[-6] + num))
|
return range - 1;
|
||||||
return scan - 6;
|
if (diff2 == target)
|
||||||
if (unlikely(scan[-num - 7] == scan[-7] + num))
|
return range - 2;
|
||||||
return scan - 7;
|
if (diff3 == target)
|
||||||
scan -= 8;
|
return range - 3;
|
||||||
|
range -= 4;
|
||||||
|
} while (range > detent + 3);
|
||||||
|
if (range == detent)
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
do
|
||||||
|
if (*range - range[offset] == target)
|
||||||
|
return range;
|
||||||
|
while (--range > detent);
|
||||||
|
#endif /* MDBX_PNL sort-order */
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
MDBX_MAYBE_UNUSED static const pgno_t *scan4range_checker(const MDBX_PNL pnl,
|
||||||
|
const unsigned seq) {
|
||||||
|
size_t begin = MDBX_PNL_ASCENDING ? 1 : MDBX_PNL_SIZE(pnl);
|
||||||
|
#if MDBX_PNL_ASCENDING
|
||||||
|
while (seq <= MDBX_PNL_SIZE(pnl) - begin) {
|
||||||
|
if (pnl[begin + seq] - pnl[begin] == seq)
|
||||||
|
return pnl + begin;
|
||||||
|
++begin;
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
while (begin > seq) {
|
||||||
|
if (pnl[begin - seq] - pnl[begin] == seq)
|
||||||
|
return pnl + begin;
|
||||||
|
--begin;
|
||||||
}
|
}
|
||||||
for (; scan >= detent; --scan)
|
|
||||||
if (scan[-num] == *scan + num)
|
|
||||||
return scan;
|
|
||||||
#endif /* MDBX_PNL sort-order */
|
#endif /* MDBX_PNL sort-order */
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
@ -6692,7 +6719,12 @@ static pgr_t page_alloc_slowpath(MDBX_cursor *mc, const pgno_t num, int flags) {
|
|||||||
if (!(flags & (MDBX_ALLOC_COALESCE | MDBX_ALLOC_SLOT)) && re_len >= num) {
|
if (!(flags & (MDBX_ALLOC_COALESCE | MDBX_ALLOC_SLOT)) && re_len >= num) {
|
||||||
mdbx_assert(env, MDBX_PNL_LAST(re_list) < txn->mt_next_pgno &&
|
mdbx_assert(env, MDBX_PNL_LAST(re_list) < txn->mt_next_pgno &&
|
||||||
MDBX_PNL_FIRST(re_list) < txn->mt_next_pgno);
|
MDBX_PNL_FIRST(re_list) < txn->mt_next_pgno);
|
||||||
range = scan4range(re_list, re_len, num);
|
range = re_list + (MDBX_PNL_ASCENDING ? 1 : re_len);
|
||||||
|
pgno = *range;
|
||||||
|
if (num == 1)
|
||||||
|
goto done;
|
||||||
|
range = scan4seq(range, re_len, num - 1);
|
||||||
|
mdbx_tassert(txn, range == scan4range_checker(re_list, num - 1));
|
||||||
if (likely(range)) {
|
if (likely(range)) {
|
||||||
pgno = *range;
|
pgno = *range;
|
||||||
goto done;
|
goto done;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user