mdbx: simplify/speedup scan4seq().

This commit is contained in:
Леонид Юрьев (Leonid Yuriev) 2022-07-25 12:53:58 +03:00
parent 75d19b5806
commit 268b33cbf7

View File

@ -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;