mirror of
https://github.com/isar/libmdbx.git
synced 2025-01-21 21:58:20 +08:00
mdbx: incremental/combined linear+binary search for dirty-pages list.
Change-Id: I20f606a0a5d3729bfc92cb20c9db00cc1971d2ad
This commit is contained in:
parent
f3a51be7ff
commit
3549744f40
@ -951,7 +951,7 @@ static int lcklist_detach_locked(MDBX_env *env) {
|
|||||||
static __always_inline TYPE_LIST *NAME(TYPE_LIST *first, unsigned length, \
|
static __always_inline TYPE_LIST *NAME(TYPE_LIST *first, unsigned length, \
|
||||||
const TYPE_ARG item) { \
|
const TYPE_ARG item) { \
|
||||||
while (length > 3) { \
|
while (length > 3) { \
|
||||||
const size_t half = length >> 1; \
|
const unsigned half = length >> 1; \
|
||||||
TYPE_LIST *const middle = first + half; \
|
TYPE_LIST *const middle = first + half; \
|
||||||
if (CMP(*middle, item)) { \
|
if (CMP(*middle, item)) { \
|
||||||
first = middle + 1; \
|
first = middle + 1; \
|
||||||
@ -1308,20 +1308,55 @@ static __inline MDBX_DPL mdbx_dpl_sort(MDBX_DPL dl) {
|
|||||||
|
|
||||||
/* Returns the index of the first dirty-page whose pgno
|
/* Returns the index of the first dirty-page whose pgno
|
||||||
* member is greater than or equal to id. */
|
* member is greater than or equal to id. */
|
||||||
|
#define DP_SEARCH_CMP(dp, id) ((dp).pgno < (id))
|
||||||
|
SEARCH_IMPL(dp_bsearch, MDBX_DP, pgno_t, DP_SEARCH_CMP)
|
||||||
|
|
||||||
static unsigned __hot mdbx_dpl_search(MDBX_DPL dl, pgno_t id) {
|
static unsigned __hot mdbx_dpl_search(MDBX_DPL dl, pgno_t id) {
|
||||||
|
if (dl->sorted < dl->length) {
|
||||||
|
/* unsorted tail case */
|
||||||
#if MDBX_DEBUG
|
#if MDBX_DEBUG
|
||||||
for (const MDBX_DP *ptr = dl + dl->sorted; --ptr > dl;)
|
for (const MDBX_DP *ptr = dl + dl->sorted; --ptr > dl;) {
|
||||||
assert(ptr[0].pgno < ptr[1].pgno);
|
assert(ptr[0].pgno < ptr[1].pgno);
|
||||||
|
assert(ptr[0].pgno >= NUM_METAS);
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
dl = mdbx_dpl_sort(dl);
|
/* try linear search until the threshold */
|
||||||
/* binary search of id in array
|
if (dl->length - dl->sorted < SORT_THRESHOLD / 2) {
|
||||||
* if found, returns position of id
|
unsigned i = dl->length;
|
||||||
* if not found, returns first position greater than id */
|
while (i - dl->sorted > 7) {
|
||||||
unsigned base = 0;
|
if (dl[i].pgno == id)
|
||||||
unsigned cursor = 1;
|
return i;
|
||||||
int val = 0;
|
if (dl[i - 1].pgno == id)
|
||||||
unsigned n = dl->length;
|
return i - 1;
|
||||||
|
if (dl[i - 2].pgno == id)
|
||||||
|
return i - 2;
|
||||||
|
if (dl[i - 3].pgno == id)
|
||||||
|
return i - 3;
|
||||||
|
if (dl[i - 4].pgno == id)
|
||||||
|
return i - 4;
|
||||||
|
if (dl[i - 5].pgno == id)
|
||||||
|
return i - 5;
|
||||||
|
if (dl[i - 6].pgno == id)
|
||||||
|
return i - 6;
|
||||||
|
if (dl[i - 7].pgno == id)
|
||||||
|
return i - 7;
|
||||||
|
i -= 8;
|
||||||
|
}
|
||||||
|
while (i > dl->sorted) {
|
||||||
|
if (dl[i].pgno == id)
|
||||||
|
return i;
|
||||||
|
--i;
|
||||||
|
}
|
||||||
|
|
||||||
|
MDBX_DPL it = dp_bsearch(dl + 1, i, id);
|
||||||
|
return (unsigned)(it - dl);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* sort a whole */
|
||||||
|
dl->sorted = dl->length;
|
||||||
|
dp_sort(dl + 1, dl + dl->length + 1);
|
||||||
|
}
|
||||||
|
|
||||||
#if MDBX_DEBUG
|
#if MDBX_DEBUG
|
||||||
for (const MDBX_DP *ptr = dl + dl->length; --ptr > dl;) {
|
for (const MDBX_DP *ptr = dl + dl->length; --ptr > dl;) {
|
||||||
@ -1330,25 +1365,8 @@ static unsigned __hot mdbx_dpl_search(MDBX_DPL dl, pgno_t id) {
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
while (n > 0) {
|
MDBX_DPL it = dp_bsearch(dl + 1, dl->length, id);
|
||||||
unsigned pivot = n >> 1;
|
return (unsigned)(it - dl);
|
||||||
cursor = base + pivot + 1;
|
|
||||||
val = mdbx_cmp2int(id, dl[cursor].pgno);
|
|
||||||
|
|
||||||
if (val < 0) {
|
|
||||||
n = pivot;
|
|
||||||
} else if (val > 0) {
|
|
||||||
base = cursor;
|
|
||||||
n -= pivot + 1;
|
|
||||||
} else {
|
|
||||||
return cursor;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (val > 0)
|
|
||||||
++cursor;
|
|
||||||
|
|
||||||
return cursor;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static __inline MDBX_page *mdbx_dpl_find(MDBX_DPL dl, pgno_t id) {
|
static __inline MDBX_page *mdbx_dpl_find(MDBX_DPL dl, pgno_t id) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user