mirror of
https://github.com/microsoft/mimalloc.git
synced 2025-05-08 00:09:31 +03:00
Merge branch 'dev' into dev-slice
This commit is contained in:
commit
a1cfe9667c
5 changed files with 56 additions and 40 deletions
|
@ -121,7 +121,7 @@ bool _mi_os_decommit(void* addr, size_t size);
|
||||||
bool _mi_os_protect(void* addr, size_t size);
|
bool _mi_os_protect(void* addr, size_t size);
|
||||||
bool _mi_os_unprotect(void* addr, size_t size);
|
bool _mi_os_unprotect(void* addr, size_t size);
|
||||||
bool _mi_os_purge(void* p, size_t size);
|
bool _mi_os_purge(void* p, size_t size);
|
||||||
bool _mi_os_purge_ex(void* p, size_t size, bool allow_reset);
|
bool _mi_os_purge_ex(void* p, size_t size, bool allow_reset, size_t stat_size);
|
||||||
|
|
||||||
void* _mi_os_alloc_aligned(size_t size, size_t alignment, bool commit, bool allow_large, mi_memid_t* memid);
|
void* _mi_os_alloc_aligned(size_t size, size_t alignment, bool commit, bool allow_large, mi_memid_t* memid);
|
||||||
void* _mi_os_alloc_aligned_at_offset(size_t size, size_t alignment, size_t align_offset, bool commit, bool allow_large, mi_memid_t* memid);
|
void* _mi_os_alloc_aligned_at_offset(size_t size, size_t alignment, size_t align_offset, bool commit, bool allow_large, mi_memid_t* memid);
|
||||||
|
|
|
@ -422,7 +422,7 @@ static inline bool mi_memkind_is_os(mi_memkind_t memkind) {
|
||||||
|
|
||||||
typedef struct mi_memid_os_info {
|
typedef struct mi_memid_os_info {
|
||||||
void* base; // actual base address of the block (used for offset aligned allocations)
|
void* base; // actual base address of the block (used for offset aligned allocations)
|
||||||
size_t alignment; // alignment at allocation
|
size_t size; // full allocation size
|
||||||
} mi_memid_os_info_t;
|
} mi_memid_os_info_t;
|
||||||
|
|
||||||
typedef struct mi_memid_arena_info {
|
typedef struct mi_memid_arena_info {
|
||||||
|
|
|
@ -478,8 +478,7 @@ static void mi_arena_purge(mi_arena_t* arena, size_t bitmap_idx, size_t blocks)
|
||||||
// we need to ensure we do not try to reset (as that may be invalid for uncommitted memory),
|
// we need to ensure we do not try to reset (as that may be invalid for uncommitted memory),
|
||||||
// and also undo the decommit stats (as it was already adjusted)
|
// and also undo the decommit stats (as it was already adjusted)
|
||||||
mi_assert_internal(mi_option_is_enabled(mi_option_purge_decommits));
|
mi_assert_internal(mi_option_is_enabled(mi_option_purge_decommits));
|
||||||
needs_recommit = _mi_os_purge_ex(p, size, false /* allow reset? */);
|
needs_recommit = _mi_os_purge_ex(p, size, false /* allow reset? */, 0);
|
||||||
if (needs_recommit) { _mi_stat_increase(&_mi_stats_main.committed, size); }
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// clear the purged blocks
|
// clear the purged blocks
|
||||||
|
|
84
src/os.c
84
src/os.c
|
@ -9,7 +9,9 @@ terms of the MIT license. A copy of the license can be found in the file
|
||||||
#include "mimalloc/atomic.h"
|
#include "mimalloc/atomic.h"
|
||||||
#include "mimalloc/prim.h"
|
#include "mimalloc/prim.h"
|
||||||
|
|
||||||
#define os_stats (&_mi_stats_main)
|
#define mi_os_stat_increase(stat,amount) _mi_stat_increase(&_mi_stats_main.stat, amount)
|
||||||
|
#define mi_os_stat_decrease(stat,amount) _mi_stat_decrease(&_mi_stats_main.stat, amount)
|
||||||
|
#define mi_os_stat_counter_increase(stat,inc) _mi_stat_counter_increase(&_mi_stats_main.stat, inc)
|
||||||
|
|
||||||
/* -----------------------------------------------------------
|
/* -----------------------------------------------------------
|
||||||
Initialization.
|
Initialization.
|
||||||
|
@ -142,34 +144,42 @@ void* _mi_os_get_aligned_hint(size_t try_alignment, size_t size) {
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
/* -----------------------------------------------------------
|
/* -----------------------------------------------------------
|
||||||
Free memory
|
Free memory
|
||||||
-------------------------------------------------------------- */
|
-------------------------------------------------------------- */
|
||||||
|
|
||||||
static void mi_os_free_huge_os_pages(void* p, size_t size);
|
static void mi_os_free_huge_os_pages(void* p, size_t size);
|
||||||
|
|
||||||
static void mi_os_prim_free(void* addr, size_t size, bool still_committed) {
|
static void mi_os_prim_free(void* addr, size_t size, size_t commit_size) {
|
||||||
mi_assert_internal((size % _mi_os_page_size()) == 0);
|
mi_assert_internal((size % _mi_os_page_size()) == 0);
|
||||||
if (addr == NULL || size == 0) return; // || _mi_os_is_huge_reserved(addr)
|
if (addr == NULL || size == 0) return; // || _mi_os_is_huge_reserved(addr)
|
||||||
int err = _mi_prim_free(addr, size);
|
int err = _mi_prim_free(addr, size);
|
||||||
if (err != 0) {
|
if (err != 0) {
|
||||||
_mi_warning_message("unable to free OS memory (error: %d (0x%x), size: 0x%zx bytes, address: %p)\n", err, err, size, addr);
|
_mi_warning_message("unable to free OS memory (error: %d (0x%x), size: 0x%zx bytes, address: %p)\n", err, err, size, addr);
|
||||||
}
|
}
|
||||||
if (still_committed) { _mi_stat_decrease(&os_stats->committed, size); }
|
if (commit_size > 0) {
|
||||||
_mi_stat_decrease(&os_stats->reserved, size);
|
mi_os_stat_decrease(committed, commit_size);
|
||||||
|
}
|
||||||
|
mi_os_stat_decrease(reserved, size);
|
||||||
}
|
}
|
||||||
|
|
||||||
void _mi_os_free_ex(void* addr, size_t size, bool still_committed, mi_memid_t memid) {
|
void _mi_os_free_ex(void* addr, size_t size, bool still_committed, mi_memid_t memid) {
|
||||||
if (mi_memkind_is_os(memid.memkind)) {
|
if (mi_memkind_is_os(memid.memkind)) {
|
||||||
size_t csize = _mi_os_good_alloc_size(size);
|
size_t csize = memid.mem.os.size;
|
||||||
|
if (csize==0) { _mi_os_good_alloc_size(size); }
|
||||||
|
size_t commit_size = (still_committed ? csize : 0);
|
||||||
void* base = addr;
|
void* base = addr;
|
||||||
// different base? (due to alignment)
|
// different base? (due to alignment)
|
||||||
if (memid.mem.os.base != NULL) {
|
if (memid.mem.os.base != base) {
|
||||||
mi_assert(memid.mem.os.base <= addr);
|
mi_assert(memid.mem.os.base <= addr);
|
||||||
mi_assert((uint8_t*)memid.mem.os.base + memid.mem.os.alignment >= (uint8_t*)addr);
|
|
||||||
base = memid.mem.os.base;
|
base = memid.mem.os.base;
|
||||||
csize += ((uint8_t*)addr - (uint8_t*)memid.mem.os.base);
|
const size_t diff = (uint8_t*)addr - (uint8_t*)memid.mem.os.base;
|
||||||
|
if (memid.mem.os.size==0) {
|
||||||
|
csize += diff;
|
||||||
|
}
|
||||||
|
if (still_committed) {
|
||||||
|
commit_size -= diff; // the (addr-base) part was already un-committed
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// free it
|
// free it
|
||||||
if (memid.memkind == MI_MEM_OS_HUGE) {
|
if (memid.memkind == MI_MEM_OS_HUGE) {
|
||||||
|
@ -177,7 +187,7 @@ void _mi_os_free_ex(void* addr, size_t size, bool still_committed, mi_memid_t me
|
||||||
mi_os_free_huge_os_pages(base, csize);
|
mi_os_free_huge_os_pages(base, csize);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
mi_os_prim_free(base, csize, still_committed);
|
mi_os_prim_free(base, csize, (still_committed ? commit_size : 0));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
@ -213,11 +223,11 @@ static void* mi_os_prim_alloc_at(void* hint_addr, size_t size, size_t try_alignm
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
mi_stat_counter_increase(os_stats->mmap_calls, 1);
|
mi_os_stat_counter_increase(mmap_calls, 1);
|
||||||
if (p != NULL) {
|
if (p != NULL) {
|
||||||
_mi_stat_increase(&os_stats->reserved, size);
|
mi_os_stat_increase(reserved, size);
|
||||||
if (commit) {
|
if (commit) {
|
||||||
_mi_stat_increase(&os_stats->committed, size);
|
mi_os_stat_increase(committed, size);
|
||||||
// seems needed for asan (or `mimalloc-test-api` fails)
|
// seems needed for asan (or `mimalloc-test-api` fails)
|
||||||
#ifdef MI_TRACK_ASAN
|
#ifdef MI_TRACK_ASAN
|
||||||
if (*is_zero) { mi_track_mem_defined(p,size); }
|
if (*is_zero) { mi_track_mem_defined(p,size); }
|
||||||
|
@ -258,7 +268,7 @@ static void* mi_os_prim_alloc_aligned(size_t size, size_t alignment, bool commit
|
||||||
#if !MI_TRACK_ASAN
|
#if !MI_TRACK_ASAN
|
||||||
_mi_warning_message("unable to allocate aligned OS memory directly, fall back to over-allocation (size: 0x%zx bytes, address: %p, alignment: 0x%zx, commit: %d)\n", size, p, alignment, commit);
|
_mi_warning_message("unable to allocate aligned OS memory directly, fall back to over-allocation (size: 0x%zx bytes, address: %p, alignment: 0x%zx, commit: %d)\n", size, p, alignment, commit);
|
||||||
#endif
|
#endif
|
||||||
mi_os_prim_free(p, size, commit);
|
if (p != NULL) { mi_os_prim_free(p, size, (commit ? size : 0)); }
|
||||||
if (size >= (SIZE_MAX - alignment)) return NULL; // overflow
|
if (size >= (SIZE_MAX - alignment)) return NULL; // overflow
|
||||||
const size_t over_size = size + alignment;
|
const size_t over_size = size + alignment;
|
||||||
|
|
||||||
|
@ -289,8 +299,8 @@ static void* mi_os_prim_alloc_aligned(size_t size, size_t alignment, bool commit
|
||||||
size_t mid_size = _mi_align_up(size, _mi_os_page_size());
|
size_t mid_size = _mi_align_up(size, _mi_os_page_size());
|
||||||
size_t post_size = over_size - pre_size - mid_size;
|
size_t post_size = over_size - pre_size - mid_size;
|
||||||
mi_assert_internal(pre_size < over_size&& post_size < over_size&& mid_size >= size);
|
mi_assert_internal(pre_size < over_size&& post_size < over_size&& mid_size >= size);
|
||||||
if (pre_size > 0) { mi_os_prim_free(p, pre_size, commit); }
|
if (pre_size > 0) { mi_os_prim_free(p, pre_size, (commit ? pre_size : 0)); }
|
||||||
if (post_size > 0) { mi_os_prim_free((uint8_t*)aligned_p + mid_size, post_size, commit); }
|
if (post_size > 0) { mi_os_prim_free((uint8_t*)aligned_p + mid_size, post_size, (commit ? post_size : 0)); }
|
||||||
// we can return the aligned pointer on `mmap` systems
|
// we can return the aligned pointer on `mmap` systems
|
||||||
p = aligned_p;
|
p = aligned_p;
|
||||||
*base = aligned_p; // since we freed the pre part, `*base == p`.
|
*base = aligned_p; // since we freed the pre part, `*base == p`.
|
||||||
|
@ -334,7 +344,8 @@ void* _mi_os_alloc_aligned(size_t size, size_t alignment, bool commit, bool allo
|
||||||
if (p != NULL) {
|
if (p != NULL) {
|
||||||
*memid = _mi_memid_create_os(commit, os_is_zero, os_is_large);
|
*memid = _mi_memid_create_os(commit, os_is_zero, os_is_large);
|
||||||
memid->mem.os.base = os_base;
|
memid->mem.os.base = os_base;
|
||||||
memid->mem.os.alignment = alignment;
|
// memid->mem.os.alignment = alignment;
|
||||||
|
memid->mem.os.size += ((uint8_t*)p - (uint8_t*)os_base); // todo: return from prim_alloc_aligned
|
||||||
}
|
}
|
||||||
return p;
|
return p;
|
||||||
}
|
}
|
||||||
|
@ -402,10 +413,10 @@ static void* mi_os_page_align_area_conservative(void* addr, size_t size, size_t*
|
||||||
return mi_os_page_align_areax(true, addr, size, newsize);
|
return mi_os_page_align_areax(true, addr, size, newsize);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool _mi_os_commit(void* addr, size_t size, bool* is_zero) {
|
bool _mi_os_commit_ex(void* addr, size_t size, bool* is_zero, size_t stat_size) {
|
||||||
if (is_zero != NULL) { *is_zero = false; }
|
if (is_zero != NULL) { *is_zero = false; }
|
||||||
_mi_stat_increase(&os_stats->committed, size); // use size for precise commit vs. decommit
|
mi_os_stat_increase(committed, stat_size); // use size for precise commit vs. decommit
|
||||||
_mi_stat_counter_increase(&os_stats->commit_calls, 1);
|
mi_os_stat_counter_increase(commit_calls, 1);
|
||||||
|
|
||||||
// page align range
|
// page align range
|
||||||
size_t csize;
|
size_t csize;
|
||||||
|
@ -431,8 +442,13 @@ bool _mi_os_commit(void* addr, size_t size, bool* is_zero) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool mi_os_decommit_ex(void* addr, size_t size, bool* needs_recommit) { mi_assert_internal(needs_recommit!=NULL);
|
bool _mi_os_commit(void* addr, size_t size, bool* is_zero) {
|
||||||
_mi_stat_decrease(&os_stats->committed, size);
|
return _mi_os_commit_ex(addr, size, is_zero, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool mi_os_decommit_ex(void* addr, size_t size, bool* needs_recommit, size_t stat_size) {
|
||||||
|
mi_assert_internal(needs_recommit!=NULL);
|
||||||
|
mi_os_stat_decrease(committed, stat_size);
|
||||||
|
|
||||||
// page align
|
// page align
|
||||||
size_t csize;
|
size_t csize;
|
||||||
|
@ -451,7 +467,7 @@ static bool mi_os_decommit_ex(void* addr, size_t size, bool* needs_recommit) {
|
||||||
|
|
||||||
bool _mi_os_decommit(void* addr, size_t size) {
|
bool _mi_os_decommit(void* addr, size_t size) {
|
||||||
bool needs_recommit;
|
bool needs_recommit;
|
||||||
return mi_os_decommit_ex(addr, size, &needs_recommit);
|
return mi_os_decommit_ex(addr, size, &needs_recommit, size);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -464,8 +480,8 @@ bool _mi_os_reset(void* addr, size_t size) {
|
||||||
size_t csize;
|
size_t csize;
|
||||||
void* start = mi_os_page_align_area_conservative(addr, size, &csize);
|
void* start = mi_os_page_align_area_conservative(addr, size, &csize);
|
||||||
if (csize == 0) return true; // || _mi_os_is_huge_reserved(addr)
|
if (csize == 0) return true; // || _mi_os_is_huge_reserved(addr)
|
||||||
_mi_stat_increase(&os_stats->reset, csize);
|
mi_os_stat_increase(reset, csize);
|
||||||
_mi_stat_counter_increase(&os_stats->reset_calls, 1);
|
mi_os_stat_counter_increase(reset_calls, 1);
|
||||||
|
|
||||||
#if (MI_DEBUG>1) && !MI_SECURE && !MI_TRACK_ENABLED // && !MI_TSAN
|
#if (MI_DEBUG>1) && !MI_SECURE && !MI_TRACK_ENABLED // && !MI_TSAN
|
||||||
memset(start, 0, csize); // pretend it is eagerly reset
|
memset(start, 0, csize); // pretend it is eagerly reset
|
||||||
|
@ -481,17 +497,17 @@ bool _mi_os_reset(void* addr, size_t size) {
|
||||||
|
|
||||||
// either resets or decommits memory, returns true if the memory needs
|
// either resets or decommits memory, returns true if the memory needs
|
||||||
// to be recommitted if it is to be re-used later on.
|
// to be recommitted if it is to be re-used later on.
|
||||||
bool _mi_os_purge_ex(void* p, size_t size, bool allow_reset)
|
bool _mi_os_purge_ex(void* p, size_t size, bool allow_reset, size_t stat_size)
|
||||||
{
|
{
|
||||||
if (mi_option_get(mi_option_purge_delay) < 0) return false; // is purging allowed?
|
if (mi_option_get(mi_option_purge_delay) < 0) return false; // is purging allowed?
|
||||||
_mi_stat_counter_increase(&os_stats->purge_calls, 1);
|
mi_os_stat_counter_increase(purge_calls, 1);
|
||||||
_mi_stat_increase(&os_stats->purged, size);
|
mi_os_stat_increase(purged, size);
|
||||||
|
|
||||||
if (mi_option_is_enabled(mi_option_purge_decommits) && // should decommit?
|
if (mi_option_is_enabled(mi_option_purge_decommits) && // should decommit?
|
||||||
!_mi_preloading()) // don't decommit during preloading (unsafe)
|
!_mi_preloading()) // don't decommit during preloading (unsafe)
|
||||||
{
|
{
|
||||||
bool needs_recommit = true;
|
bool needs_recommit = true;
|
||||||
mi_os_decommit_ex(p, size, &needs_recommit);
|
mi_os_decommit_ex(p, size, &needs_recommit, stat_size);
|
||||||
return needs_recommit;
|
return needs_recommit;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
@ -505,7 +521,7 @@ bool _mi_os_purge_ex(void* p, size_t size, bool allow_reset)
|
||||||
// either resets or decommits memory, returns true if the memory needs
|
// either resets or decommits memory, returns true if the memory needs
|
||||||
// to be recommitted if it is to be re-used later on.
|
// to be recommitted if it is to be re-used later on.
|
||||||
bool _mi_os_purge(void* p, size_t size) {
|
bool _mi_os_purge(void* p, size_t size) {
|
||||||
return _mi_os_purge_ex(p, size, true);
|
return _mi_os_purge_ex(p, size, true, size);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Protect a region in memory to be not accessible.
|
// Protect a region in memory to be not accessible.
|
||||||
|
@ -612,15 +628,15 @@ void* _mi_os_alloc_huge_os_pages(size_t pages, int numa_node, mi_msecs_t max_mse
|
||||||
// no success, issue a warning and break
|
// no success, issue a warning and break
|
||||||
if (p != NULL) {
|
if (p != NULL) {
|
||||||
_mi_warning_message("could not allocate contiguous huge OS page %zu at %p\n", page, addr);
|
_mi_warning_message("could not allocate contiguous huge OS page %zu at %p\n", page, addr);
|
||||||
mi_os_prim_free(p, MI_HUGE_OS_PAGE_SIZE, true);
|
mi_os_prim_free(p, MI_HUGE_OS_PAGE_SIZE, MI_HUGE_OS_PAGE_SIZE);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
// success, record it
|
// success, record it
|
||||||
page++; // increase before timeout check (see issue #711)
|
page++; // increase before timeout check (see issue #711)
|
||||||
_mi_stat_increase(&os_stats->committed, MI_HUGE_OS_PAGE_SIZE);
|
mi_os_stat_increase(committed, MI_HUGE_OS_PAGE_SIZE);
|
||||||
_mi_stat_increase(&os_stats->reserved, MI_HUGE_OS_PAGE_SIZE);
|
mi_os_stat_increase(reserved, MI_HUGE_OS_PAGE_SIZE);
|
||||||
|
|
||||||
// check for timeout
|
// check for timeout
|
||||||
if (max_msecs > 0) {
|
if (max_msecs > 0) {
|
||||||
|
@ -658,7 +674,7 @@ static void mi_os_free_huge_os_pages(void* p, size_t size) {
|
||||||
if (p==NULL || size==0) return;
|
if (p==NULL || size==0) return;
|
||||||
uint8_t* base = (uint8_t*)p;
|
uint8_t* base = (uint8_t*)p;
|
||||||
while (size >= MI_HUGE_OS_PAGE_SIZE) {
|
while (size >= MI_HUGE_OS_PAGE_SIZE) {
|
||||||
mi_os_prim_free(base, MI_HUGE_OS_PAGE_SIZE, true);
|
mi_os_prim_free(base, MI_HUGE_OS_PAGE_SIZE, MI_HUGE_OS_PAGE_SIZE);
|
||||||
size -= MI_HUGE_OS_PAGE_SIZE;
|
size -= MI_HUGE_OS_PAGE_SIZE;
|
||||||
base += MI_HUGE_OS_PAGE_SIZE;
|
base += MI_HUGE_OS_PAGE_SIZE;
|
||||||
}
|
}
|
||||||
|
|
|
@ -64,6 +64,7 @@ static bool main_participates = false; // main thread participates as a
|
||||||
#define custom_calloc(n,s) mi_calloc(n,s)
|
#define custom_calloc(n,s) mi_calloc(n,s)
|
||||||
#define custom_realloc(p,s) mi_realloc(p,s)
|
#define custom_realloc(p,s) mi_realloc(p,s)
|
||||||
#define custom_free(p) mi_free(p)
|
#define custom_free(p) mi_free(p)
|
||||||
|
|
||||||
#ifndef NDEBUG
|
#ifndef NDEBUG
|
||||||
#define HEAP_WALK // walk the heap objects?
|
#define HEAP_WALK // walk the heap objects?
|
||||||
#endif
|
#endif
|
||||||
|
@ -223,7 +224,7 @@ static void test_stress(void) {
|
||||||
run_os_threads(THREADS, &stress);
|
run_os_threads(THREADS, &stress);
|
||||||
#if !defined(NDEBUG) && !defined(USE_STD_MALLOC)
|
#if !defined(NDEBUG) && !defined(USE_STD_MALLOC)
|
||||||
// switch between arena and OS allocation for testing
|
// switch between arena and OS allocation for testing
|
||||||
mi_option_set_enabled(mi_option_disallow_arena_alloc, (n%2)==1);
|
// mi_option_set_enabled(mi_option_disallow_arena_alloc, (n%2)==1);
|
||||||
#endif
|
#endif
|
||||||
#ifdef HEAP_WALK
|
#ifdef HEAP_WALK
|
||||||
size_t total = 0;
|
size_t total = 0;
|
||||||
|
|
Loading…
Add table
Reference in a new issue