mirror of
https://github.com/isar/libmdbx.git
synced 2025-01-04 18:24:12 +08:00
mdbx: add MDBX_ASAN_(UN)POISON_MEMORY_REGION()
macros.
This commit is contained in:
parent
7da64b725d
commit
6034985686
38
src/core.c
38
src/core.c
@ -4128,7 +4128,7 @@ static MDBX_page *mdbx_page_malloc(MDBX_txn *txn, unsigned num) {
|
|||||||
size_t size = env->me_psize;
|
size_t size = env->me_psize;
|
||||||
if (likely(num == 1 && np)) {
|
if (likely(num == 1 && np)) {
|
||||||
mdbx_assert(env, env->me_dp_reserve_len > 0);
|
mdbx_assert(env, env->me_dp_reserve_len > 0);
|
||||||
ASAN_UNPOISON_MEMORY_REGION(np, size);
|
MDBX_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_dp_reserve = np->mp_next;
|
env->me_dp_reserve = np->mp_next;
|
||||||
@ -4164,13 +4164,14 @@ static MDBX_page *mdbx_page_malloc(MDBX_txn *txn, unsigned num) {
|
|||||||
/* Free a shadow dirty page */
|
/* Free a shadow dirty page */
|
||||||
static void mdbx_dpage_free(MDBX_env *env, MDBX_page *dp, unsigned npages) {
|
static void mdbx_dpage_free(MDBX_env *env, MDBX_page *dp, unsigned npages) {
|
||||||
VALGRIND_MAKE_MEM_UNDEFINED(dp, pgno2bytes(env, npages));
|
VALGRIND_MAKE_MEM_UNDEFINED(dp, pgno2bytes(env, npages));
|
||||||
ASAN_UNPOISON_MEMORY_REGION(dp, pgno2bytes(env, npages));
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(dp, pgno2bytes(env, npages));
|
||||||
if (MDBX_DEBUG != 0 || unlikely(env->me_flags & MDBX_PAGEPERTURB))
|
if (MDBX_DEBUG != 0 || unlikely(env->me_flags & MDBX_PAGEPERTURB))
|
||||||
memset(dp, -1, pgno2bytes(env, npages));
|
memset(dp, -1, pgno2bytes(env, npages));
|
||||||
if (npages == 1 &&
|
if (npages == 1 &&
|
||||||
env->me_dp_reserve_len < env->me_options.dp_reserve_limit) {
|
env->me_dp_reserve_len < env->me_options.dp_reserve_limit) {
|
||||||
ASAN_POISON_MEMORY_REGION((char *)dp + sizeof(dp->mp_next),
|
MDBX_ASAN_POISON_MEMORY_REGION((char *)dp + sizeof(dp->mp_next),
|
||||||
pgno2bytes(env, npages) - sizeof(dp->mp_next));
|
pgno2bytes(env, npages) -
|
||||||
|
sizeof(dp->mp_next));
|
||||||
dp->mp_next = env->me_dp_reserve;
|
dp->mp_next = env->me_dp_reserve;
|
||||||
VALGRIND_MEMPOOL_FREE(env, dp);
|
VALGRIND_MEMPOOL_FREE(env, dp);
|
||||||
env->me_dp_reserve = dp;
|
env->me_dp_reserve = dp;
|
||||||
@ -4526,7 +4527,7 @@ static __inline void mdbx_page_wash(MDBX_txn *txn, const unsigned di,
|
|||||||
if (txn->mt_flags & MDBX_WRITEMAP) {
|
if (txn->mt_flags & MDBX_WRITEMAP) {
|
||||||
VALGRIND_MAKE_MEM_NOACCESS(page_data(mp),
|
VALGRIND_MAKE_MEM_NOACCESS(page_data(mp),
|
||||||
pgno2bytes(txn->mt_env, npages) - PAGEHDRSZ);
|
pgno2bytes(txn->mt_env, npages) - PAGEHDRSZ);
|
||||||
ASAN_POISON_MEMORY_REGION(page_data(mp),
|
MDBX_ASAN_POISON_MEMORY_REGION(page_data(mp),
|
||||||
pgno2bytes(txn->mt_env, npages) - PAGEHDRSZ);
|
pgno2bytes(txn->mt_env, npages) - PAGEHDRSZ);
|
||||||
} else
|
} else
|
||||||
mdbx_dpage_free(txn->mt_env, mp, npages);
|
mdbx_dpage_free(txn->mt_env, mp, npages);
|
||||||
@ -4756,7 +4757,7 @@ status_done:
|
|||||||
memset(page_data(mp), -1, txn->mt_env->me_psize - PAGEHDRSZ);
|
memset(page_data(mp), -1, txn->mt_env->me_psize - PAGEHDRSZ);
|
||||||
VALGRIND_MAKE_MEM_NOACCESS(page_data(mp),
|
VALGRIND_MAKE_MEM_NOACCESS(page_data(mp),
|
||||||
txn->mt_env->me_psize - PAGEHDRSZ);
|
txn->mt_env->me_psize - PAGEHDRSZ);
|
||||||
ASAN_POISON_MEMORY_REGION(page_data(mp),
|
MDBX_ASAN_POISON_MEMORY_REGION(page_data(mp),
|
||||||
txn->mt_env->me_psize - PAGEHDRSZ);
|
txn->mt_env->me_psize - PAGEHDRSZ);
|
||||||
return MDBX_SUCCESS;
|
return MDBX_SUCCESS;
|
||||||
}
|
}
|
||||||
@ -4786,8 +4787,9 @@ status_done:
|
|||||||
if (!(txn->mt_flags & MDBX_WRITEMAP)) {
|
if (!(txn->mt_flags & MDBX_WRITEMAP)) {
|
||||||
VALGRIND_MAKE_MEM_NOACCESS(page_data(pgno2page(txn->mt_env, pgno)),
|
VALGRIND_MAKE_MEM_NOACCESS(page_data(pgno2page(txn->mt_env, pgno)),
|
||||||
pgno2bytes(txn->mt_env, npages) - PAGEHDRSZ);
|
pgno2bytes(txn->mt_env, npages) - PAGEHDRSZ);
|
||||||
ASAN_POISON_MEMORY_REGION(page_data(pgno2page(txn->mt_env, pgno)),
|
MDBX_ASAN_POISON_MEMORY_REGION(page_data(pgno2page(txn->mt_env, pgno)),
|
||||||
pgno2bytes(txn->mt_env, npages) - PAGEHDRSZ);
|
pgno2bytes(txn->mt_env, npages) -
|
||||||
|
PAGEHDRSZ);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
skip_invalidate:
|
skip_invalidate:
|
||||||
@ -4910,7 +4912,7 @@ static int mdbx_iov_write(MDBX_txn *const txn, struct mdbx_iov_ctx *ctx) {
|
|||||||
else {
|
else {
|
||||||
VALGRIND_MAKE_MEM_DEFINED(txn->mt_env->me_map + ctx->iov_off,
|
VALGRIND_MAKE_MEM_DEFINED(txn->mt_env->me_map + ctx->iov_off,
|
||||||
ctx->iov_bytes);
|
ctx->iov_bytes);
|
||||||
ASAN_UNPOISON_MEMORY_REGION(txn->mt_env->me_map + ctx->iov_off,
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(txn->mt_env->me_map + ctx->iov_off,
|
||||||
ctx->iov_bytes);
|
ctx->iov_bytes);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -6257,7 +6259,8 @@ __hot static struct page_result mdbx_page_alloc(MDBX_cursor *mc,
|
|||||||
mdbx_tassert(txn, ret.page->mp_pgno < txn->mt_next_pgno);
|
mdbx_tassert(txn, ret.page->mp_pgno < txn->mt_next_pgno);
|
||||||
mdbx_ensure(env, ret.page->mp_pgno >= NUM_METAS);
|
mdbx_ensure(env, ret.page->mp_pgno >= NUM_METAS);
|
||||||
VALGRIND_MAKE_MEM_UNDEFINED(page_data(ret.page), page_space(txn->mt_env));
|
VALGRIND_MAKE_MEM_UNDEFINED(page_data(ret.page), page_space(txn->mt_env));
|
||||||
ASAN_UNPOISON_MEMORY_REGION(page_data(ret.page), page_space(txn->mt_env));
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(page_data(ret.page),
|
||||||
|
page_space(txn->mt_env));
|
||||||
ret.page->mp_txnid = txn->mt_front;
|
ret.page->mp_txnid = txn->mt_front;
|
||||||
ret.err = MDBX_SUCCESS;
|
ret.err = MDBX_SUCCESS;
|
||||||
return ret;
|
return ret;
|
||||||
@ -6651,7 +6654,7 @@ done:
|
|||||||
ret.page = pgno2page(env, pgno);
|
ret.page = pgno2page(env, pgno);
|
||||||
/* LY: reset no-access flag from mdbx_page_loose() */
|
/* LY: reset no-access flag from mdbx_page_loose() */
|
||||||
VALGRIND_MAKE_MEM_UNDEFINED(ret.page, pgno2bytes(env, num));
|
VALGRIND_MAKE_MEM_UNDEFINED(ret.page, pgno2bytes(env, num));
|
||||||
ASAN_UNPOISON_MEMORY_REGION(ret.page, pgno2bytes(env, num));
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(ret.page, pgno2bytes(env, num));
|
||||||
} else {
|
} else {
|
||||||
if (unlikely(!(ret.page = mdbx_page_malloc(txn, num)))) {
|
if (unlikely(!(ret.page = mdbx_page_malloc(txn, num)))) {
|
||||||
ret.err = MDBX_ENOMEM;
|
ret.err = MDBX_ENOMEM;
|
||||||
@ -7201,7 +7204,7 @@ static void mdbx_txn_valgrind(MDBX_env *env, MDBX_txn *txn) {
|
|||||||
if (env->me_poison_edge < txn->mt_next_pgno)
|
if (env->me_poison_edge < txn->mt_next_pgno)
|
||||||
env->me_poison_edge = txn->mt_next_pgno;
|
env->me_poison_edge = txn->mt_next_pgno;
|
||||||
VALGRIND_MAKE_MEM_DEFINED(env->me_map, pgno2bytes(env, txn->mt_next_pgno));
|
VALGRIND_MAKE_MEM_DEFINED(env->me_map, pgno2bytes(env, txn->mt_next_pgno));
|
||||||
ASAN_UNPOISON_MEMORY_REGION(env->me_map,
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(env->me_map,
|
||||||
pgno2bytes(env, txn->mt_next_pgno));
|
pgno2bytes(env, txn->mt_next_pgno));
|
||||||
/* don't touch more, it should be already poisoned */
|
/* don't touch more, it should be already poisoned */
|
||||||
} else { /* transaction end */
|
} else { /* transaction end */
|
||||||
@ -7230,7 +7233,7 @@ static void mdbx_txn_valgrind(MDBX_env *env, MDBX_txn *txn) {
|
|||||||
env->me_poison_edge = last;
|
env->me_poison_edge = last;
|
||||||
VALGRIND_MAKE_MEM_NOACCESS(env->me_map + pgno2bytes(env, last),
|
VALGRIND_MAKE_MEM_NOACCESS(env->me_map + pgno2bytes(env, last),
|
||||||
pgno2bytes(env, edge - last));
|
pgno2bytes(env, edge - last));
|
||||||
ASAN_POISON_MEMORY_REGION(env->me_map + pgno2bytes(env, last),
|
MDBX_ASAN_POISON_MEMORY_REGION(env->me_map + pgno2bytes(env, last),
|
||||||
pgno2bytes(env, edge - last));
|
pgno2bytes(env, edge - last));
|
||||||
}
|
}
|
||||||
if (should_unlock)
|
if (should_unlock)
|
||||||
@ -10554,7 +10557,8 @@ static int mdbx_sync_locked(MDBX_env *env, unsigned flags,
|
|||||||
env->me_poison_edge = largest_pgno;
|
env->me_poison_edge = largest_pgno;
|
||||||
VALGRIND_MAKE_MEM_NOACCESS(env->me_map + pgno2bytes(env, largest_pgno),
|
VALGRIND_MAKE_MEM_NOACCESS(env->me_map + pgno2bytes(env, largest_pgno),
|
||||||
pgno2bytes(env, edge - largest_pgno));
|
pgno2bytes(env, edge - largest_pgno));
|
||||||
ASAN_POISON_MEMORY_REGION(env->me_map + pgno2bytes(env, largest_pgno),
|
MDBX_ASAN_POISON_MEMORY_REGION(env->me_map +
|
||||||
|
pgno2bytes(env, largest_pgno),
|
||||||
pgno2bytes(env, edge - largest_pgno));
|
pgno2bytes(env, edge - largest_pgno));
|
||||||
}
|
}
|
||||||
#endif /* MDBX_USE_VALGRIND || __SANITIZE_ADDRESS__ */
|
#endif /* MDBX_USE_VALGRIND || __SANITIZE_ADDRESS__ */
|
||||||
@ -11641,7 +11645,7 @@ __cold static int mdbx_setup_dxb(MDBX_env *env, const int lck_rc,
|
|||||||
if (env->me_dxb_mmap.filesize > used_bytes) {
|
if (env->me_dxb_mmap.filesize > used_bytes) {
|
||||||
VALGRIND_MAKE_MEM_NOACCESS(env->me_map + used_bytes,
|
VALGRIND_MAKE_MEM_NOACCESS(env->me_map + used_bytes,
|
||||||
env->me_dxb_mmap.filesize - used_bytes);
|
env->me_dxb_mmap.filesize - used_bytes);
|
||||||
ASAN_POISON_MEMORY_REGION(env->me_map + used_bytes,
|
MDBX_ASAN_POISON_MEMORY_REGION(env->me_map + used_bytes,
|
||||||
env->me_dxb_mmap.filesize - used_bytes);
|
env->me_dxb_mmap.filesize - used_bytes);
|
||||||
}
|
}
|
||||||
env->me_poison_edge = bytes2pgno(env, env->me_dxb_mmap.filesize);
|
env->me_poison_edge = bytes2pgno(env, env->me_dxb_mmap.filesize);
|
||||||
@ -12825,7 +12829,7 @@ __cold int mdbx_env_close_ex(MDBX_env *env, bool dont_sync) {
|
|||||||
#endif /* MDBX_LOCKING */
|
#endif /* MDBX_LOCKING */
|
||||||
|
|
||||||
while ((dp = env->me_dp_reserve) != NULL) {
|
while ((dp = env->me_dp_reserve) != NULL) {
|
||||||
ASAN_UNPOISON_MEMORY_REGION(dp, env->me_psize);
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(dp, env->me_psize);
|
||||||
VALGRIND_MAKE_MEM_DEFINED(&dp->mp_next, sizeof(dp->mp_next));
|
VALGRIND_MAKE_MEM_DEFINED(&dp->mp_next, sizeof(dp->mp_next));
|
||||||
env->me_dp_reserve = dp->mp_next;
|
env->me_dp_reserve = dp->mp_next;
|
||||||
mdbx_free(dp);
|
mdbx_free(dp);
|
||||||
@ -21759,7 +21763,7 @@ __cold int mdbx_env_set_option(MDBX_env *env, const MDBX_option_t option,
|
|||||||
while (env->me_dp_reserve_len > env->me_options.dp_reserve_limit) {
|
while (env->me_dp_reserve_len > env->me_options.dp_reserve_limit) {
|
||||||
mdbx_assert(env, env->me_dp_reserve != NULL);
|
mdbx_assert(env, env->me_dp_reserve != NULL);
|
||||||
MDBX_page *dp = env->me_dp_reserve;
|
MDBX_page *dp = env->me_dp_reserve;
|
||||||
ASAN_UNPOISON_MEMORY_REGION(dp, env->me_psize);
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(dp, env->me_psize);
|
||||||
VALGRIND_MAKE_MEM_DEFINED(&dp->mp_next, sizeof(dp->mp_next));
|
VALGRIND_MAKE_MEM_DEFINED(&dp->mp_next, sizeof(dp->mp_next));
|
||||||
env->me_dp_reserve = dp->mp_next;
|
env->me_dp_reserve = dp->mp_next;
|
||||||
VALGRIND_MEMPOOL_FREE(env, dp);
|
VALGRIND_MEMPOOL_FREE(env, dp);
|
||||||
|
@ -1646,3 +1646,17 @@ MDBX_MAYBE_UNUSED static void static_checks(void) {
|
|||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#define MDBX_ASAN_POISON_MEMORY_REGION(addr, size) \
|
||||||
|
do { \
|
||||||
|
mdbx_trace("POISON_MEMORY_REGION(%p, %zu) at %u", (void *)(addr), \
|
||||||
|
(size_t)(size), __LINE__); \
|
||||||
|
ASAN_POISON_MEMORY_REGION(addr, size); \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
|
#define MDBX_ASAN_UNPOISON_MEMORY_REGION(addr, size) \
|
||||||
|
do { \
|
||||||
|
mdbx_trace("UNPOISON_MEMORY_REGION(%p, %zu) at %u", (void *)(addr), \
|
||||||
|
(size_t)(size), __LINE__); \
|
||||||
|
ASAN_UNPOISON_MEMORY_REGION(addr, size); \
|
||||||
|
} while (0)
|
||||||
|
18
src/osal.c
18
src/osal.c
@ -1538,7 +1538,7 @@ MDBX_INTERNAL_FUNC int mdbx_mmap(const int flags, mdbx_mmap_t *map,
|
|||||||
#endif /* ! Windows */
|
#endif /* ! Windows */
|
||||||
|
|
||||||
VALGRIND_MAKE_MEM_DEFINED(map->address, map->current);
|
VALGRIND_MAKE_MEM_DEFINED(map->address, map->current);
|
||||||
ASAN_UNPOISON_MEMORY_REGION(map->address, map->current);
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(map->address, map->current);
|
||||||
return MDBX_SUCCESS;
|
return MDBX_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1547,7 +1547,7 @@ MDBX_INTERNAL_FUNC int mdbx_munmap(mdbx_mmap_t *map) {
|
|||||||
/* Unpoisoning is required for ASAN to avoid false-positive diagnostic
|
/* Unpoisoning is required for ASAN to avoid false-positive diagnostic
|
||||||
* when this memory will re-used by malloc or another mmapping.
|
* when this memory will re-used by malloc or another mmapping.
|
||||||
* See https://github.com/erthink/libmdbx/pull/93#issuecomment-613687203 */
|
* See https://github.com/erthink/libmdbx/pull/93#issuecomment-613687203 */
|
||||||
ASAN_UNPOISON_MEMORY_REGION(map->address,
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(map->address,
|
||||||
(map->filesize && map->filesize < map->limit)
|
(map->filesize && map->filesize < map->limit)
|
||||||
? map->filesize
|
? map->filesize
|
||||||
: map->limit);
|
: map->limit);
|
||||||
@ -1623,7 +1623,7 @@ MDBX_INTERNAL_FUNC int mdbx_mresize(const int flags, mdbx_mmap_t *map,
|
|||||||
/* Unpoisoning is required for ASAN to avoid false-positive diagnostic
|
/* Unpoisoning is required for ASAN to avoid false-positive diagnostic
|
||||||
* when this memory will re-used by malloc or another mmapping.
|
* when this memory will re-used by malloc or another mmapping.
|
||||||
* See https://github.com/erthink/libmdbx/pull/93#issuecomment-613687203 */
|
* See https://github.com/erthink/libmdbx/pull/93#issuecomment-613687203 */
|
||||||
ASAN_UNPOISON_MEMORY_REGION(map->address, map->limit);
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(map->address, map->limit);
|
||||||
status = NtUnmapViewOfSection(GetCurrentProcess(), map->address);
|
status = NtUnmapViewOfSection(GetCurrentProcess(), map->address);
|
||||||
if (!NT_SUCCESS(status))
|
if (!NT_SUCCESS(status))
|
||||||
return ntstatus2errcode(status);
|
return ntstatus2errcode(status);
|
||||||
@ -1769,7 +1769,7 @@ retry_mapview:;
|
|||||||
* this region and (therefore) do not need the help of ASAN.
|
* this region and (therefore) do not need the help of ASAN.
|
||||||
* - this allows us to clear the mask only within the file size
|
* - this allows us to clear the mask only within the file size
|
||||||
* when closing the mapping. */
|
* when closing the mapping. */
|
||||||
ASAN_UNPOISON_MEMORY_REGION(
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(
|
||||||
(char *)map->address + size,
|
(char *)map->address + size,
|
||||||
((map->current < map->limit) ? map->current : map->limit) - size);
|
((map->current < map->limit) ? map->current : map->limit) - size);
|
||||||
}
|
}
|
||||||
@ -1885,9 +1885,9 @@ retry_mapview:;
|
|||||||
* when this memory will re-used by malloc or another mmapping.
|
* when this memory will re-used by malloc or another mmapping.
|
||||||
* See https://github.com/erthink/libmdbx/pull/93#issuecomment-613687203
|
* See https://github.com/erthink/libmdbx/pull/93#issuecomment-613687203
|
||||||
*/
|
*/
|
||||||
ASAN_UNPOISON_MEMORY_REGION(map->address, (map->current < map->limit)
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(
|
||||||
? map->current
|
map->address,
|
||||||
: map->limit);
|
(map->current < map->limit) ? map->current : map->limit);
|
||||||
map->limit = 0;
|
map->limit = 0;
|
||||||
map->current = 0;
|
map->current = 0;
|
||||||
map->address = nullptr;
|
map->address = nullptr;
|
||||||
@ -1904,11 +1904,11 @@ retry_mapview:;
|
|||||||
/* Unpoisoning is required for ASAN to avoid false-positive diagnostic
|
/* Unpoisoning is required for ASAN to avoid false-positive diagnostic
|
||||||
* when this memory will re-used by malloc or another mmapping.
|
* when this memory will re-used by malloc or another mmapping.
|
||||||
* See https://github.com/erthink/libmdbx/pull/93#issuecomment-613687203 */
|
* See https://github.com/erthink/libmdbx/pull/93#issuecomment-613687203 */
|
||||||
ASAN_UNPOISON_MEMORY_REGION(
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(
|
||||||
map->address, (map->current < map->limit) ? map->current : map->limit);
|
map->address, (map->current < map->limit) ? map->current : map->limit);
|
||||||
|
|
||||||
VALGRIND_MAKE_MEM_DEFINED(ptr, map->current);
|
VALGRIND_MAKE_MEM_DEFINED(ptr, map->current);
|
||||||
ASAN_UNPOISON_MEMORY_REGION(ptr, map->current);
|
MDBX_ASAN_UNPOISON_MEMORY_REGION(ptr, map->current);
|
||||||
map->address = ptr;
|
map->address = ptr;
|
||||||
}
|
}
|
||||||
map->limit = limit;
|
map->limit = limit;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user