mdbx: подсчет грязных страниц в режиме MDBX_WRITEMAP для статистики.

This commit is contained in:
Леонид Юрьев (Leonid Yuriev) 2022-11-13 20:59:31 +03:00
parent 9b062cf0c7
commit 6c840cf58e
2 changed files with 81 additions and 72 deletions

View File

@ -2460,49 +2460,50 @@ __hot static size_t pnl_merge(MDBX_PNL dst, const MDBX_PNL src) {
} }
static void spill_remove(MDBX_txn *txn, size_t idx, pgno_t npages) { static void spill_remove(MDBX_txn *txn, size_t idx, pgno_t npages) {
tASSERT(txn, idx > 0 && idx <= MDBX_PNL_GETSIZE(txn->tw.spill_pages) && tASSERT(txn, idx > 0 && idx <= MDBX_PNL_GETSIZE(txn->tw.spilled.list) &&
txn->tw.spill_least_removed > 0); txn->tw.spilled.least_removed > 0);
txn->tw.spill_least_removed = txn->tw.spilled.least_removed = (idx < txn->tw.spilled.least_removed)
(idx < txn->tw.spill_least_removed) ? idx : txn->tw.spill_least_removed; ? idx
txn->tw.spill_pages[idx] |= 1; : txn->tw.spilled.least_removed;
MDBX_PNL_SETSIZE(txn->tw.spill_pages, txn->tw.spilled.list[idx] |= 1;
MDBX_PNL_GETSIZE(txn->tw.spill_pages) - MDBX_PNL_SETSIZE(txn->tw.spilled.list,
(idx == MDBX_PNL_GETSIZE(txn->tw.spill_pages))); MDBX_PNL_GETSIZE(txn->tw.spilled.list) -
(idx == MDBX_PNL_GETSIZE(txn->tw.spilled.list)));
while (unlikely(npages > 1)) { while (unlikely(npages > 1)) {
const pgno_t pgno = (txn->tw.spill_pages[idx] >> 1) + 1; const pgno_t pgno = (txn->tw.spilled.list[idx] >> 1) + 1;
if (MDBX_PNL_ASCENDING) { if (MDBX_PNL_ASCENDING) {
if (++idx > MDBX_PNL_GETSIZE(txn->tw.spill_pages) || if (++idx > MDBX_PNL_GETSIZE(txn->tw.spilled.list) ||
(txn->tw.spill_pages[idx] >> 1) != pgno) (txn->tw.spilled.list[idx] >> 1) != pgno)
return; return;
} else { } else {
if (--idx < 1 || (txn->tw.spill_pages[idx] >> 1) != pgno) if (--idx < 1 || (txn->tw.spilled.list[idx] >> 1) != pgno)
return; return;
txn->tw.spill_least_removed = (idx < txn->tw.spill_least_removed) txn->tw.spilled.least_removed = (idx < txn->tw.spilled.least_removed)
? idx ? idx
: txn->tw.spill_least_removed; : txn->tw.spilled.least_removed;
} }
txn->tw.spill_pages[idx] |= 1; txn->tw.spilled.list[idx] |= 1;
MDBX_PNL_SETSIZE(txn->tw.spill_pages, MDBX_PNL_SETSIZE(txn->tw.spilled.list,
MDBX_PNL_GETSIZE(txn->tw.spill_pages) - MDBX_PNL_GETSIZE(txn->tw.spilled.list) -
(idx == MDBX_PNL_GETSIZE(txn->tw.spill_pages))); (idx == MDBX_PNL_GETSIZE(txn->tw.spilled.list)));
--npages; --npages;
} }
} }
static MDBX_PNL spill_purge(MDBX_txn *txn) { static MDBX_PNL spill_purge(MDBX_txn *txn) {
tASSERT(txn, txn->tw.spill_least_removed > 0); tASSERT(txn, txn->tw.spilled.least_removed > 0);
const MDBX_PNL sl = txn->tw.spill_pages; const MDBX_PNL sl = txn->tw.spilled.list;
if (txn->tw.spill_least_removed != INT_MAX) { if (txn->tw.spilled.least_removed != INT_MAX) {
size_t len = MDBX_PNL_GETSIZE(sl), r, w; size_t len = MDBX_PNL_GETSIZE(sl), r, w;
for (w = r = txn->tw.spill_least_removed; r <= len; ++r) { for (w = r = txn->tw.spilled.least_removed; r <= len; ++r) {
sl[w] = sl[r]; sl[w] = sl[r];
w += 1 - (sl[r] & 1); w += 1 - (sl[r] & 1);
} }
for (size_t i = 1; i < w; ++i) for (size_t i = 1; i < w; ++i)
tASSERT(txn, (sl[i] & 1) == 0); tASSERT(txn, (sl[i] & 1) == 0);
MDBX_PNL_SETSIZE(sl, w - 1); MDBX_PNL_SETSIZE(sl, w - 1);
txn->tw.spill_least_removed = INT_MAX; txn->tw.spilled.least_removed = INT_MAX;
} else { } else {
for (size_t i = 1; i <= MDBX_PNL_GETSIZE(sl); ++i) for (size_t i = 1; i <= MDBX_PNL_GETSIZE(sl); ++i)
tASSERT(txn, (sl[i] & 1) == 0); tASSERT(txn, (sl[i] & 1) == 0);
@ -2558,7 +2559,8 @@ static __inline size_t pnl_search(const MDBX_PNL pnl, pgno_t pgno,
} }
static __inline size_t search_spilled(const MDBX_txn *txn, pgno_t pgno) { static __inline size_t search_spilled(const MDBX_txn *txn, pgno_t pgno) {
const MDBX_PNL pnl = txn->tw.spill_pages; tASSERT(txn, (txn->mt_flags & MDBX_WRITEMAP) == 0 || MDBX_AVOID_MSYNC);
const MDBX_PNL pnl = txn->tw.spilled.list;
if (likely(!pnl)) if (likely(!pnl))
return 0; return 0;
pgno <<= 1; pgno <<= 1;
@ -2568,7 +2570,7 @@ static __inline size_t search_spilled(const MDBX_txn *txn, pgno_t pgno) {
static __inline bool intersect_spilled(const MDBX_txn *txn, pgno_t pgno, static __inline bool intersect_spilled(const MDBX_txn *txn, pgno_t pgno,
pgno_t npages) { pgno_t npages) {
const MDBX_PNL pnl = txn->tw.spill_pages; const MDBX_PNL pnl = txn->tw.spilled.list;
if (likely(!pnl)) if (likely(!pnl))
return false; return false;
const size_t len = MDBX_PNL_GETSIZE(pnl); const size_t len = MDBX_PNL_GETSIZE(pnl);
@ -4023,7 +4025,7 @@ static bool txn_refund(MDBX_txn *txn) {
if (before == txn->mt_next_pgno) if (before == txn->mt_next_pgno)
return false; return false;
if (txn->tw.spill_pages) if (txn->tw.spilled.list)
/* Squash deleted pagenums if we refunded any */ /* Squash deleted pagenums if we refunded any */
spill_purge(txn); spill_purge(txn);
@ -4132,7 +4134,7 @@ static int page_retire_ex(MDBX_cursor *mc, const pgno_t pgno,
is_frozen = true; is_frozen = true;
if (ASSERT_ENABLED()) { if (ASSERT_ENABLED()) {
for (MDBX_txn *scan = txn; scan; scan = scan->mt_parent) { for (MDBX_txn *scan = txn; scan; scan = scan->mt_parent) {
tASSERT(txn, !search_spilled(scan, pgno)); tASSERT(txn, !txn->tw.spilled.list || !search_spilled(scan, pgno));
tASSERT(txn, !scan->tw.dirtylist || !debug_dpl_find(scan, pgno)); tASSERT(txn, !scan->tw.dirtylist || !debug_dpl_find(scan, pgno));
} }
} }
@ -4177,7 +4179,7 @@ static int page_retire_ex(MDBX_cursor *mc, const pgno_t pgno,
is_shadowed = IS_SHADOWED(txn, mp); is_shadowed = IS_SHADOWED(txn, mp);
if (is_dirty) { if (is_dirty) {
tASSERT(txn, !is_spilled); tASSERT(txn, !is_spilled);
tASSERT(txn, !search_spilled(txn, pgno)); tASSERT(txn, !txn->tw.spilled.list || !search_spilled(txn, pgno));
tASSERT(txn, debug_dpl_find(txn, pgno) == mp || txn->mt_parent || tASSERT(txn, debug_dpl_find(txn, pgno) == mp || txn->mt_parent ||
(txn->mt_flags & MDBX_WRITEMAP)); (txn->mt_flags & MDBX_WRITEMAP));
} else { } else {
@ -4559,7 +4561,7 @@ static int spill_page(MDBX_txn *txn, iov_ctx_t *ctx, MDBX_page *dp,
int err = iov_page(txn, ctx, dp, npages); int err = iov_page(txn, ctx, dp, npages);
if (likely(err == MDBX_SUCCESS) && if (likely(err == MDBX_SUCCESS) &&
(!MDBX_AVOID_MSYNC || !(txn->mt_flags & MDBX_WRITEMAP))) (!MDBX_AVOID_MSYNC || !(txn->mt_flags & MDBX_WRITEMAP)))
err = pnl_append_range(true, &txn->tw.spill_pages, pgno << 1, npages); err = pnl_append_range(true, &txn->tw.spilled.list, pgno << 1, npages);
return err; return err;
} }
@ -4758,7 +4760,7 @@ __cold static int txn_spill_slowpath(MDBX_txn *const txn, MDBX_cursor *const m0,
if (txn->mt_flags & MDBX_WRITEMAP) { if (txn->mt_flags & MDBX_WRITEMAP) {
NOTICE("%s-spilling %zu dirty-entries, %zu dirty-npages", "msync", NOTICE("%s-spilling %zu dirty-entries, %zu dirty-npages", "msync",
dirty_entries, dirty_npages); dirty_entries, dirty_npages);
tASSERT(txn, txn->tw.spill_pages == nullptr); tASSERT(txn, txn->tw.spilled.list == nullptr);
const MDBX_env *env = txn->mt_env; const MDBX_env *env = txn->mt_env;
rc = rc =
osal_msync(&txn->mt_env->me_dxb_mmap, 0, osal_msync(&txn->mt_env->me_dxb_mmap, 0,
@ -4782,10 +4784,10 @@ __cold static int txn_spill_slowpath(MDBX_txn *const txn, MDBX_cursor *const m0,
tASSERT(txn, txn->tw.dirtylist->pages_including_loose - txn->tw.loose_count >= tASSERT(txn, txn->tw.dirtylist->pages_including_loose - txn->tw.loose_count >=
need_spill_npages); need_spill_npages);
if (!MDBX_AVOID_MSYNC || !(txn->mt_flags & MDBX_WRITEMAP)) { if (!MDBX_AVOID_MSYNC || !(txn->mt_flags & MDBX_WRITEMAP)) {
if (!txn->tw.spill_pages) { if (!txn->tw.spilled.list) {
txn->tw.spill_least_removed = INT_MAX; txn->tw.spilled.least_removed = INT_MAX;
txn->tw.spill_pages = pnl_alloc(need_spill); txn->tw.spilled.list = pnl_alloc(need_spill);
if (unlikely(!txn->tw.spill_pages)) { if (unlikely(!txn->tw.spilled.list)) {
rc = MDBX_ENOMEM; rc = MDBX_ENOMEM;
bailout: bailout:
txn->mt_flags |= MDBX_TXN_ERROR; txn->mt_flags |= MDBX_TXN_ERROR;
@ -4794,7 +4796,7 @@ __cold static int txn_spill_slowpath(MDBX_txn *const txn, MDBX_cursor *const m0,
} else { } else {
/* purge deleted slots */ /* purge deleted slots */
spill_purge(txn); spill_purge(txn);
rc = pnl_reserve(&txn->tw.spill_pages, need_spill); rc = pnl_reserve(&txn->tw.spilled.list, need_spill);
(void)rc /* ignore since the resulting list may be shorter (void)rc /* ignore since the resulting list may be shorter
and pnl_append() will increase pnl on demand */ and pnl_append() will increase pnl on demand */
; ;
@ -4978,7 +4980,7 @@ __cold static int txn_spill_slowpath(MDBX_txn *const txn, MDBX_cursor *const m0,
goto bailout; goto bailout;
if (!MDBX_AVOID_MSYNC || !(txn->mt_flags & MDBX_WRITEMAP)) { if (!MDBX_AVOID_MSYNC || !(txn->mt_flags & MDBX_WRITEMAP)) {
pnl_sort(txn->tw.spill_pages, (size_t)txn->mt_next_pgno << 1); pnl_sort(txn->tw.spilled.list, (size_t)txn->mt_next_pgno << 1);
txn->mt_flags |= MDBX_TXN_SPILLS; txn->mt_flags |= MDBX_TXN_SPILLS;
} }
NOTICE("spilled %u dirty-entries, %u dirty-npages, now have %zu dirty-room", NOTICE("spilled %u dirty-entries, %u dirty-npages, now have %zu dirty-room",
@ -5484,6 +5486,8 @@ __hot static int __must_check_result page_dirty(MDBX_txn *txn, MDBX_page *mp,
tASSERT(txn, (txn->mt_flags & MDBX_TXN_RDONLY) == 0); tASSERT(txn, (txn->mt_flags & MDBX_TXN_RDONLY) == 0);
if (!txn->tw.dirtylist) { if (!txn->tw.dirtylist) {
tASSERT(txn, (txn->mt_flags & MDBX_WRITEMAP) != 0 && !MDBX_AVOID_MSYNC); tASSERT(txn, (txn->mt_flags & MDBX_WRITEMAP) != 0 && !MDBX_AVOID_MSYNC);
txn->tw.writemap_dirty_npages += npages;
tASSERT(txn, txn->tw.spilled.list == nullptr);
return MDBX_SUCCESS; return MDBX_SUCCESS;
} }
tASSERT(txn, (txn->mt_flags & MDBX_WRITEMAP) == 0 || MDBX_AVOID_MSYNC); tASSERT(txn, (txn->mt_flags & MDBX_WRITEMAP) == 0 || MDBX_AVOID_MSYNC);
@ -8332,8 +8336,8 @@ static int txn_renew(MDBX_txn *txn, const unsigned flags) {
txn->tw.loose_refund_wl = 0; txn->tw.loose_refund_wl = 0;
#endif /* MDBX_ENABLE_REFUND */ #endif /* MDBX_ENABLE_REFUND */
MDBX_PNL_SETSIZE(txn->tw.retired_pages, 0); MDBX_PNL_SETSIZE(txn->tw.retired_pages, 0);
txn->tw.spill_pages = NULL; txn->tw.spilled.list = NULL;
txn->tw.spill_least_removed = 0; txn->tw.spilled.least_removed = 0;
txn->tw.last_reclaimed = 0; txn->tw.last_reclaimed = 0;
if (txn->tw.lifo_reclaimed) if (txn->tw.lifo_reclaimed)
MDBX_PNL_SETSIZE(txn->tw.lifo_reclaimed, 0); MDBX_PNL_SETSIZE(txn->tw.lifo_reclaimed, 0);
@ -8627,7 +8631,7 @@ int mdbx_txn_begin_ex(MDBX_env *env, MDBX_txn *parent, MDBX_txn_flags_t flags,
txn->tw.dirtylru = parent->tw.dirtylru; txn->tw.dirtylru = parent->tw.dirtylru;
dpl_sort(parent); dpl_sort(parent);
if (parent->tw.spill_pages) if (parent->tw.spilled.list)
spill_purge(parent); spill_purge(parent);
tASSERT(txn, MDBX_PNL_ALLOCLEN(txn->tw.relist) >= tASSERT(txn, MDBX_PNL_ALLOCLEN(txn->tw.relist) >=
@ -8704,7 +8708,7 @@ int mdbx_txn_begin_ex(MDBX_env *env, MDBX_txn *parent, MDBX_txn_flags_t flags,
eASSERT(env, (txn->mt_flags & eASSERT(env, (txn->mt_flags &
~(MDBX_WRITEMAP | MDBX_SHRINK_ALLOWED | MDBX_NOMETASYNC | ~(MDBX_WRITEMAP | MDBX_SHRINK_ALLOWED | MDBX_NOMETASYNC |
MDBX_SAFE_NOSYNC | MDBX_TXN_SPILLS)) == 0); MDBX_SAFE_NOSYNC | MDBX_TXN_SPILLS)) == 0);
assert(!txn->tw.spill_pages && !txn->tw.spill_least_removed); assert(!txn->tw.spilled.list && !txn->tw.spilled.least_removed);
} }
txn->mt_signature = MDBX_MT_SIGNATURE; txn->mt_signature = MDBX_MT_SIGNATURE;
txn->mt_userctx = context; txn->mt_userctx = context;
@ -8809,10 +8813,9 @@ int mdbx_txn_info(const MDBX_txn *txn, MDBX_txn_info *info, bool scan_rlt) {
env, txn->mt_child ? (size_t)txn->tw.retired_pages env, txn->mt_child ? (size_t)txn->tw.retired_pages
: MDBX_PNL_GETSIZE(txn->tw.retired_pages)); : MDBX_PNL_GETSIZE(txn->tw.retired_pages));
info->txn_space_leftover = pgno2bytes(env, txn->tw.dirtyroom); info->txn_space_leftover = pgno2bytes(env, txn->tw.dirtyroom);
info->txn_space_dirty = info->txn_space_dirty = pgno2bytes(
txn->tw.dirtylist env, txn->tw.dirtylist ? txn->tw.dirtylist->pages_including_loose
? pgno2bytes(env, txn->tw.dirtylist->pages_including_loose) : txn->tw.writemap_dirty_npages);
: 0;
info->txn_reader_lag = INT64_MAX; info->txn_reader_lag = INT64_MAX;
MDBX_lockinfo *const lck = env->me_lck_mmap.lck; MDBX_lockinfo *const lck = env->me_lck_mmap.lck;
if (scan_rlt && lck) { if (scan_rlt && lck) {
@ -9128,8 +9131,8 @@ static int txn_end(MDBX_txn *txn, const unsigned mode) {
txn->mt_flags = MDBX_TXN_FINISHED; txn->mt_flags = MDBX_TXN_FINISHED;
txn->mt_owner = 0; txn->mt_owner = 0;
env->me_txn = txn->mt_parent; env->me_txn = txn->mt_parent;
pnl_free(txn->tw.spill_pages); pnl_free(txn->tw.spilled.list);
txn->tw.spill_pages = nullptr; txn->tw.spilled.list = nullptr;
if (txn == env->me_txn0) { if (txn == env->me_txn0) {
eASSERT(env, txn->mt_parent == NULL); eASSERT(env, txn->mt_parent == NULL);
/* Export or close DBI handles created in this txn */ /* Export or close DBI handles created in this txn */
@ -10476,7 +10479,8 @@ static __inline void txn_merge(MDBX_txn *const parent, MDBX_txn *const txn,
MDBX_PNL_SETSIZE(parent->tw.retired_pages, w); MDBX_PNL_SETSIZE(parent->tw.retired_pages, w);
/* Filter-out parent spill list */ /* Filter-out parent spill list */
if (parent->tw.spill_pages && MDBX_PNL_GETSIZE(parent->tw.spill_pages) > 0) { if (parent->tw.spilled.list &&
MDBX_PNL_GETSIZE(parent->tw.spilled.list) > 0) {
const MDBX_PNL sl = spill_purge(parent); const MDBX_PNL sl = spill_purge(parent);
size_t len = MDBX_PNL_GETSIZE(sl); size_t len = MDBX_PNL_GETSIZE(sl);
if (len) { if (len) {
@ -10564,10 +10568,10 @@ static __inline void txn_merge(MDBX_txn *const parent, MDBX_txn *const txn,
} }
/* Remove anything in our spill list from parent's dirty list */ /* Remove anything in our spill list from parent's dirty list */
if (txn->tw.spill_pages) { if (txn->tw.spilled.list) {
tASSERT(txn, pnl_check_allocated(txn->tw.spill_pages, tASSERT(txn, pnl_check_allocated(txn->tw.spilled.list,
(size_t)parent->mt_next_pgno << 1)); (size_t)parent->mt_next_pgno << 1));
dpl_sift(parent, txn->tw.spill_pages, true); dpl_sift(parent, txn->tw.spilled.list, true);
tASSERT(parent, tASSERT(parent,
parent->tw.dirtyroom + parent->tw.dirtylist->length == parent->tw.dirtyroom + parent->tw.dirtylist->length ==
(parent->mt_parent ? parent->mt_parent->tw.dirtyroom (parent->mt_parent ? parent->mt_parent->tw.dirtyroom
@ -10719,23 +10723,23 @@ static __inline void txn_merge(MDBX_txn *const parent, MDBX_txn *const txn,
tASSERT(parent, dirtylist_check(parent)); tASSERT(parent, dirtylist_check(parent));
dpl_free(txn); dpl_free(txn);
if (txn->tw.spill_pages) { if (txn->tw.spilled.list) {
if (parent->tw.spill_pages) { if (parent->tw.spilled.list) {
/* Must not fail since space was preserved above. */ /* Must not fail since space was preserved above. */
pnl_merge(parent->tw.spill_pages, txn->tw.spill_pages); pnl_merge(parent->tw.spilled.list, txn->tw.spilled.list);
pnl_free(txn->tw.spill_pages); pnl_free(txn->tw.spilled.list);
} else { } else {
parent->tw.spill_pages = txn->tw.spill_pages; parent->tw.spilled.list = txn->tw.spilled.list;
parent->tw.spill_least_removed = txn->tw.spill_least_removed; parent->tw.spilled.least_removed = txn->tw.spilled.least_removed;
} }
tASSERT(parent, dirtylist_check(parent)); tASSERT(parent, dirtylist_check(parent));
} }
parent->mt_flags &= ~MDBX_TXN_HAS_CHILD; parent->mt_flags &= ~MDBX_TXN_HAS_CHILD;
if (parent->tw.spill_pages) { if (parent->tw.spilled.list) {
assert(pnl_check_allocated(parent->tw.spill_pages, assert(pnl_check_allocated(parent->tw.spilled.list,
(size_t)parent->mt_next_pgno << 1)); (size_t)parent->mt_next_pgno << 1));
if (MDBX_PNL_GETSIZE(parent->tw.spill_pages)) if (MDBX_PNL_GETSIZE(parent->tw.spilled.list))
parent->mt_flags |= MDBX_TXN_SPILLS; parent->mt_flags |= MDBX_TXN_SPILLS;
} }
} }
@ -10806,8 +10810,8 @@ int mdbx_txn_commit_ex(MDBX_txn *txn, MDBX_commit_latency *latency) {
sizeof(parent->mt_geo)) == 0); sizeof(parent->mt_geo)) == 0);
tASSERT(txn, memcmp(&parent->mt_canary, &txn->mt_canary, tASSERT(txn, memcmp(&parent->mt_canary, &txn->mt_canary,
sizeof(parent->mt_canary)) == 0); sizeof(parent->mt_canary)) == 0);
tASSERT(txn, !txn->tw.spill_pages || tASSERT(txn, !txn->tw.spilled.list ||
MDBX_PNL_GETSIZE(txn->tw.spill_pages) == 0); MDBX_PNL_GETSIZE(txn->tw.spilled.list) == 0);
tASSERT(txn, txn->tw.loose_count == 0); tASSERT(txn, txn->tw.loose_count == 0);
/* fast completion of pure nested transaction */ /* fast completion of pure nested transaction */
@ -10827,10 +10831,10 @@ int mdbx_txn_commit_ex(MDBX_txn *txn, MDBX_commit_latency *latency) {
goto fail; goto fail;
} }
if (txn->tw.spill_pages) { if (txn->tw.spilled.list) {
if (parent->tw.spill_pages) { if (parent->tw.spilled.list) {
rc = pnl_need(&parent->tw.spill_pages, rc = pnl_need(&parent->tw.spilled.list,
MDBX_PNL_GETSIZE(txn->tw.spill_pages)); MDBX_PNL_GETSIZE(txn->tw.spilled.list));
if (unlikely(rc != MDBX_SUCCESS)) if (unlikely(rc != MDBX_SUCCESS))
goto fail; goto fail;
} }
@ -14277,7 +14281,7 @@ __cold static int env_close(MDBX_env *env) {
dpl_free(env->me_txn0); dpl_free(env->me_txn0);
txl_free(env->me_txn0->tw.lifo_reclaimed); txl_free(env->me_txn0->tw.lifo_reclaimed);
pnl_free(env->me_txn0->tw.retired_pages); pnl_free(env->me_txn0->tw.retired_pages);
pnl_free(env->me_txn0->tw.spill_pages); pnl_free(env->me_txn0->tw.spilled.list);
pnl_free(env->me_txn0->tw.relist); pnl_free(env->me_txn0->tw.relist);
osal_free(env->me_txn0); osal_free(env->me_txn0);
env->me_txn0 = nullptr; env->me_txn0 = nullptr;

View File

@ -1088,11 +1088,16 @@ struct MDBX_txn {
MDBX_page *loose_pages; MDBX_page *loose_pages;
/* Number of loose pages (tw.loose_pages) */ /* Number of loose pages (tw.loose_pages) */
size_t loose_count; size_t loose_count;
size_t spill_least_removed; union {
/* The sorted list of dirty pages we temporarily wrote to disk struct {
* because the dirty list was full. page numbers in here are size_t least_removed;
* shifted left by 1, deleted slots have the LSB set. */ /* The sorted list of dirty pages we temporarily wrote to disk
MDBX_PNL spill_pages; * because the dirty list was full. page numbers in here are
* shifted left by 1, deleted slots have the LSB set. */
MDBX_PNL list;
} spilled;
size_t writemap_dirty_npages;
};
} tw; } tw;
}; };
}; };