mirror of
https://github.com/isar/libmdbx.git
synced 2025-01-04 19:14:12 +08:00
aea40fb79f
Поддерживающий выпуск с исправлением обнаруженных ошибок и устранением недочетов в память Героя России гвардии майора Дмитрия Семёнова с позывным "СЭМ". Значимые исправления и доработки: --------------------------------- - Устранение унаследованной от LMDB ошибки приводящей к повреждению БД при использовании `MDBX_DUPFIXED`. - Исправление ложной ошибки `MDBX_CORRUPTED (-30796)` в сценарии работы в режиме `MDBX_DUPFIXED` и нечетной длинной мульти-значений. - Исправление недочета корректировки сопутствующих курсоров при разделении страницы по сценарию добавления пустой страницы слева. - Доработка `rebalance()` ради уменьшения WAF. - Исправление assert-проверки внутри `check_txn()` для случая завершенных транзакций в режиме `MDBX_NO_TLS`. Последствий ошибки, кроме срабатывания assert-проверки в отладочных сборках, нет. - Устранение ошибки при открытии БД на файловой системе только-для-чтения. - Удалены излишне строгие проверки в утилите `mdbx_chk`, которые приводили к ложно-позитивным ошибкам при проверке БД после серии последних доработок. Более подробная информация в [ChangeLog](https://libmdbx.dqdkfa.ru/md__change_log.html). git diff' stat: 19 commits, 57 files changed, 751 insertions(+), 331 deletions(-) Signed-off-by: Леонид Юрьев (Leonid Yuriev) <leo@yuriev.ru>
473 lines
16 KiB
C++
473 lines
16 KiB
C++
/*
|
||
* Copyright 2017-2024 Leonid Yuriev <leo@yuriev.ru>
|
||
* and other libmdbx authors: please see AUTHORS file.
|
||
* All rights reserved.
|
||
*
|
||
* Redistribution and use in source and binary forms, with or without
|
||
* modification, are permitted only as authorized by the OpenLDAP
|
||
* Public License.
|
||
*
|
||
* A copy of this license is available in the file LICENSE in the
|
||
* top-level directory of the distribution or, alternatively, at
|
||
* <http://www.OpenLDAP.org/license.html>.
|
||
*/
|
||
|
||
#include "test.h++"
|
||
|
||
/* LY: тест "холмиком":
|
||
* - сначала наполняем таблицу циклическими CRUD-манипуляциями,
|
||
* которые в каждом цикле делают несколько операций, включая удаление,
|
||
* но в результате добавляют записи.
|
||
* - затем очищаем таблицу также CRUD-манипуляциями, но уже с другой
|
||
* пропорцией удалений.
|
||
*
|
||
* При этом очень многое зависит от порядка перебора ключей:
|
||
* - (псевдо)случайное распределение требуется лишь для полноты картины,
|
||
* но в целом не покрывает важных кейсов.
|
||
* - кроме (псевдо)случайного перебора требуется последовательное
|
||
* итерирование ключей интервалами различной ширины, с тем чтобы
|
||
* проверить различные варианты как разделения, так и слияния страниц
|
||
* внутри движка.
|
||
* - при не-уникальных ключах (MDBX_DUPSORT с подвариантами), для каждого
|
||
* повтора внутри движка формируется вложенное btree-дерево,
|
||
* соответственно требуется соблюдение аналогичных принципов
|
||
* итерирования для значений.
|
||
*/
|
||
|
||
class testcase_hill : public testcase {
|
||
public:
|
||
testcase_hill(const actor_config &config, const mdbx_pid_t pid)
|
||
: testcase(config, pid) {}
|
||
bool run() override;
|
||
};
|
||
REGISTER_TESTCASE(hill);
|
||
|
||
bool testcase_hill::run() {
|
||
int err = db_open__begin__table_create_open_clean(dbi);
|
||
if (unlikely(err != MDBX_SUCCESS)) {
|
||
log_notice("hill: bailout-prepare due '%s'", mdbx_strerror(err));
|
||
return false;
|
||
}
|
||
speculum.clear();
|
||
speculum_committed.clear();
|
||
|
||
/* TODO: работа в несколько потоков */
|
||
keyvalue_maker.setup(config.params, config.actor_id, 0 /* thread_number */);
|
||
|
||
keygen::buffer a_key = keygen::alloc(config.params.keylen_max);
|
||
keygen::buffer a_data_0 = keygen::alloc(config.params.datalen_max);
|
||
keygen::buffer a_data_1 = keygen::alloc(config.params.datalen_max);
|
||
keygen::buffer b_key = keygen::alloc(config.params.keylen_max);
|
||
keygen::buffer b_data = keygen::alloc(config.params.datalen_max);
|
||
|
||
const MDBX_put_flags_t insert_flags =
|
||
(config.params.table_flags & MDBX_DUPSORT)
|
||
? MDBX_NODUPDATA
|
||
: MDBX_NODUPDATA | MDBX_NOOVERWRITE;
|
||
const MDBX_put_flags_t update_flags =
|
||
(config.params.table_flags & MDBX_DUPSORT)
|
||
? MDBX_CURRENT | MDBX_NODUPDATA | MDBX_NOOVERWRITE
|
||
: MDBX_NODUPDATA;
|
||
|
||
uint64_t serial_count = 0;
|
||
uint64_t committed_serial = serial_count;
|
||
unsigned txn_nops = 0;
|
||
|
||
bool rc = speculum_verify();
|
||
if (!rc) {
|
||
log_notice("uphill: bailout before main loop");
|
||
goto bailout;
|
||
}
|
||
|
||
while (should_continue()) {
|
||
const keygen::serial_t a_serial = serial_count;
|
||
if (unlikely(!keyvalue_maker.increment(serial_count, 1))) {
|
||
log_notice("uphill: unexpected key-space overflow");
|
||
break;
|
||
}
|
||
|
||
const keygen::serial_t b_serial = serial_count;
|
||
assert(b_serial > a_serial);
|
||
|
||
// создаем первую запись из пары
|
||
const keygen::serial_t age_shift = UINT64_C(1) << (a_serial % 31);
|
||
log_trace("uphill: insert-a (age %" PRIu64 ") %" PRIu64, age_shift,
|
||
a_serial);
|
||
generate_pair(a_serial, a_key, a_data_1, age_shift);
|
||
|
||
err = insert(a_key, a_data_1, insert_flags);
|
||
if (unlikely(err != MDBX_SUCCESS)) {
|
||
if (err == MDBX_MAP_FULL && config.params.ignore_dbfull) {
|
||
log_notice("uphill: bailout at insert-a due '%s'", mdbx_strerror(err));
|
||
txn_restart(true, false);
|
||
serial_count = committed_serial;
|
||
speculum = speculum_committed;
|
||
break;
|
||
}
|
||
failure_perror("mdbx_put(insert-a.1)", err);
|
||
}
|
||
if (!speculum_verify()) {
|
||
log_notice("uphill: bailout after insert-a, before commit");
|
||
goto bailout;
|
||
}
|
||
|
||
if (++txn_nops >= config.params.batch_write) {
|
||
err = breakable_restart();
|
||
if (unlikely(err != MDBX_SUCCESS)) {
|
||
log_notice("uphill: bailout at commit due '%s'", mdbx_strerror(err));
|
||
serial_count = committed_serial;
|
||
speculum = speculum_committed;
|
||
break;
|
||
}
|
||
speculum_committed = speculum;
|
||
committed_serial = a_serial;
|
||
txn_nops = 0;
|
||
if (!speculum_verify()) {
|
||
log_notice("uphill: bailout after insert-a, after commit");
|
||
goto bailout;
|
||
}
|
||
}
|
||
|
||
// создаем вторую запись из пары
|
||
log_trace("uphill: insert-b %" PRIu64, b_serial);
|
||
generate_pair(b_serial, b_key, b_data, 0);
|
||
err = insert(b_key, b_data, insert_flags);
|
||
if (unlikely(err != MDBX_SUCCESS)) {
|
||
if (err == MDBX_MAP_FULL && config.params.ignore_dbfull) {
|
||
log_notice("uphill: bailout at insert-b due '%s'", mdbx_strerror(err));
|
||
txn_restart(true, false);
|
||
serial_count = committed_serial;
|
||
speculum = speculum_committed;
|
||
break;
|
||
}
|
||
failure_perror("mdbx_put(insert-b)", err);
|
||
}
|
||
if (!speculum_verify()) {
|
||
log_notice("uphill: bailout after insert-b, before commit");
|
||
goto bailout;
|
||
}
|
||
|
||
if (++txn_nops >= config.params.batch_write) {
|
||
err = breakable_restart();
|
||
if (unlikely(err != MDBX_SUCCESS)) {
|
||
log_notice("uphill: bailout at commit due '%s'", mdbx_strerror(err));
|
||
serial_count = committed_serial;
|
||
speculum = speculum_committed;
|
||
break;
|
||
}
|
||
speculum_committed = speculum;
|
||
committed_serial = a_serial;
|
||
txn_nops = 0;
|
||
if (!speculum_verify()) {
|
||
log_notice("uphill: bailout after insert-b, after commit");
|
||
goto bailout;
|
||
}
|
||
}
|
||
|
||
// обновляем данные в первой записи
|
||
log_trace("uphill: update-a (age %" PRIu64 "->0) %" PRIu64, age_shift,
|
||
a_serial);
|
||
generate_pair(a_serial, a_key, a_data_0, 0);
|
||
checkdata("uphill: update-a", dbi, a_key->value, a_data_1->value);
|
||
err = replace(a_key, a_data_0, a_data_1, update_flags);
|
||
if (unlikely(err != MDBX_SUCCESS)) {
|
||
if (err == MDBX_MAP_FULL && config.params.ignore_dbfull) {
|
||
log_notice("uphill: bailout at update-a due '%s'", mdbx_strerror(err));
|
||
txn_restart(true, false);
|
||
serial_count = committed_serial;
|
||
speculum = speculum_committed;
|
||
break;
|
||
}
|
||
failure_perror("mdbx_replace(update-a: 1->0)", err);
|
||
}
|
||
if (!speculum_verify()) {
|
||
log_notice("uphill: bailout after update-a, before commit");
|
||
goto bailout;
|
||
}
|
||
|
||
if (++txn_nops >= config.params.batch_write) {
|
||
err = breakable_restart();
|
||
if (unlikely(err != MDBX_SUCCESS)) {
|
||
log_notice("uphill: bailout at commit due '%s'", mdbx_strerror(err));
|
||
serial_count = committed_serial;
|
||
speculum = speculum_committed;
|
||
break;
|
||
}
|
||
speculum_committed = speculum;
|
||
committed_serial = a_serial;
|
||
txn_nops = 0;
|
||
if (!speculum_verify()) {
|
||
log_notice("uphill: bailout after update-a, after commit");
|
||
goto bailout;
|
||
}
|
||
}
|
||
|
||
// удаляем вторую запись
|
||
log_trace("uphill: delete-b %" PRIu64, b_serial);
|
||
checkdata("uphill: delete-b", dbi, b_key->value, b_data->value);
|
||
err = remove(b_key, b_data);
|
||
if (unlikely(err != MDBX_SUCCESS)) {
|
||
if (err == MDBX_MAP_FULL && config.params.ignore_dbfull) {
|
||
log_notice("uphill: bailout at delete-b due '%s'", mdbx_strerror(err));
|
||
txn_restart(true, false);
|
||
serial_count = committed_serial;
|
||
speculum = speculum_committed;
|
||
break;
|
||
}
|
||
failure_perror("mdbx_del(b)", err);
|
||
}
|
||
if (!speculum_verify()) {
|
||
log_notice("uphill: bailout after delete-b, before commit");
|
||
goto bailout;
|
||
}
|
||
|
||
if (++txn_nops >= config.params.batch_write) {
|
||
err = breakable_restart();
|
||
if (unlikely(err != MDBX_SUCCESS)) {
|
||
log_notice("uphill: bailout at commit due '%s'", mdbx_strerror(err));
|
||
serial_count = committed_serial;
|
||
speculum = speculum_committed;
|
||
break;
|
||
}
|
||
speculum_committed = speculum;
|
||
committed_serial = a_serial;
|
||
txn_nops = 0;
|
||
if (!speculum_verify()) {
|
||
log_notice("uphill: bailout after delete-b, after commit");
|
||
goto bailout;
|
||
}
|
||
}
|
||
|
||
report(1);
|
||
if (!keyvalue_maker.increment(serial_count, 1)) {
|
||
// дошли до границы пространства ключей
|
||
serial_count = a_serial;
|
||
goto overflow;
|
||
}
|
||
}
|
||
|
||
if (txn_guard) {
|
||
MDBX_stat stat;
|
||
err = mdbx_dbi_stat(txn_guard.get(), dbi, &stat, sizeof(stat));
|
||
if (unlikely(err != MDBX_SUCCESS))
|
||
failure_perror("mdbx_dbi_stat()", err);
|
||
|
||
uint32_t nested_deepmask;
|
||
err = mdbx_dbi_dupsort_depthmask(txn_guard.get(), dbi, &nested_deepmask);
|
||
if (unlikely(err != MDBX_SUCCESS && err != MDBX_RESULT_TRUE))
|
||
failure_perror("mdbx_dbi_stat_nested_deepmask()", err);
|
||
|
||
if (err != MDBX_SUCCESS) {
|
||
log_notice("hill: reached %d tree depth", stat.ms_depth);
|
||
} else {
|
||
std::string str;
|
||
int prev = -2, i = 0;
|
||
do {
|
||
while (!(nested_deepmask & 1))
|
||
++i, nested_deepmask >>= 1;
|
||
if (prev + 1 == i) {
|
||
if (str.back() != '-')
|
||
str.push_back('-');
|
||
prev = i;
|
||
continue;
|
||
}
|
||
if (!str.empty()) {
|
||
if (str.back() == '-')
|
||
str.append(std::to_string(prev));
|
||
str.push_back(',');
|
||
}
|
||
str.append(std::to_string(i));
|
||
prev = i;
|
||
} while (++i, nested_deepmask >>= 1);
|
||
if (str.back() == '-')
|
||
str.append(std::to_string(prev));
|
||
|
||
log_notice("hill: reached %d tree depth & %s sub-tree depth(s)",
|
||
stat.ms_depth, str.c_str());
|
||
}
|
||
|
||
if ((config.params.table_flags & MDBX_DUPSORT) == 0) {
|
||
if (!check_batch_get())
|
||
failure("batch-get verification failed");
|
||
}
|
||
}
|
||
|
||
while (serial_count > 1) {
|
||
if (unlikely(!keyvalue_maker.increment(serial_count, -2)))
|
||
failure("downhill: unexpected key-space underflow");
|
||
|
||
overflow:
|
||
const keygen::serial_t a_serial = serial_count;
|
||
const keygen::serial_t b_serial = a_serial + 1;
|
||
assert(b_serial > a_serial);
|
||
|
||
// обновляем первую запись из пары
|
||
const keygen::serial_t age_shift = UINT64_C(1) << (a_serial % 31);
|
||
log_trace("downhill: update-a (age 0->%" PRIu64 ") %" PRIu64, age_shift,
|
||
a_serial);
|
||
generate_pair(a_serial, a_key, a_data_0, 0);
|
||
generate_pair(a_serial, a_key, a_data_1, age_shift);
|
||
checkdata("downhill: update-a", dbi, a_key->value, a_data_0->value);
|
||
err = replace(a_key, a_data_1, a_data_0, update_flags);
|
||
if (unlikely(err != MDBX_SUCCESS)) {
|
||
if (err == MDBX_MAP_FULL && config.params.ignore_dbfull) {
|
||
log_notice("downhill: bailout at update-a due '%s'",
|
||
mdbx_strerror(err));
|
||
txn_end(true);
|
||
speculum = speculum_committed;
|
||
break;
|
||
}
|
||
failure_perror("mdbx_put(update-a: 0->1)", err);
|
||
}
|
||
if (!speculum_verify()) {
|
||
log_notice("downhill: bailout after update-a, before commit");
|
||
break;
|
||
}
|
||
|
||
if (++txn_nops >= config.params.batch_write) {
|
||
err = breakable_restart();
|
||
if (unlikely(err != MDBX_SUCCESS)) {
|
||
log_notice("downhill: bailout at commit due '%s'", mdbx_strerror(err));
|
||
speculum = speculum_committed;
|
||
break;
|
||
}
|
||
speculum_committed = speculum;
|
||
txn_nops = 0;
|
||
if (!speculum_verify()) {
|
||
log_notice("downhill: bailout after update-a, after commit");
|
||
break;
|
||
}
|
||
}
|
||
|
||
// создаем вторую запись из пары
|
||
log_trace("downhill: insert-b %" PRIu64, b_serial);
|
||
generate_pair(b_serial, b_key, b_data, 0);
|
||
err = insert(b_key, b_data, insert_flags);
|
||
if (unlikely(err != MDBX_SUCCESS)) {
|
||
if (err == MDBX_MAP_FULL && config.params.ignore_dbfull) {
|
||
log_notice("downhill: bailout at insert-a due '%s'",
|
||
mdbx_strerror(err));
|
||
txn_end(true);
|
||
speculum = speculum_committed;
|
||
break;
|
||
}
|
||
failure_perror("mdbx_put(insert-b)", err);
|
||
}
|
||
if (!speculum_verify()) {
|
||
log_notice("downhill: bailout after insert-b, before commit");
|
||
break;
|
||
}
|
||
|
||
if (++txn_nops >= config.params.batch_write) {
|
||
err = breakable_restart();
|
||
if (unlikely(err != MDBX_SUCCESS)) {
|
||
log_notice("downhill: bailout at commit due '%s'", mdbx_strerror(err));
|
||
speculum = speculum_committed;
|
||
break;
|
||
}
|
||
speculum_committed = speculum;
|
||
txn_nops = 0;
|
||
if (!speculum_verify()) {
|
||
log_notice("downhill: bailout after insert-b, after commit");
|
||
break;
|
||
}
|
||
}
|
||
|
||
// удаляем первую запись
|
||
log_trace("downhill: delete-a (age %" PRIu64 ") %" PRIu64, age_shift,
|
||
a_serial);
|
||
checkdata("downhill: delete-a", dbi, a_key->value, a_data_1->value);
|
||
err = remove(a_key, a_data_1);
|
||
if (unlikely(err != MDBX_SUCCESS)) {
|
||
if (err == MDBX_MAP_FULL && config.params.ignore_dbfull) {
|
||
log_notice("downhill: bailout at delete-a due '%s'",
|
||
mdbx_strerror(err));
|
||
txn_end(true);
|
||
speculum = speculum_committed;
|
||
break;
|
||
}
|
||
failure_perror("mdbx_del(a)", err);
|
||
}
|
||
if (!speculum_verify()) {
|
||
log_notice("downhill: bailout after delete-a, before commit");
|
||
break;
|
||
}
|
||
|
||
if (++txn_nops >= config.params.batch_write) {
|
||
err = breakable_restart();
|
||
if (unlikely(err != MDBX_SUCCESS)) {
|
||
log_notice("downhill: bailout at commit due '%s'", mdbx_strerror(err));
|
||
speculum = speculum_committed;
|
||
break;
|
||
}
|
||
speculum_committed = speculum;
|
||
txn_nops = 0;
|
||
if (!speculum_verify()) {
|
||
log_notice("downhill: bailout after delete-a, after commit");
|
||
break;
|
||
}
|
||
}
|
||
|
||
// удаляем вторую запись
|
||
log_trace("downhill: delete-b %" PRIu64, b_serial);
|
||
checkdata("downhill: delete-b", dbi, b_key->value, b_data->value);
|
||
err = remove(b_key, b_data);
|
||
if (unlikely(err != MDBX_SUCCESS)) {
|
||
if (err == MDBX_MAP_FULL && config.params.ignore_dbfull) {
|
||
log_notice("downhill: bailout at delete-b due '%s'",
|
||
mdbx_strerror(err));
|
||
txn_end(true);
|
||
speculum = speculum_committed;
|
||
break;
|
||
}
|
||
failure_perror("mdbx_del(b)", err);
|
||
}
|
||
if (!speculum_verify()) {
|
||
log_notice("downhill: bailout after delete-b, before commit");
|
||
break;
|
||
}
|
||
|
||
if (++txn_nops >= config.params.batch_write) {
|
||
err = breakable_restart();
|
||
if (unlikely(err != MDBX_SUCCESS)) {
|
||
log_notice("downhill: bailout at commit due '%s'", mdbx_strerror(err));
|
||
speculum = speculum_committed;
|
||
break;
|
||
}
|
||
speculum_committed = speculum;
|
||
txn_nops = 0;
|
||
if (!speculum_verify()) {
|
||
log_notice("downhill: bailout after delete-b, after commit");
|
||
goto bailout;
|
||
}
|
||
}
|
||
|
||
report(1);
|
||
}
|
||
|
||
rc = speculum_verify();
|
||
bailout:
|
||
if (txn_guard) {
|
||
err = breakable_commit();
|
||
if (unlikely(err != MDBX_SUCCESS))
|
||
log_notice("downhill: bailout at commit due '%s'", mdbx_strerror(err));
|
||
}
|
||
|
||
if (!rc && err == MDBX_MAP_FULL && config.params.ignore_dbfull)
|
||
rc = true;
|
||
if (dbi) {
|
||
if (config.params.drop_table && !mode_readonly()) {
|
||
txn_begin(false);
|
||
db_table_drop(dbi);
|
||
err = breakable_commit();
|
||
if (unlikely(err != MDBX_SUCCESS)) {
|
||
log_notice("hill: bailout-clean due '%s'", mdbx_strerror(err));
|
||
if (err != MDBX_MAP_FULL || !config.params.ignore_dbfull)
|
||
rc = false;
|
||
}
|
||
} else
|
||
db_table_close(dbi);
|
||
}
|
||
return rc;
|
||
}
|