mdbx: fix LRU spilling internal prio celling.

Related to https://github.com/erthink/libmdbx/issues/186

Change-Id: Ia4be12e063d1754a38889dc311eb1314b3676c21
This commit is contained in:
Leonid Yuriev 2021-04-28 16:38:08 +03:00
parent 329a2a50e6
commit 9fed78c92d

View File

@ -4837,7 +4837,7 @@ loop:;
const MDBX_page *mp = NULL; const MDBX_page *mp = NULL;
for (unsigned i = 0; i < mc->mc_snum; i++) { for (unsigned i = 0; i < mc->mc_snum; i++) {
mp = mc->mc_pg[i]; mp = mc->mc_pg[i];
if (IS_MODIFIABLE(txn, mp) && mp->mp_flags < P_SUBP) { if (IS_MODIFIABLE(txn, mp)) {
unsigned const n = mdbx_dpl_search(txn, mp->mp_pgno); unsigned const n = mdbx_dpl_search(txn, mp->mp_pgno);
if (txn->tw.dirtylist->items[n].pgno == mp->mp_pgno) if (txn->tw.dirtylist->items[n].pgno == mp->mp_pgno)
txn->tw.dirtylist->items[n].lru = txn->tw.dirtylru; txn->tw.dirtylist->items[n].lru = txn->tw.dirtylru;
@ -5229,11 +5229,11 @@ static int mdbx_txn_spill(MDBX_txn *const txn, MDBX_cursor *const m0,
} }
if (likely(spillable > 0)) { if (likely(spillable > 0)) {
unsigned prio2spill = 0, prio2adjacent = 127, amount = radix_counters[0]; unsigned prio2spill = 0, prio2adjacent = 128, amount = radix_counters[0];
for (unsigned i = 1; i < 256; i++) { for (unsigned i = 1; i < 256; i++) {
if (amount < wanna_spill) { if (amount < wanna_spill) {
prio2spill = i; prio2spill = i;
prio2adjacent = i + (255 - i) / 2; prio2adjacent = i + (257 - i) / 2;
amount += radix_counters[i]; amount += radix_counters[i];
} else if (amount + amount < spillable + wanna_spill } else if (amount + amount < spillable + wanna_spill
/* РАВНОЗНАЧНО: amount - wanna_spill < spillable - amount */) { /* РАВНОЗНАЧНО: amount - wanna_spill < spillable - amount */) {
@ -5243,6 +5243,11 @@ static int mdbx_txn_spill(MDBX_txn *const txn, MDBX_cursor *const m0,
break; break;
} }
mdbx_verbose("prio2spill %u, prio2adjacent %u, amount %u, spillable %u, "
"wanna_spill %u",
prio2spill, prio2adjacent, amount, spillable, wanna_spill);
mdbx_tassert(txn, prio2spill < prio2adjacent && prio2adjacent <= 256);
unsigned prev_prio = 256; unsigned prev_prio = 256;
unsigned r, w, prio; unsigned r, w, prio;
for (w = 0, r = 1; r <= dl->length && spilled < wanna_spill; for (w = 0, r = 1; r <= dl->length && spilled < wanna_spill;
@ -5293,6 +5298,8 @@ static int mdbx_txn_spill(MDBX_txn *const txn, MDBX_cursor *const m0,
dl->items[++w] = dl->items[r]; dl->items[++w] = dl->items[r];
} }
mdbx_tassert(txn, spillable == 0 || spilled > 0);
while (r <= dl->length) while (r <= dl->length)
dl->items[++w] = dl->items[r++]; dl->items[++w] = dl->items[r++];
mdbx_tassert(txn, r - 1 - w == spilled); mdbx_tassert(txn, r - 1 - w == spilled);