mdbx: доработка update_gc() для улучшения сходимости с исправлением вероятности MDBX_PROBLEM.

This commit is contained in:
Леонид Юрьев (Leonid Yuriev) 2024-04-16 08:31:37 +03:00
parent 45377f20c5
commit 3518b2f36c

View File

@ -10599,7 +10599,7 @@ __cold static int audit_ex(MDBX_txn *txn, size_t retired_stored,
typedef struct gc_update_context { typedef struct gc_update_context {
size_t loop, reserve_adj; size_t loop, reserve_adj;
size_t retired_stored; size_t retired_stored;
size_t reserved, cleaned_slot, reused_slot, fill_idx; size_t amount, reserved, cleaned_slot, reused_slot, fill_idx;
txnid_t cleaned_id, rid; txnid_t cleaned_id, rid;
bool lifo, dense; bool lifo, dense;
#if MDBX_ENABLE_BIGFOOT #if MDBX_ENABLE_BIGFOOT
@ -10781,13 +10781,17 @@ static int update_gc(MDBX_txn *txn, gcu_context_t *ctx) {
ctx->cursor.mc_next = txn->mt_cursors[FREE_DBI]; ctx->cursor.mc_next = txn->mt_cursors[FREE_DBI];
txn->mt_cursors[FREE_DBI] = &ctx->cursor; txn->mt_cursors[FREE_DBI] = &ctx->cursor;
pgno_t prev_next_pgno = 0;
/* txn->tw.relist[] can grow and shrink during this call. /* txn->tw.relist[] can grow and shrink during this call.
* txn->tw.last_reclaimed and txn->tw.retired_pages[] can only grow. * txn->tw.last_reclaimed and txn->tw.retired_pages[] can only grow.
* But page numbers cannot disappear from txn->tw.retired_pages[]. */ * But page numbers cannot disappear from txn->tw.retired_pages[]. */
retry_clean_adj: retry_clean_adj:
ctx->reserve_adj = 0; ctx->reserve_adj = 0;
retry: retry:
if (ctx->loop++) ctx->loop += prev_next_pgno == txn->mt_next_pgno;
prev_next_pgno = txn->mt_next_pgno;
if (ctx->loop)
TRACE("%s", " >> restart"); TRACE("%s", " >> restart");
int rc = MDBX_SUCCESS; int rc = MDBX_SUCCESS;
tASSERT(txn, pnl_check_allocated(txn->tw.relist, tASSERT(txn, pnl_check_allocated(txn->tw.relist,
@ -10808,7 +10812,7 @@ retry:
ctx->reserved = 0; ctx->reserved = 0;
ctx->cleaned_slot = 0; ctx->cleaned_slot = 0;
ctx->reused_slot = 0; ctx->reused_slot = 0;
ctx->fill_idx = ~0u; ctx->amount = ctx->fill_idx = ~0u;
ctx->cleaned_id = 0; ctx->cleaned_id = 0;
ctx->rid = txn->tw.last_reclaimed; ctx->rid = txn->tw.last_reclaimed;
while (true) { while (true) {
@ -10877,9 +10881,11 @@ retry:
rc = MDBX_CORRUPTED; rc = MDBX_CORRUPTED;
goto bailout; goto bailout;
} }
if (ctx->rid != ctx->cleaned_id) {
ctx->rid = ctx->cleaned_id; ctx->rid = ctx->cleaned_id;
ctx->reserved = 0; ctx->reserved = 0;
ctx->reused_slot = 0; ctx->reused_slot = 0;
}
ctx->cleaned_id = unaligned_peek_u64(4, key.iov_base); ctx->cleaned_id = unaligned_peek_u64(4, key.iov_base);
if (ctx->cleaned_id > txn->tw.last_reclaimed) if (ctx->cleaned_id > txn->tw.last_reclaimed)
break; break;
@ -11012,7 +11018,16 @@ retry:
#endif /* MDBX_ENABLE_REFUND */ #endif /* MDBX_ENABLE_REFUND */
} }
const size_t amount = MDBX_PNL_GETSIZE(txn->tw.relist); if (unlikely(ctx->reserved > MDBX_PNL_GETSIZE(txn->tw.relist)) &&
(ctx->loop < 5 || ctx->reserved - MDBX_PNL_GETSIZE(txn->tw.relist) >
env->me_maxgc_ov1page / 2)) {
TRACE("%s: reclaimed-list changed %zu -> %zu, retry", dbg_prefix_mode,
ctx->amount, MDBX_PNL_GETSIZE(txn->tw.relist));
ctx->reserve_adj += ctx->reserved - MDBX_PNL_GETSIZE(txn->tw.relist);
goto retry;
}
ctx->amount = MDBX_PNL_GETSIZE(txn->tw.relist);
/* handle retired-list - store ones into single gc-record */ /* handle retired-list - store ones into single gc-record */
if (ctx->retired_stored < MDBX_PNL_GETSIZE(txn->tw.retired_pages)) { if (ctx->retired_stored < MDBX_PNL_GETSIZE(txn->tw.retired_pages)) {
if (unlikely(!ctx->retired_stored)) { if (unlikely(!ctx->retired_stored)) {
@ -11148,14 +11163,6 @@ retry:
DEBUG_EXTRA_PRINT(" %" PRIaPGNO, txn->tw.retired_pages[i]); DEBUG_EXTRA_PRINT(" %" PRIaPGNO, txn->tw.retired_pages[i]);
DEBUG_EXTRA_PRINT("%s\n", "."); DEBUG_EXTRA_PRINT("%s\n", ".");
} }
if (unlikely(amount != MDBX_PNL_GETSIZE(txn->tw.relist) &&
ctx->reserved)) {
TRACE("%s: reclaimed-list changed %zu -> %zu, retry", dbg_prefix_mode,
amount, MDBX_PNL_GETSIZE(txn->tw.relist));
goto retry_clean_adj /* rare case, but avoids GC fragmentation
and one cycle. */
;
}
continue; continue;
} }
@ -11170,11 +11177,11 @@ retry:
if (unlikely(rc != MDBX_SUCCESS)) if (unlikely(rc != MDBX_SUCCESS))
goto bailout; goto bailout;
} }
const size_t left = amount - ctx->reserved - ctx->reserve_adj; const size_t left = ctx->amount - ctx->reserved - ctx->reserve_adj;
TRACE("%s: amount %zu, settled %zd, reserve_adj %zu, left %zd, " TRACE("%s: amount %zu, reserved %zd, reserve_adj %zu, left %zd, "
"lifo-reclaimed-slots %zu, " "lifo-reclaimed-slots %zu, "
"reused-gc-slots %zu", "reused-gc-slots %zu",
dbg_prefix_mode, amount, ctx->reserved, ctx->reserve_adj, left, dbg_prefix_mode, ctx->amount, ctx->reserved, ctx->reserve_adj, left,
txn->tw.lifo_reclaimed ? MDBX_PNL_GETSIZE(txn->tw.lifo_reclaimed) : 0, txn->tw.lifo_reclaimed ? MDBX_PNL_GETSIZE(txn->tw.lifo_reclaimed) : 0,
ctx->reused_slot); ctx->reused_slot);
if (0 >= (intptr_t)left) if (0 >= (intptr_t)left)
@ -11249,7 +11256,7 @@ retry:
if (unlikely(MDBX_PNL_GETSIZE(txn->tw.lifo_reclaimed) <= if (unlikely(MDBX_PNL_GETSIZE(txn->tw.lifo_reclaimed) <=
ctx->reused_slot)) { ctx->reused_slot)) {
NOTICE("** restart: reserve depleted (reused_gc_slot %zu >= " NOTICE("** restart: reserve depleted (reused_gc_slot %zu >= "
"lifo_reclaimed %zu" PRIaTXN, "lifo_reclaimed %zu)",
ctx->reused_slot, ctx->reused_slot,
MDBX_PNL_GETSIZE(txn->tw.lifo_reclaimed)); MDBX_PNL_GETSIZE(txn->tw.lifo_reclaimed));
goto retry; goto retry;
@ -11336,7 +11343,7 @@ retry:
if (unlikely(ctx->rid == 0)) { if (unlikely(ctx->rid == 0)) {
ERROR("%s", "** no GC tail-space to store (going dense-mode)"); ERROR("%s", "** no GC tail-space to store (going dense-mode)");
ctx->dense = true; ctx->dense = true;
goto retry_clean_adj; goto retry;
} }
} else if (rc != MDBX_NOTFOUND) } else if (rc != MDBX_NOTFOUND)
goto bailout; goto bailout;
@ -11350,17 +11357,24 @@ retry:
++ctx->reused_slot; ++ctx->reused_slot;
size_t chunk = left; size_t chunk = left;
if (unlikely(chunk > env->me_maxgc_ov1page)) { if (unlikely(left > env->me_maxgc_ov1page)) {
const size_t avail_gc_slots = const size_t avail_gc_slots =
txn->tw.lifo_reclaimed txn->tw.lifo_reclaimed
? MDBX_PNL_GETSIZE(txn->tw.lifo_reclaimed) - ctx->reused_slot + 1 ? MDBX_PNL_GETSIZE(txn->tw.lifo_reclaimed) - ctx->reused_slot + 1
: (ctx->rid < INT16_MAX) ? (size_t)ctx->rid : (ctx->rid < INT16_MAX) ? (size_t)ctx->rid
: INT16_MAX; : INT16_MAX;
if (avail_gc_slots > 1) { if (likely(avail_gc_slots > 1)) {
#if MDBX_ENABLE_BIGFOOT #if MDBX_ENABLE_BIGFOOT
chunk = (chunk < env->me_maxgc_ov1page * (size_t)2) chunk = env->me_maxgc_ov1page;
? chunk / 2 if (avail_gc_slots < INT16_MAX &&
: env->me_maxgc_ov1page; unlikely(left > env->me_maxgc_ov1page * avail_gc_slots))
/* TODO: Можно смотреть последовательности какой длины есть в relist
* и пробовать нарезать куски соответствующего размера.
* Смысл в том, чтобы не дробить последовательности страниц,
* а использовать целиком. */
chunk = env->me_maxgc_ov1page +
left / (env->me_maxgc_ov1page * avail_gc_slots) *
env->me_maxgc_ov1page;
#else #else
if (chunk < env->me_maxgc_ov1page * 2) if (chunk < env->me_maxgc_ov1page * 2)
chunk /= 2; chunk /= 2;
@ -11377,7 +11391,7 @@ retry:
size_t avail = ((pgno2bytes(env, span) - PAGEHDRSZ) / size_t avail = ((pgno2bytes(env, span) - PAGEHDRSZ) /
sizeof(pgno_t)) /* - 1 + span */; sizeof(pgno_t)) /* - 1 + span */;
if (tail > avail) { if (tail > avail) {
for (size_t i = amount - span; i > 0; --i) { for (size_t i = ctx->amount - span; i > 0; --i) {
if (MDBX_PNL_ASCENDING ? (txn->tw.relist[i] + span) if (MDBX_PNL_ASCENDING ? (txn->tw.relist[i] + span)
: (txn->tw.relist[i] - span) == : (txn->tw.relist[i] - span) ==
txn->tw.relist[i + span]) { txn->tw.relist[i + span]) {
@ -11436,18 +11450,9 @@ retry:
gcu_clean_reserved(env, data); gcu_clean_reserved(env, data);
ctx->reserved += chunk; ctx->reserved += chunk;
TRACE("%s: settled %zu (+%zu), continue", dbg_prefix_mode, ctx->reserved, TRACE("%s: reserved %zu (+%zu), continue", dbg_prefix_mode, ctx->reserved,
chunk); chunk);
if (txn->tw.lifo_reclaimed &&
unlikely(amount < MDBX_PNL_GETSIZE(txn->tw.relist)) &&
(ctx->loop < 5 || MDBX_PNL_GETSIZE(txn->tw.relist) - amount >
env->me_maxgc_ov1page / 2)) {
NOTICE("** restart: reclaimed-list growth %zu -> %zu", amount,
MDBX_PNL_GETSIZE(txn->tw.relist));
goto retry_clean_adj;
}
continue; continue;
} }
@ -11467,13 +11472,12 @@ retry:
tASSERT(txn, pnl_check_allocated(txn->tw.relist, tASSERT(txn, pnl_check_allocated(txn->tw.relist,
txn->mt_next_pgno - MDBX_ENABLE_REFUND)); txn->mt_next_pgno - MDBX_ENABLE_REFUND));
tASSERT(txn, dirtylist_check(txn)); tASSERT(txn, dirtylist_check(txn));
if (ctx->reserved || MDBX_PNL_GETSIZE(txn->tw.relist)) { if (ctx->amount) {
MDBX_val key, data; MDBX_val key, data;
key.iov_len = data.iov_len = 0; /* avoid MSVC warning */ key.iov_len = data.iov_len = 0; /* avoid MSVC warning */
key.iov_base = data.iov_base = NULL; key.iov_base = data.iov_base = NULL;
const size_t amount = MDBX_PNL_GETSIZE(txn->tw.relist); size_t left = ctx->amount, excess = 0;
size_t left = amount, excess = 0;
if (txn->tw.lifo_reclaimed == nullptr) { if (txn->tw.lifo_reclaimed == nullptr) {
tASSERT(txn, ctx->lifo == 0); tASSERT(txn, ctx->lifo == 0);
rc = cursor_first(&ctx->cursor, &key, &data); rc = cursor_first(&ctx->cursor, &key, &data);
@ -11558,17 +11562,21 @@ retry:
gcu_clean_reserved(env, data); gcu_clean_reserved(env, data);
if (unlikely(txn->tw.loose_count || if (unlikely(txn->tw.loose_count ||
amount != MDBX_PNL_GETSIZE(txn->tw.relist))) { ctx->amount != MDBX_PNL_GETSIZE(txn->tw.relist))) {
NOTICE("** restart: reclaimed-list growth (%zu -> %zu, loose +%zu)", NOTICE("** restart: reclaimed-list changed (%zu -> %zu, loose +%zu)",
amount, MDBX_PNL_GETSIZE(txn->tw.relist), txn->tw.loose_count); ctx->amount, MDBX_PNL_GETSIZE(txn->tw.relist),
txn->tw.loose_count);
if (ctx->loop < 5 || (ctx->loop > 10 && (ctx->loop & 1)))
goto retry_clean_adj; goto retry_clean_adj;
goto retry;
} }
if (unlikely(txn->tw.lifo_reclaimed if (unlikely(txn->tw.lifo_reclaimed
? ctx->cleaned_slot < ? ctx->cleaned_slot <
MDBX_PNL_GETSIZE(txn->tw.lifo_reclaimed) MDBX_PNL_GETSIZE(txn->tw.lifo_reclaimed)
: ctx->cleaned_id < txn->tw.last_reclaimed)) { : ctx->cleaned_id < txn->tw.last_reclaimed)) {
NOTICE("%s", "** restart: reclaimed-slots changed"); NOTICE("%s", "** restart: reclaimed-slots changed");
goto retry_clean_adj; goto retry;
} }
if (unlikely(ctx->retired_stored != if (unlikely(ctx->retired_stored !=
MDBX_PNL_GETSIZE(txn->tw.retired_pages))) { MDBX_PNL_GETSIZE(txn->tw.retired_pages))) {
@ -11576,7 +11584,7 @@ retry:
ctx->retired_stored < MDBX_PNL_GETSIZE(txn->tw.retired_pages)); ctx->retired_stored < MDBX_PNL_GETSIZE(txn->tw.retired_pages));
NOTICE("** restart: retired-list growth (%zu -> %zu)", NOTICE("** restart: retired-list growth (%zu -> %zu)",
ctx->retired_stored, MDBX_PNL_GETSIZE(txn->tw.retired_pages)); ctx->retired_stored, MDBX_PNL_GETSIZE(txn->tw.retired_pages));
goto retry_clean_adj; goto retry;
} }
pgno_t *dst = data.iov_base; pgno_t *dst = data.iov_base;
@ -11590,7 +11598,7 @@ retry:
left -= chunk; left -= chunk;
if (AUDIT_ENABLED()) { if (AUDIT_ENABLED()) {
rc = audit_ex(txn, ctx->retired_stored + amount - left, true); rc = audit_ex(txn, ctx->retired_stored + ctx->amount - left, true);
if (unlikely(rc != MDBX_SUCCESS)) if (unlikely(rc != MDBX_SUCCESS))
goto bailout; goto bailout;
} }
@ -11621,9 +11629,11 @@ retry:
} }
tASSERT(txn, rc == MDBX_SUCCESS); tASSERT(txn, rc == MDBX_SUCCESS);
if (unlikely(txn->tw.loose_count != 0)) { if (unlikely(txn->tw.loose_count != 0 ||
NOTICE("** restart: got %zu loose pages", txn->tw.loose_count); ctx->amount != MDBX_PNL_GETSIZE(txn->tw.relist))) {
goto retry_clean_adj; NOTICE("** restart: got %zu loose pages (reclaimed-list %zu -> %zu)",
txn->tw.loose_count, ctx->amount, MDBX_PNL_GETSIZE(txn->tw.relist));
goto retry;
} }
if (unlikely(excess_slots)) { if (unlikely(excess_slots)) {