mdbx: Merge branch 'devel'.

This commit is contained in:
Leo Yuriev 2016-05-19 22:13:55 +03:00
commit 701d3767f2
2 changed files with 30 additions and 12 deletions

38
mdb.c
View File

@ -1505,6 +1505,7 @@ mdb_page_malloc(MDB_txn *txn, unsigned num)
size_t size = env->me_psize; size_t size = env->me_psize;
MDB_page *np = env->me_dpages; MDB_page *np = env->me_dpages;
if (likely(num == 1 && np)) { if (likely(num == 1 && np)) {
ASAN_UNPOISON_MEMORY_REGION(np, size);
VALGRIND_MEMPOOL_ALLOC(env, np, size); VALGRIND_MEMPOOL_ALLOC(env, np, size);
VALGRIND_MAKE_MEM_DEFINED(&np->mp_next, sizeof(np->mp_next)); VALGRIND_MAKE_MEM_DEFINED(&np->mp_next, sizeof(np->mp_next));
env->me_dpages = np->mp_next; env->me_dpages = np->mp_next;
@ -1583,6 +1584,7 @@ mdb_kill_page(MDB_env *env, pgno_t pgno)
MDB_page *mp = (MDB_page *)(env->me_map + offs); MDB_page *mp = (MDB_page *)(env->me_map + offs);
memset(&mp->mp_pb, 0x6F /* 'o', 111 */, env->me_psize - shift); memset(&mp->mp_pb, 0x6F /* 'o', 111 */, env->me_psize - shift);
VALGRIND_MAKE_MEM_NOACCESS(&mp->mp_pb, env->me_psize - shift); VALGRIND_MAKE_MEM_NOACCESS(&mp->mp_pb, env->me_psize - shift);
ASAN_POISON_MEMORY_REGION(&mp->mp_pb, env->me_psize - shift);
} else { } else {
struct iovec iov[1]; struct iovec iov[1];
iov[0].iov_len = env->me_psize - shift; iov[0].iov_len = env->me_psize - shift;
@ -1636,9 +1638,13 @@ mdb_page_loose(MDB_cursor *mc, MDB_page *mp)
} }
if (loose) { if (loose) {
mdb_debug("loosen db %d page %zu", DDBI(mc), mp->mp_pgno); mdb_debug("loosen db %d page %zu", DDBI(mc), mp->mp_pgno);
if (unlikely(txn->mt_env->me_flags & MDBX_PAGEPERTURB)) MDB_page **link = &NEXT_LOOSE_PAGE(mp);
if (unlikely(txn->mt_env->me_flags & MDBX_PAGEPERTURB)) {
mdb_kill_page(txn->mt_env, pgno); mdb_kill_page(txn->mt_env, pgno);
NEXT_LOOSE_PAGE(mp) = txn->mt_loose_pgs; VALGRIND_MAKE_MEM_UNDEFINED(link, sizeof(MDB_page*));
ASAN_UNPOISON_MEMORY_REGION(link, sizeof(MDB_page*));
}
*link = txn->mt_loose_pgs;
txn->mt_loose_pgs = mp; txn->mt_loose_pgs = mp;
txn->mt_loose_count++; txn->mt_loose_count++;
mp->mp_flags |= P_LOOSE; mp->mp_flags |= P_LOOSE;
@ -2110,6 +2116,7 @@ mdb_page_alloc(MDB_cursor *mc, int num, MDB_page **mp, int flags)
txn->mt_loose_pgs = NEXT_LOOSE_PAGE(np); txn->mt_loose_pgs = NEXT_LOOSE_PAGE(np);
txn->mt_loose_count--; txn->mt_loose_count--;
mdb_debug("db %d use loose page %zu", DDBI(mc), np->mp_pgno); mdb_debug("db %d use loose page %zu", DDBI(mc), np->mp_pgno);
ASAN_UNPOISON_MEMORY_REGION(np, env->me_psize);
*mp = np; *mp = np;
return MDB_SUCCESS; return MDB_SUCCESS;
} }
@ -2361,6 +2368,7 @@ done:
np = (MDB_page *)(env->me_map + env->me_psize * pgno); np = (MDB_page *)(env->me_map + env->me_psize * pgno);
/* LY: reset no-access flag from mdb_kill_page() */ /* LY: reset no-access flag from mdb_kill_page() */
VALGRIND_MAKE_MEM_UNDEFINED(np, env->me_psize * num); VALGRIND_MAKE_MEM_UNDEFINED(np, env->me_psize * num);
ASAN_UNPOISON_MEMORY_REGION(np, env->me_psize * num);
} else { } else {
if (unlikely(!(np = mdb_page_malloc(txn, num)))) { if (unlikely(!(np = mdb_page_malloc(txn, num)))) {
rc = ENOMEM; rc = ENOMEM;
@ -5110,6 +5118,7 @@ mdbx_env_close_ex(MDB_env *env, int dont_sync)
VALGRIND_DESTROY_MEMPOOL(env); VALGRIND_DESTROY_MEMPOOL(env);
while ((dp = env->me_dpages) != NULL) { while ((dp = env->me_dpages) != NULL) {
ASAN_UNPOISON_MEMORY_REGION(&dp->mp_next, sizeof(dp->mp_next));
VALGRIND_MAKE_MEM_DEFINED(&dp->mp_next, sizeof(dp->mp_next)); VALGRIND_MAKE_MEM_DEFINED(&dp->mp_next, sizeof(dp->mp_next));
env->me_dpages = dp->mp_next; env->me_dpages = dp->mp_next;
free(dp); free(dp);
@ -6885,7 +6894,6 @@ current:
*/ */
if (unlikely(level > 1)) { if (unlikely(level > 1)) {
/* It is writable only in a parent txn */ /* It is writable only in a parent txn */
size_t sz = (size_t) env->me_psize * ovpages, off;
MDB_page *np = mdb_page_malloc(mc->mc_txn, ovpages); MDB_page *np = mdb_page_malloc(mc->mc_txn, ovpages);
MDB_ID2 id2; MDB_ID2 id2;
if (unlikely(!np)) if (unlikely(!np))
@ -6895,9 +6903,19 @@ current:
/* Note - this page is already counted in parent's dirty_room */ /* Note - this page is already counted in parent's dirty_room */
rc2 = mdb_mid2l_insert(mc->mc_txn->mt_u.dirty_list, &id2); rc2 = mdb_mid2l_insert(mc->mc_txn->mt_u.dirty_list, &id2);
mdb_cassert(mc, rc2 == 0); mdb_cassert(mc, rc2 == 0);
if (1 || /* LY: Hm, why we should do this differently in dependence from MDB_RESERVE? */ /* Currently we make the page look as with put() in the
!(flags & MDB_RESERVE)) { * parent txn, in case the user peeks at MDB_RESERVEd
/* Copy end of page, adjusting alignment so * or unused parts. Some users treat ovpages specially.
*/
#if MDBX_MODE_ENABLED
/* LY: New page will contain only header from origin,
* but no any payload */
memcpy(np, omp, PAGEHDRSZ);
#else
size_t sz = (size_t) env->me_psize * ovpages, off;
if (!(flags & MDB_RESERVE)) {
/* Skip the part where LMDB will put *data.
* Copy end of page, adjusting alignment so
* compiler may copy words instead of bytes. * compiler may copy words instead of bytes.
*/ */
off = (PAGEHDRSZ + data->mv_size) & -sizeof(size_t); off = (PAGEHDRSZ + data->mv_size) & -sizeof(size_t);
@ -6905,7 +6923,8 @@ current:
(size_t *)((char *)omp + off), sz - off); (size_t *)((char *)omp + off), sz - off);
sz = PAGEHDRSZ; sz = PAGEHDRSZ;
} }
memcpy(np, omp, sz); /* Copy beginning of page */ memcpy(np, omp, sz); /* Copy whole or header of page */
#endif /* MDBX_MODE_ENABLED */
omp = np; omp = np;
} }
SETDSZ(leaf, data->mv_size); SETDSZ(leaf, data->mv_size);
@ -9177,7 +9196,7 @@ mdb_env_cthr_toggle(mdb_copy *my, int st)
static int __cold static int __cold
mdb_env_cwalk(mdb_copy *my, pgno_t *pg, int flags) mdb_env_cwalk(mdb_copy *my, pgno_t *pg, int flags)
{ {
MDB_cursor mc; MDB_cursor mc = {0};
MDB_txn *txn = my->mc_txn; MDB_txn *txn = my->mc_txn;
MDB_node *ni; MDB_node *ni;
MDB_page *mo, *mp, *leaf; MDB_page *mo, *mp, *leaf;
@ -9190,10 +9209,9 @@ mdb_env_cwalk(mdb_copy *my, pgno_t *pg, int flags)
return MDB_SUCCESS; return MDB_SUCCESS;
mc.mc_snum = 1; mc.mc_snum = 1;
mc.mc_top = 0;
mc.mc_txn = txn; mc.mc_txn = txn;
rc = mdb_page_get(my->mc_txn, *pg, &mc.mc_pg[0], NULL); rc = mdb_page_get(txn, *pg, &mc.mc_pg[0], NULL);
if (rc) if (rc)
return rc; return rc;
rc = mdb_page_search_root(&mc, NULL, MDB_PS_FIRST); rc = mdb_page_search_root(&mc, NULL, MDB_PS_FIRST);

4
mdbx.h
View File

@ -31,7 +31,7 @@
When needed drop-in replacement for liblmdb you should: When needed drop-in replacement for liblmdb you should:
- 'make lmdb' to build liblmdb.so and liblmdb.a; - 'make lmdb' to build liblmdb.so and liblmdb.a;
- #include <lmdb.h> and use mdb_* functions; - #include <lmdb.h> and use mdb_* functions;
- linking with liblmdb.so оr liblmdb.a; - linking with liblmdb.so or liblmdb.a;
= This provides nearly full compatibility with = This provides nearly full compatibility with
original LMDB from Symas Corp. original LMDB from Symas Corp.
@ -41,7 +41,7 @@
When exactly the libmdbx is needed, you should: When exactly the libmdbx is needed, you should:
- 'make mdbx' to build libmdbx.so and libmdbx.a; - 'make mdbx' to build libmdbx.so and libmdbx.a;
- #include <mdbx.h> and use mdbx_* functions; - #include <mdbx.h> and use mdbx_* functions;
- linking with libmdbx.so оr libmdbx.a; - linking with libmdbx.so or libmdbx.a;
= This allows using (linking) both MDBX and LMDB = This allows using (linking) both MDBX and LMDB
simultaneously in the one application, for instance simultaneously in the one application, for instance