mirror of
https://github.com/isar/libmdbx.git
synced 2025-01-02 01:44:14 +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>
135 lines
4.7 KiB
C++
135 lines
4.7 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++"
|
|
|
|
registry *registry::instance() {
|
|
static registry *singleton;
|
|
if (!singleton)
|
|
singleton = new registry();
|
|
return singleton;
|
|
}
|
|
|
|
bool registry::add(const record *item) {
|
|
auto const singleton = instance();
|
|
assert(singleton->name2id.count(std::string(item->name)) == 0);
|
|
assert(singleton->id2record.count(item->id) == 0);
|
|
if (singleton->name2id.count(std::string(item->name)) +
|
|
singleton->id2record.count(item->id) ==
|
|
0) {
|
|
singleton->name2id[std::string(item->name)] = item;
|
|
singleton->id2record[item->id] = item;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
testcase *registry::create_actor(const actor_config &config,
|
|
const mdbx_pid_t pid) {
|
|
return instance()->id2record.at(config.testcase)->constructor(config, pid);
|
|
}
|
|
|
|
bool registry::review_actor_params(const actor_testcase id,
|
|
actor_params ¶ms) {
|
|
return instance()->id2record.at(id)->review_params(params);
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void configure_actor(unsigned &last_space_id, const actor_testcase testcase,
|
|
const char *space_id_cstr, actor_params params) {
|
|
unsigned wait4id = 0;
|
|
if (params.waitfor_nops) {
|
|
for (auto i = global::actors.rbegin(); i != global::actors.rend(); ++i) {
|
|
if (i->is_waitable(params.waitfor_nops)) {
|
|
if (i->signal_nops && i->signal_nops != params.waitfor_nops)
|
|
failure("Previous waitable actor (id=%u) already linked on %u-ops\n",
|
|
i->actor_id, i->signal_nops);
|
|
wait4id = i->actor_id;
|
|
i->signal_nops = params.waitfor_nops;
|
|
break;
|
|
}
|
|
}
|
|
if (!wait4id)
|
|
failure("No previous waitable actor for %u-ops\n", params.waitfor_nops);
|
|
}
|
|
|
|
unsigned long space_id = 0;
|
|
if (!space_id_cstr || strcmp(space_id_cstr, "auto") == 0)
|
|
space_id = last_space_id + 1;
|
|
else {
|
|
char *end = nullptr;
|
|
errno = 0;
|
|
space_id = strtoul(space_id_cstr, &end, 0);
|
|
if (errno)
|
|
failure_perror("Expects an integer value for space-id\n", errno);
|
|
if (end && *end)
|
|
failure("The '%s' is unexpected for space-id\n", end);
|
|
}
|
|
|
|
if (!registry::review_actor_params(testcase, params))
|
|
failure("Actor config-review failed for space-id %lu\n", space_id);
|
|
|
|
if (space_id > ACTOR_ID_MAX)
|
|
failure("Invalid space-id %lu\n", space_id);
|
|
last_space_id = unsigned(space_id);
|
|
|
|
log_trace("configure_actor: space %lu for %s", space_id,
|
|
testcase2str(testcase));
|
|
global::actors.emplace_back(
|
|
actor_config(testcase, params, unsigned(space_id), wait4id));
|
|
global::databases.insert(params.pathname_db);
|
|
}
|
|
|
|
void testcase_setup(const char *casename, const actor_params ¶ms,
|
|
unsigned &last_space_id) {
|
|
if (strcmp(casename, "basic") == 0) {
|
|
log_notice(">>> testcase_setup(%s)", casename);
|
|
configure_actor(last_space_id, ac_nested, nullptr, params);
|
|
configure_actor(last_space_id, ac_hill, nullptr, params);
|
|
configure_actor(last_space_id, ac_ttl, nullptr, params);
|
|
configure_actor(last_space_id, ac_copy, nullptr, params);
|
|
configure_actor(last_space_id, ac_append, nullptr, params);
|
|
configure_actor(last_space_id, ac_jitter, nullptr, params);
|
|
configure_actor(last_space_id, ac_try, nullptr, params);
|
|
configure_actor(last_space_id, ac_jitter, nullptr, params);
|
|
configure_actor(last_space_id, ac_try, nullptr, params);
|
|
log_notice("<<< testcase_setup(%s): done", casename);
|
|
} else {
|
|
failure("unknown testcase `%s`", casename);
|
|
}
|
|
}
|
|
|
|
void keycase_setup(const char *casename, actor_params ¶ms) {
|
|
if (strcmp(casename, "random") == 0 || strcmp(casename, "prng") == 0) {
|
|
log_notice(">>> keycase_setup(%s)", casename);
|
|
params.keygen.keycase = kc_random;
|
|
// TODO
|
|
log_notice("<<< keycase_setup(%s): done", casename);
|
|
} else if (strcmp(casename, "dashes") == 0 ||
|
|
strcmp(casename, "aside") == 0) {
|
|
log_notice(">>> keycase_setup(%s)", casename);
|
|
params.keygen.keycase = kc_dashes;
|
|
// TODO
|
|
log_notice("<<< keycase_setup(%s): done", casename);
|
|
} else if (strcmp(casename, "custom") == 0) {
|
|
log_notice("=== keycase_setup(%s): skip", casename);
|
|
params.keygen.keycase = kc_custom;
|
|
} else {
|
|
failure("unknown keycase `%s`", casename);
|
|
}
|
|
}
|
|
|
|
/* TODO */
|