mirror of
https://github.com/isar/libmdbx.git
synced 2025-01-04 18:04:13 +08:00
mdbx-test: add workaround for CLANG/LLVM STL stupidity of std::set<>
.
This commit is contained in:
parent
cf5f31c577
commit
c4a696be1d
@ -110,16 +110,24 @@ bool testcase_nested::teardown() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void testcase_nested::push_txn() {
|
void testcase_nested::push_txn() {
|
||||||
MDBX_txn *txn;
|
MDBX_txn *nested_txn;
|
||||||
MDBX_txn_flags_t flags = MDBX_txn_flags_t(
|
MDBX_txn_flags_t flags = MDBX_txn_flags_t(
|
||||||
prng32() & uint32_t(MDBX_TXN_NOSYNC | MDBX_TXN_NOMETASYNC));
|
prng32() & uint32_t(MDBX_TXN_NOSYNC | MDBX_TXN_NOMETASYNC));
|
||||||
int err = mdbx_txn_begin(db_guard.get(), txn_guard.get(), flags, &txn);
|
int err = mdbx_txn_begin(db_guard.get(), txn_guard.get(), flags, &nested_txn);
|
||||||
if (unlikely(err != MDBX_SUCCESS))
|
if (unlikely(err != MDBX_SUCCESS))
|
||||||
failure_perror("mdbx_txn_begin(nested)", err);
|
failure_perror("mdbx_txn_begin(nested)", err);
|
||||||
stack.emplace(scoped_txn_guard(txn), serial, fifo, speculum);
|
/* CLANG/LLVM C++ library could stupidly copy std::set<> item-by-item,
|
||||||
std::swap(txn_guard, std::get<0>(stack.top()));
|
* i.e. with insertion(s) & comparison(s), which will cause null dereference
|
||||||
|
* during call mdbx_cmp() with zero txn. So it is the workaround for this:
|
||||||
|
* - explicitly set txn_guard with the new nested txn;
|
||||||
|
* - explicitly copy the `speculum` (an instance of std::set<>). */
|
||||||
|
scoped_txn_guard nested_txn_guard(nested_txn);
|
||||||
|
txn_guard.swap(nested_txn_guard);
|
||||||
|
SET speculum_snapshot(speculum);
|
||||||
|
stack.emplace(std::move(nested_txn_guard), serial, fifo,
|
||||||
|
std::move(speculum_snapshot));
|
||||||
log_verbose("begin level#%zu txn #%" PRIu64 ", flags 0x%x, serial %" PRIu64,
|
log_verbose("begin level#%zu txn #%" PRIu64 ", flags 0x%x, serial %" PRIu64,
|
||||||
stack.size(), mdbx_txn_id(txn), flags, serial);
|
stack.size(), mdbx_txn_id(nested_txn), flags, serial);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool testcase_nested::pop_txn(bool abort) {
|
bool testcase_nested::pop_txn(bool abort) {
|
||||||
|
16
test/test.cc
16
test/test.cc
@ -197,6 +197,18 @@ int testcase::breakable_commit() {
|
|||||||
log_trace(">> txn_commit");
|
log_trace(">> txn_commit");
|
||||||
assert(txn_guard);
|
assert(txn_guard);
|
||||||
|
|
||||||
|
/* CLANG/LLVM C++ library could stupidly copy std::set<> item-by-item,
|
||||||
|
* i.e. with insertion(s) & comparison(s), which will cause null dereference
|
||||||
|
* during call mdbx_cmp() with zero txn. So it is the workaround for this:
|
||||||
|
* - explicitly make copies of the `speculums`;
|
||||||
|
* - explicitly move relevant copy after transaction commit. */
|
||||||
|
SET speculum_committed_copy(ItemCompare(this)),
|
||||||
|
speculum_copy(ItemCompare(this));
|
||||||
|
if (need_speculum_assign) {
|
||||||
|
speculum_committed_copy = speculum_committed;
|
||||||
|
speculum_copy = speculum;
|
||||||
|
}
|
||||||
|
|
||||||
MDBX_txn *txn = txn_guard.release();
|
MDBX_txn *txn = txn_guard.release();
|
||||||
txn_inject_writefault(txn);
|
txn_inject_writefault(txn);
|
||||||
int rc = mdbx_txn_commit(txn);
|
int rc = mdbx_txn_commit(txn);
|
||||||
@ -207,9 +219,9 @@ int testcase::breakable_commit() {
|
|||||||
if (need_speculum_assign) {
|
if (need_speculum_assign) {
|
||||||
need_speculum_assign = false;
|
need_speculum_assign = false;
|
||||||
if (unlikely(rc != MDBX_SUCCESS))
|
if (unlikely(rc != MDBX_SUCCESS))
|
||||||
speculum = speculum_committed;
|
speculum = std::move(speculum_committed_copy);
|
||||||
else
|
else
|
||||||
speculum_committed = speculum;
|
speculum_committed = std::move(speculum_copy);
|
||||||
}
|
}
|
||||||
|
|
||||||
log_trace("<< txn_commit: %s", rc ? "failed" : "Ok");
|
log_trace("<< txn_commit: %s", rc ? "failed" : "Ok");
|
||||||
|
Loading…
x
Reference in New Issue
Block a user