mirror of
https://github.com/isar/libmdbx.git
synced 2025-04-01 02:32:57 +08:00
mdbx-tests: дополнение extra/txn (backport).
This commit is contained in:
parent
5f37ea60d2
commit
9c177de034
@ -1,4 +1,5 @@
|
|||||||
#include "mdbx.h++"
|
#include "mdbx.h++"
|
||||||
|
#include MDBX_CONFIG_H
|
||||||
|
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
|
|
||||||
@ -25,267 +26,301 @@ static void logger_nofmt(MDBX_log_level_t loglevel, const char *function, int li
|
|||||||
fprintf(stdout, "%s:%u %s", function, line, msg);
|
fprintf(stdout, "%s:%u %s", function, line, msg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool case0(const mdbx::path &path) {
|
||||||
|
mdbx::env_managed::create_parameters createParameters;
|
||||||
|
createParameters.geometry.make_dynamic(21 * mdbx::env::geometry::MiB, 84 * mdbx::env::geometry::MiB);
|
||||||
|
|
||||||
|
mdbx::env::operate_parameters operateParameters(100, 10);
|
||||||
|
operateParameters.options.no_sticky_threads = false;
|
||||||
|
mdbx::env_managed env(path, createParameters, operateParameters);
|
||||||
|
auto txn = env.start_write(false);
|
||||||
|
/* mdbx::map_handle testHandle = */ txn.create_map("xyz", mdbx::key_mode::usual, mdbx::value_mode::single);
|
||||||
|
txn.commit();
|
||||||
|
|
||||||
|
//-------------------------------------
|
||||||
|
txn = env.start_write();
|
||||||
|
MDBX_txn *c_txn = txn;
|
||||||
|
int err = mdbx_txn_reset(txn);
|
||||||
|
assert(err == MDBX_EINVAL);
|
||||||
|
bool ok = err == MDBX_EINVAL;
|
||||||
|
|
||||||
|
err = mdbx_txn_break(txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
|
||||||
|
err = mdbx_txn_commit(txn);
|
||||||
|
assert(err == MDBX_RESULT_TRUE);
|
||||||
|
ok = ok && err == MDBX_RESULT_TRUE;
|
||||||
|
|
||||||
|
//-------------------------------------
|
||||||
|
err = mdbx_txn_begin(env, nullptr, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
assert(c_txn == (const MDBX_txn *)txn);
|
||||||
|
|
||||||
|
err = mdbx_txn_break(txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
|
||||||
|
err = mdbx_txn_reset(txn);
|
||||||
|
assert(err == MDBX_EINVAL);
|
||||||
|
ok = ok && err == MDBX_EINVAL;
|
||||||
|
|
||||||
|
err = mdbx_txn_commit(txn);
|
||||||
|
assert(err == MDBX_RESULT_TRUE);
|
||||||
|
ok = ok && err == MDBX_RESULT_TRUE;
|
||||||
|
|
||||||
|
err = mdbx_txn_abort(c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
//-------------------------------------
|
||||||
|
err = mdbx_txn_begin(env, nullptr, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
assert(c_txn == (const MDBX_txn *)txn);
|
||||||
|
txn.commit();
|
||||||
|
|
||||||
|
err = mdbx_txn_reset(c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
err = mdbx_txn_break(c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
err = mdbx_txn_abort(c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
|
||||||
|
//=====================================
|
||||||
|
|
||||||
|
txn = env.start_read();
|
||||||
|
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
txn.make_broken();
|
||||||
|
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
txn.reset_reading();
|
||||||
|
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
txn.abort();
|
||||||
|
|
||||||
|
//-------------------------------------
|
||||||
|
|
||||||
|
txn = env.start_read();
|
||||||
|
txn.reset_reading();
|
||||||
|
txn.make_broken();
|
||||||
|
txn.abort();
|
||||||
|
|
||||||
|
//=====================================
|
||||||
|
|
||||||
|
std::latch s(1);
|
||||||
|
txn = env.start_read();
|
||||||
|
c_txn = txn;
|
||||||
|
|
||||||
|
std::thread t([&]() {
|
||||||
|
s.wait();
|
||||||
|
#if MDBX_TXN_CHECKOWNER
|
||||||
|
err = mdbx_txn_reset(c_txn);
|
||||||
|
assert(err == MDBX_THREAD_MISMATCH);
|
||||||
|
ok = ok && err == MDBX_THREAD_MISMATCH;
|
||||||
|
err = mdbx_txn_break(c_txn);
|
||||||
|
assert(err == MDBX_THREAD_MISMATCH);
|
||||||
|
ok = ok && err == MDBX_THREAD_MISMATCH;
|
||||||
|
err = mdbx_txn_commit(c_txn);
|
||||||
|
assert(err == MDBX_THREAD_MISMATCH);
|
||||||
|
ok = ok && err == MDBX_THREAD_MISMATCH;
|
||||||
|
err = mdbx_txn_abort(c_txn);
|
||||||
|
assert(err == MDBX_THREAD_MISMATCH);
|
||||||
|
ok = ok && err == MDBX_THREAD_MISMATCH;
|
||||||
|
#endif /* MDBX_TXN_CHECKOWNER */
|
||||||
|
|
||||||
|
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
#if MDBX_TXN_CHECKOWNER
|
||||||
|
assert(err == MDBX_THREAD_MISMATCH);
|
||||||
|
ok = ok && err == MDBX_THREAD_MISMATCH;
|
||||||
|
#else
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
#endif /* MDBX_TXN_CHECKOWNER */
|
||||||
|
});
|
||||||
|
|
||||||
|
s.count_down();
|
||||||
|
t.join();
|
||||||
|
|
||||||
|
return ok;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool case1(const mdbx::path &path) {
|
||||||
|
mdbx::env::operate_parameters operateParameters(100, 10);
|
||||||
|
operateParameters.options.no_sticky_threads = true;
|
||||||
|
operateParameters.options.nested_write_transactions = true;
|
||||||
|
mdbx::env_managed env(path, operateParameters);
|
||||||
|
|
||||||
|
//-------------------------------------
|
||||||
|
auto txn = env.start_write();
|
||||||
|
MDBX_txn *c_txn = txn;
|
||||||
|
int err = mdbx_txn_reset(txn);
|
||||||
|
assert(err == MDBX_EINVAL);
|
||||||
|
bool ok = err == MDBX_EINVAL;
|
||||||
|
|
||||||
|
err = mdbx_txn_break(txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
|
||||||
|
err = mdbx_txn_commit(txn);
|
||||||
|
assert(err == MDBX_RESULT_TRUE);
|
||||||
|
ok = ok && err == MDBX_RESULT_TRUE;
|
||||||
|
|
||||||
|
//-------------------------------------
|
||||||
|
err = mdbx_txn_begin(env, nullptr, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
assert(c_txn == (const MDBX_txn *)txn);
|
||||||
|
|
||||||
|
err = mdbx_txn_break(txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
|
||||||
|
err = mdbx_txn_reset(txn);
|
||||||
|
assert(err == MDBX_EINVAL);
|
||||||
|
ok = ok && err == MDBX_EINVAL;
|
||||||
|
|
||||||
|
err = mdbx_txn_commit(txn);
|
||||||
|
assert(err == MDBX_RESULT_TRUE);
|
||||||
|
ok = ok && err == MDBX_RESULT_TRUE;
|
||||||
|
|
||||||
|
err = mdbx_txn_abort(c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
//-------------------------------------
|
||||||
|
err = mdbx_txn_begin(env, nullptr, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
assert(c_txn == (const MDBX_txn *)txn);
|
||||||
|
txn.commit();
|
||||||
|
|
||||||
|
err = mdbx_txn_reset(c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
err = mdbx_txn_break(c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
err = mdbx_txn_abort(c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
|
||||||
|
//=====================================
|
||||||
|
|
||||||
|
txn = env.start_read();
|
||||||
|
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
txn.make_broken();
|
||||||
|
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
txn.reset_reading();
|
||||||
|
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_BAD_TXN);
|
||||||
|
ok = ok && err == MDBX_BAD_TXN;
|
||||||
|
txn.abort();
|
||||||
|
|
||||||
|
//-------------------------------------
|
||||||
|
|
||||||
|
txn = env.start_read();
|
||||||
|
txn.reset_reading();
|
||||||
|
txn.make_broken();
|
||||||
|
txn.abort();
|
||||||
|
|
||||||
|
//=====================================
|
||||||
|
|
||||||
|
std::latch s1(1), s2(1), s3(1);
|
||||||
|
txn = env.start_read();
|
||||||
|
c_txn = txn;
|
||||||
|
|
||||||
|
std::thread t([&]() {
|
||||||
|
s1.wait();
|
||||||
|
err = mdbx_txn_break(c_txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
err = mdbx_txn_reset(c_txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
txn.renew_reading();
|
||||||
|
s2.count_down();
|
||||||
|
|
||||||
|
s3.wait();
|
||||||
|
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
err = mdbx_txn_commit(c_txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
c_txn = txn;
|
||||||
|
err = mdbx_txn_commit(c_txn);
|
||||||
|
assert(err == MDBX_THREAD_MISMATCH);
|
||||||
|
ok = ok && err == MDBX_THREAD_MISMATCH;
|
||||||
|
err = mdbx_txn_abort(c_txn);
|
||||||
|
assert(err == MDBX_THREAD_MISMATCH);
|
||||||
|
ok = ok && err == MDBX_THREAD_MISMATCH;
|
||||||
|
err = mdbx_txn_break(c_txn);
|
||||||
|
assert(err == MDBX_SUCCESS);
|
||||||
|
ok = ok && err == MDBX_SUCCESS;
|
||||||
|
err = mdbx_txn_reset(c_txn);
|
||||||
|
assert(err == MDBX_EINVAL);
|
||||||
|
ok = ok && err == MDBX_EINVAL;
|
||||||
|
});
|
||||||
|
|
||||||
|
s1.count_down();
|
||||||
|
s2.wait();
|
||||||
|
txn.commit();
|
||||||
|
txn = env.start_write();
|
||||||
|
s3.count_down();
|
||||||
|
|
||||||
|
t.join();
|
||||||
|
txn.abort();
|
||||||
|
|
||||||
|
return ok;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool case2(const mdbx::path &path, bool no_sticky_threads) {
|
||||||
|
mdbx::env::operate_parameters operateParameters(100, 10);
|
||||||
|
operateParameters.options.no_sticky_threads = no_sticky_threads;
|
||||||
|
mdbx::env_managed env(path, operateParameters);
|
||||||
|
|
||||||
|
std::latch s(1);
|
||||||
|
std::vector<std::thread> l;
|
||||||
|
for (size_t n = 0; n < 8; ++n)
|
||||||
|
l.push_back(std::thread([&]() {
|
||||||
|
s.wait();
|
||||||
|
for (size_t i = 0; i < 1000000; ++i) {
|
||||||
|
auto txn = env.start_read();
|
||||||
|
txn.abort();
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
|
||||||
|
s.count_down();
|
||||||
|
for (auto &t : l)
|
||||||
|
t.join();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
int main(int argc, const char *argv[]) {
|
int main(int argc, const char *argv[]) {
|
||||||
(void)argc;
|
(void)argc;
|
||||||
(void)argv;
|
(void)argv;
|
||||||
bool ok = true;
|
|
||||||
int err;
|
|
||||||
|
|
||||||
mdbx_setup_debug_nofmt(MDBX_LOG_VERBOSE, MDBX_DBG_ASSERT, logger_nofmt, log_buffer, sizeof(log_buffer));
|
mdbx_setup_debug_nofmt(MDBX_LOG_VERBOSE, MDBX_DBG_ASSERT, logger_nofmt, log_buffer, sizeof(log_buffer));
|
||||||
|
|
||||||
mdbx::path path = "test-txn";
|
mdbx::path path = "test-txn";
|
||||||
mdbx::env::remove(path);
|
mdbx::env::remove(path);
|
||||||
mdbx::env::operate_parameters operateParameters(100, 10);
|
|
||||||
|
|
||||||
{
|
bool ok = case0(path);
|
||||||
mdbx::env_managed::create_parameters createParameters;
|
ok = case1(path) && ok;
|
||||||
createParameters.geometry.make_dynamic(21 * mdbx::env::geometry::MiB, 84 * mdbx::env::geometry::MiB);
|
ok = case2(path, false) && ok;
|
||||||
|
ok = case2(path, true) && ok;
|
||||||
operateParameters.options.no_sticky_threads = false;
|
|
||||||
mdbx::env_managed env(path, createParameters, operateParameters);
|
|
||||||
auto txn = env.start_write(false);
|
|
||||||
/* mdbx::map_handle testHandle = */ txn.create_map("xyz", mdbx::key_mode::usual, mdbx::value_mode::single);
|
|
||||||
txn.commit();
|
|
||||||
|
|
||||||
//-------------------------------------
|
|
||||||
txn = env.start_write();
|
|
||||||
MDBX_txn *c_txn = txn;
|
|
||||||
err = mdbx_txn_reset(txn);
|
|
||||||
assert(err == MDBX_EINVAL);
|
|
||||||
ok = ok && err == MDBX_EINVAL;
|
|
||||||
|
|
||||||
err = mdbx_txn_break(txn);
|
|
||||||
assert(err == MDBX_SUCCESS);
|
|
||||||
ok = ok && err == MDBX_SUCCESS;
|
|
||||||
|
|
||||||
err = mdbx_txn_commit(txn);
|
|
||||||
assert(err == MDBX_RESULT_TRUE);
|
|
||||||
ok = ok && err == MDBX_RESULT_TRUE;
|
|
||||||
|
|
||||||
//-------------------------------------
|
|
||||||
err = mdbx_txn_begin(env, nullptr, MDBX_TXN_READWRITE, &c_txn);
|
|
||||||
assert(err == MDBX_SUCCESS);
|
|
||||||
ok = ok && err == MDBX_SUCCESS;
|
|
||||||
assert(c_txn == (const MDBX_txn *)txn);
|
|
||||||
|
|
||||||
err = mdbx_txn_break(txn);
|
|
||||||
assert(err == MDBX_SUCCESS);
|
|
||||||
ok = ok && err == MDBX_SUCCESS;
|
|
||||||
|
|
||||||
err = mdbx_txn_reset(txn);
|
|
||||||
assert(err == MDBX_EINVAL);
|
|
||||||
ok = ok && err == MDBX_EINVAL;
|
|
||||||
|
|
||||||
err = mdbx_txn_commit(txn);
|
|
||||||
assert(err == MDBX_RESULT_TRUE);
|
|
||||||
ok = ok && err == MDBX_RESULT_TRUE;
|
|
||||||
|
|
||||||
err = mdbx_txn_abort(c_txn);
|
|
||||||
assert(err == MDBX_BAD_TXN);
|
|
||||||
ok = ok && err == MDBX_BAD_TXN;
|
|
||||||
//-------------------------------------
|
|
||||||
err = mdbx_txn_begin(env, nullptr, MDBX_TXN_READWRITE, &c_txn);
|
|
||||||
assert(err == MDBX_SUCCESS);
|
|
||||||
ok = ok && err == MDBX_SUCCESS;
|
|
||||||
assert(c_txn == (const MDBX_txn *)txn);
|
|
||||||
txn.commit();
|
|
||||||
|
|
||||||
err = mdbx_txn_reset(c_txn);
|
|
||||||
assert(err == MDBX_BAD_TXN);
|
|
||||||
ok = ok && err == MDBX_BAD_TXN;
|
|
||||||
err = mdbx_txn_break(c_txn);
|
|
||||||
assert(err == MDBX_BAD_TXN);
|
|
||||||
ok = ok && err == MDBX_BAD_TXN;
|
|
||||||
err = mdbx_txn_abort(c_txn);
|
|
||||||
assert(err == MDBX_BAD_TXN);
|
|
||||||
ok = ok && err == MDBX_BAD_TXN;
|
|
||||||
|
|
||||||
//=====================================
|
|
||||||
|
|
||||||
txn = env.start_read();
|
|
||||||
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
|
||||||
assert(err == MDBX_BAD_TXN);
|
|
||||||
ok = ok && err == MDBX_BAD_TXN;
|
|
||||||
txn.make_broken();
|
|
||||||
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
|
||||||
assert(err == MDBX_BAD_TXN);
|
|
||||||
ok = ok && err == MDBX_BAD_TXN;
|
|
||||||
txn.reset_reading();
|
|
||||||
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
|
||||||
assert(err == MDBX_BAD_TXN);
|
|
||||||
ok = ok && err == MDBX_BAD_TXN;
|
|
||||||
txn.abort();
|
|
||||||
|
|
||||||
//-------------------------------------
|
|
||||||
|
|
||||||
txn = env.start_read();
|
|
||||||
txn.reset_reading();
|
|
||||||
txn.make_broken();
|
|
||||||
txn.abort();
|
|
||||||
|
|
||||||
//=====================================
|
|
||||||
|
|
||||||
std::latch s(1);
|
|
||||||
txn = env.start_read();
|
|
||||||
c_txn = txn;
|
|
||||||
|
|
||||||
std::thread t([&]() {
|
|
||||||
s.wait();
|
|
||||||
#if MDBX_TXN_CHECKOWNER
|
|
||||||
err = mdbx_txn_reset(c_txn);
|
|
||||||
assert(err == MDBX_THREAD_MISMATCH);
|
|
||||||
ok = ok && err == MDBX_THREAD_MISMATCH;
|
|
||||||
err = mdbx_txn_break(c_txn);
|
|
||||||
assert(err == MDBX_THREAD_MISMATCH);
|
|
||||||
ok = ok && err == MDBX_THREAD_MISMATCH;
|
|
||||||
err = mdbx_txn_commit(c_txn);
|
|
||||||
assert(err == MDBX_THREAD_MISMATCH);
|
|
||||||
ok = ok && err == MDBX_THREAD_MISMATCH;
|
|
||||||
err = mdbx_txn_abort(c_txn);
|
|
||||||
assert(err == MDBX_THREAD_MISMATCH);
|
|
||||||
ok = ok && err == MDBX_THREAD_MISMATCH;
|
|
||||||
#endif /* MDBX_TXN_CHECKOWNER */
|
|
||||||
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
|
||||||
assert(err == MDBX_BAD_TXN);
|
|
||||||
ok = ok && err == MDBX_BAD_TXN;
|
|
||||||
});
|
|
||||||
|
|
||||||
s.count_down();
|
|
||||||
t.join();
|
|
||||||
}
|
|
||||||
|
|
||||||
//=====================================
|
|
||||||
//=====================================
|
|
||||||
|
|
||||||
{
|
|
||||||
operateParameters.options.no_sticky_threads = true;
|
|
||||||
operateParameters.options.nested_write_transactions = true;
|
|
||||||
mdbx::env_managed env(path, operateParameters);
|
|
||||||
|
|
||||||
//-------------------------------------
|
|
||||||
auto txn = env.start_write();
|
|
||||||
MDBX_txn *c_txn = txn;
|
|
||||||
err = mdbx_txn_reset(txn);
|
|
||||||
assert(err == MDBX_EINVAL);
|
|
||||||
ok = ok && err == MDBX_EINVAL;
|
|
||||||
|
|
||||||
err = mdbx_txn_break(txn);
|
|
||||||
assert(err == MDBX_SUCCESS);
|
|
||||||
ok = ok && err == MDBX_SUCCESS;
|
|
||||||
|
|
||||||
err = mdbx_txn_commit(txn);
|
|
||||||
assert(err == MDBX_RESULT_TRUE);
|
|
||||||
ok = ok && err == MDBX_RESULT_TRUE;
|
|
||||||
|
|
||||||
//-------------------------------------
|
|
||||||
err = mdbx_txn_begin(env, nullptr, MDBX_TXN_READWRITE, &c_txn);
|
|
||||||
assert(err == MDBX_SUCCESS);
|
|
||||||
ok = ok && err == MDBX_SUCCESS;
|
|
||||||
assert(c_txn == (const MDBX_txn *)txn);
|
|
||||||
|
|
||||||
err = mdbx_txn_break(txn);
|
|
||||||
assert(err == MDBX_SUCCESS);
|
|
||||||
ok = ok && err == MDBX_SUCCESS;
|
|
||||||
|
|
||||||
err = mdbx_txn_reset(txn);
|
|
||||||
assert(err == MDBX_EINVAL);
|
|
||||||
ok = ok && err == MDBX_EINVAL;
|
|
||||||
|
|
||||||
err = mdbx_txn_commit(txn);
|
|
||||||
assert(err == MDBX_RESULT_TRUE);
|
|
||||||
ok = ok && err == MDBX_RESULT_TRUE;
|
|
||||||
|
|
||||||
err = mdbx_txn_abort(c_txn);
|
|
||||||
assert(err == MDBX_BAD_TXN);
|
|
||||||
ok = ok && err == MDBX_BAD_TXN;
|
|
||||||
//-------------------------------------
|
|
||||||
err = mdbx_txn_begin(env, nullptr, MDBX_TXN_READWRITE, &c_txn);
|
|
||||||
assert(err == MDBX_SUCCESS);
|
|
||||||
ok = ok && err == MDBX_SUCCESS;
|
|
||||||
assert(c_txn == (const MDBX_txn *)txn);
|
|
||||||
txn.commit();
|
|
||||||
|
|
||||||
err = mdbx_txn_reset(c_txn);
|
|
||||||
assert(err == MDBX_BAD_TXN);
|
|
||||||
ok = ok && err == MDBX_BAD_TXN;
|
|
||||||
err = mdbx_txn_break(c_txn);
|
|
||||||
assert(err == MDBX_BAD_TXN);
|
|
||||||
ok = ok && err == MDBX_BAD_TXN;
|
|
||||||
err = mdbx_txn_abort(c_txn);
|
|
||||||
assert(err == MDBX_BAD_TXN);
|
|
||||||
ok = ok && err == MDBX_BAD_TXN;
|
|
||||||
|
|
||||||
//=====================================
|
|
||||||
|
|
||||||
txn = env.start_read();
|
|
||||||
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
|
||||||
assert(err == MDBX_BAD_TXN);
|
|
||||||
ok = ok && err == MDBX_BAD_TXN;
|
|
||||||
txn.make_broken();
|
|
||||||
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
|
||||||
assert(err == MDBX_BAD_TXN);
|
|
||||||
ok = ok && err == MDBX_BAD_TXN;
|
|
||||||
txn.reset_reading();
|
|
||||||
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
|
||||||
assert(err == MDBX_BAD_TXN);
|
|
||||||
ok = ok && err == MDBX_BAD_TXN;
|
|
||||||
txn.abort();
|
|
||||||
|
|
||||||
//-------------------------------------
|
|
||||||
|
|
||||||
txn = env.start_read();
|
|
||||||
txn.reset_reading();
|
|
||||||
txn.make_broken();
|
|
||||||
txn.abort();
|
|
||||||
|
|
||||||
//=====================================
|
|
||||||
|
|
||||||
std::latch s1(1), s2(1), s3(1);
|
|
||||||
txn = env.start_read();
|
|
||||||
c_txn = txn;
|
|
||||||
|
|
||||||
std::thread t([&]() {
|
|
||||||
s1.wait();
|
|
||||||
err = mdbx_txn_break(c_txn);
|
|
||||||
assert(err == MDBX_SUCCESS);
|
|
||||||
ok = ok && err == MDBX_SUCCESS;
|
|
||||||
err = mdbx_txn_reset(c_txn);
|
|
||||||
assert(err == MDBX_SUCCESS);
|
|
||||||
ok = ok && err == MDBX_SUCCESS;
|
|
||||||
txn.renew_reading();
|
|
||||||
s2.count_down();
|
|
||||||
|
|
||||||
s3.wait();
|
|
||||||
err = mdbx_txn_begin(env, txn, MDBX_TXN_READWRITE, &c_txn);
|
|
||||||
assert(err == MDBX_SUCCESS);
|
|
||||||
ok = ok && err == MDBX_SUCCESS;
|
|
||||||
err = mdbx_txn_commit(c_txn);
|
|
||||||
assert(err == MDBX_SUCCESS);
|
|
||||||
ok = ok && err == MDBX_SUCCESS;
|
|
||||||
c_txn = txn;
|
|
||||||
err = mdbx_txn_commit(c_txn);
|
|
||||||
assert(err == MDBX_THREAD_MISMATCH);
|
|
||||||
ok = ok && err == MDBX_THREAD_MISMATCH;
|
|
||||||
err = mdbx_txn_abort(c_txn);
|
|
||||||
assert(err == MDBX_THREAD_MISMATCH);
|
|
||||||
ok = ok && err == MDBX_THREAD_MISMATCH;
|
|
||||||
err = mdbx_txn_break(c_txn);
|
|
||||||
assert(err == MDBX_SUCCESS);
|
|
||||||
ok = ok && err == MDBX_SUCCESS;
|
|
||||||
err = mdbx_txn_reset(c_txn);
|
|
||||||
assert(err == MDBX_EINVAL);
|
|
||||||
ok = ok && err == MDBX_EINVAL;
|
|
||||||
});
|
|
||||||
|
|
||||||
s1.count_down();
|
|
||||||
s2.wait();
|
|
||||||
txn.commit();
|
|
||||||
txn = env.start_write();
|
|
||||||
s3.count_down();
|
|
||||||
|
|
||||||
t.join();
|
|
||||||
txn.abort();
|
|
||||||
}
|
|
||||||
|
|
||||||
std::cout << (ok ? "OK\n" : "FAIL\n");
|
std::cout << (ok ? "OK\n" : "FAIL\n");
|
||||||
return ok ? EXIT_SUCCESS : EXIT_FAILURE;
|
return ok ? EXIT_SUCCESS : EXIT_FAILURE;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user