From 1740043678951c5a1a088e6572722de438162ab2 Mon Sep 17 00:00:00 2001 From: Leonid Yuriev Date: Sat, 10 Jul 2021 16:09:02 +0300 Subject: [PATCH] mdbx: minimize the size of poisoned/unpoisoned regions to avoid ASAN hangs. More for second case of https://github.com/erthink/libmdbx/issues/217 --- src/core.c | 12 +++++++----- src/osal.c | 43 +++++++++++++++++++++++++++++++++---------- 2 files changed, 40 insertions(+), 15 deletions(-) diff --git a/src/core.c b/src/core.c index 07150e1a..7b8ce52f 100644 --- a/src/core.c +++ b/src/core.c @@ -11630,11 +11630,13 @@ __cold static int mdbx_setup_dxb(MDBX_env *env, const int lck_rc, mdbx_assert(env, used_bytes >= pgno2bytes(env, NUM_METAS) && used_bytes <= env->me_dxb_mmap.limit); #if defined(MDBX_USE_VALGRIND) || defined(__SANITIZE_ADDRESS__) - VALGRIND_MAKE_MEM_NOACCESS(env->me_map + used_bytes, - env->me_dxb_mmap.limit - used_bytes); - ASAN_POISON_MEMORY_REGION(env->me_map + used_bytes, - env->me_dxb_mmap.limit - used_bytes); - env->me_poison_edge = bytes2pgno(env, env->me_dxb_mmap.limit); + if (env->me_dxb_mmap.filesize > used_bytes) { + VALGRIND_MAKE_MEM_NOACCESS(env->me_map + used_bytes, + env->me_dxb_mmap.filesize - used_bytes); + ASAN_POISON_MEMORY_REGION(env->me_map + used_bytes, + env->me_dxb_mmap.filesize - used_bytes); + } + env->me_poison_edge = bytes2pgno(env, env->me_dxb_mmap.filesize); #endif /* MDBX_USE_VALGRIND || __SANITIZE_ADDRESS__ */ //----------------------------------------- validate head & steady meta-pages diff --git a/src/osal.c b/src/osal.c index dfe1564c..9bce93ed 100644 --- a/src/osal.c +++ b/src/osal.c @@ -1547,7 +1547,10 @@ MDBX_INTERNAL_FUNC int mdbx_munmap(mdbx_mmap_t *map) { /* Unpoisoning is required for ASAN to avoid false-positive diagnostic * when this memory will re-used by malloc or another mmapping. * See https://github.com/erthink/libmdbx/pull/93#issuecomment-613687203 */ - ASAN_UNPOISON_MEMORY_REGION(map->address, map->limit); + ASAN_UNPOISON_MEMORY_REGION(map->address, + (map->filesize && map->filesize < map->limit) + ? map->filesize + : map->limit); #if defined(_WIN32) || defined(_WIN64) if (map->section) NtClose(map->section); @@ -1617,6 +1620,10 @@ MDBX_INTERNAL_FUNC int mdbx_mresize(const int flags, mdbx_mmap_t *map, if ((flags & MDBX_MRESIZE_MAY_UNMAP) == 0) return MDBX_RESULT_TRUE; + /* Unpoisoning is required for ASAN to avoid false-positive diagnostic + * when this memory will re-used by malloc or another mmapping. + * See https://github.com/erthink/libmdbx/pull/93#issuecomment-613687203 */ + ASAN_UNPOISON_MEMORY_REGION(map->address, map->limit); status = NtUnmapViewOfSection(GetCurrentProcess(), map->address); if (!NT_SUCCESS(status)) return ntstatus2errcode(status); @@ -1747,11 +1754,25 @@ retry_mapview:; if (map->current != size) rc = (size > map->current) ? MDBX_UNABLE_EXTEND_MAPSIZE : MDBX_RESULT_TRUE; - } else if (map->filesize != size) { - rc = mdbx_ftruncate(map->fd, size); - if (rc != MDBX_SUCCESS) - return rc; - map->filesize = size; + } else { + if (map->filesize != size) { + rc = mdbx_ftruncate(map->fd, size); + if (rc != MDBX_SUCCESS) + return rc; + map->filesize = size; + } + + if (map->current > size) { + /* Clearing asan's bitmask for the region which released in shrinking, + * since: + * - after the shrinking we will get an exception when accessing + * this region and (therefore) do not need the help of ASAN. + * - this allows us to clear the mask only within the file size + * when closing the mapping. */ + ASAN_UNPOISON_MEMORY_REGION( + (char *)map->address + size, + ((map->current < map->limit) ? map->current : map->limit) - size); + } map->current = size; } @@ -1864,7 +1885,9 @@ retry_mapview:; * when this memory will re-used by malloc or another mmapping. * See https://github.com/erthink/libmdbx/pull/93#issuecomment-613687203 */ - ASAN_UNPOISON_MEMORY_REGION(map->address, map->limit); + ASAN_UNPOISON_MEMORY_REGION(map->address, (map->current < map->limit) + ? map->current + : map->limit); map->limit = 0; map->current = 0; map->address = nullptr; @@ -1880,9 +1903,9 @@ retry_mapview:; VALGRIND_MAKE_MEM_NOACCESS(map->address, map->current); /* Unpoisoning is required for ASAN to avoid false-positive diagnostic * when this memory will re-used by malloc or another mmapping. - * See https://github.com/erthink/libmdbx/pull/93#issuecomment-613687203 - */ - ASAN_UNPOISON_MEMORY_REGION(map->address, map->limit); + * See https://github.com/erthink/libmdbx/pull/93#issuecomment-613687203 */ + ASAN_UNPOISON_MEMORY_REGION( + map->address, (map->current < map->limit) ? map->current : map->limit); VALGRIND_MAKE_MEM_DEFINED(ptr, map->current); ASAN_UNPOISON_MEMORY_REGION(ptr, map->current);