Skip to content

Commit d389819

Browse files
committed
add initial support for _mi_prim_reuse and MADV_FREE_REUSABLE on macOS (issue #1097)
1 parent 2d34956 commit d389819

File tree

8 files changed

+64
-7
lines changed

8 files changed

+64
-7
lines changed

include/mimalloc/internal.h

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -167,6 +167,7 @@ bool _mi_os_decommit(void* addr, size_t size);
167167
bool _mi_os_unprotect(void* addr, size_t size);
168168
bool _mi_os_purge(void* p, size_t size);
169169
bool _mi_os_purge_ex(void* p, size_t size, bool allow_reset, size_t stat_size);
170+
void _mi_os_reuse(void* p, size_t size);
170171
mi_decl_nodiscard bool _mi_os_commit(void* p, size_t size, bool* is_zero);
171172
mi_decl_nodiscard bool _mi_os_commit_ex(void* addr, size_t size, bool* is_zero, size_t stat_size);
172173
mi_decl_nodiscard bool _mi_os_protect(void* addr, size_t size);

include/mimalloc/prim.h

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -63,6 +63,11 @@ int _mi_prim_decommit(void* addr, size_t size, bool* needs_recommit);
6363
// Returns error code or 0 on success.
6464
int _mi_prim_reset(void* addr, size_t size);
6565

66+
// Reuse memory. This is called for memory that is already committed but
67+
// may have been reset (`_mi_prim_reset`) or decommitted (`_mi_prim_decommit`) where `needs_recommit` was false.
68+
// Returns error code or 0 on success. On most platforms this is a no-op.
69+
int _mi_prim_reuse(void* addr, size_t size);
70+
6671
// Protect memory. Returns error code or 0 on success.
6772
int _mi_prim_protect(void* addr, size_t size, bool protect);
6873

src/arena.c

Lines changed: 5 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -266,12 +266,12 @@ static mi_decl_noinline void* mi_arena_try_alloc_at(mi_arena_t* arena, size_t ar
266266
else if (commit) {
267267
// commit requested, but the range may not be committed as a whole: ensure it is committed now
268268
memid->initially_committed = true;
269+
const size_t commit_size = mi_arena_block_size(needed_bcount);
269270
bool any_uncommitted;
270271
size_t already_committed = 0;
271272
_mi_bitmap_claim_across(arena->blocks_committed, arena->field_count, needed_bcount, bitmap_index, &any_uncommitted, &already_committed);
272273
if (any_uncommitted) {
273274
mi_assert_internal(already_committed < needed_bcount);
274-
const size_t commit_size = mi_arena_block_size(needed_bcount);
275275
const size_t stat_commit_size = commit_size - mi_arena_block_size(already_committed);
276276
bool commit_zero = false;
277277
if (!_mi_os_commit_ex(p, commit_size, &commit_zero, stat_commit_size)) {
@@ -281,6 +281,10 @@ static mi_decl_noinline void* mi_arena_try_alloc_at(mi_arena_t* arena, size_t ar
281281
if (commit_zero) { memid->initially_zero = true; }
282282
}
283283
}
284+
else {
285+
// all are already committed: signal that we are reusing memory in case it was purged before
286+
_mi_os_reuse( p, commit_size );
287+
}
284288
}
285289
else {
286290
// no need to commit, but check if already fully committed

src/os.c

Lines changed: 11 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -512,6 +512,17 @@ bool _mi_os_reset(void* addr, size_t size) {
512512
}
513513

514514

515+
void _mi_os_reuse( void* addr, size_t size ) {
516+
// page align conservatively within the range
517+
size_t csize = 0;
518+
void* const start = mi_os_page_align_area_conservative(addr, size, &csize);
519+
if (csize == 0) return;
520+
const int err = _mi_prim_reuse(start, csize);
521+
if (err != 0) {
522+
_mi_warning_message("cannot reuse OS memory (error: %d (0x%x), address: %p, size: 0x%zx bytes)\n", err, err, start, csize);
523+
}
524+
}
525+
515526
// either resets or decommits memory, returns true if the memory needs
516527
// to be recommitted if it is to be re-used later on.
517528
bool _mi_os_purge_ex(void* p, size_t size, bool allow_reset, size_t stat_size)

src/prim/emscripten/prim.c

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -114,6 +114,11 @@ int _mi_prim_reset(void* addr, size_t size) {
114114
return 0;
115115
}
116116

117+
int _mi_prim_reuse(void* addr, size_t size) {
118+
MI_UNUSED(addr); MI_UNUSED(size);
119+
return 0;
120+
}
121+
117122
int _mi_prim_protect(void* addr, size_t size, bool protect) {
118123
MI_UNUSED(addr); MI_UNUSED(size); MI_UNUSED(protect);
119124
return 0;

src/prim/unix/prim.c

Lines changed: 27 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -433,13 +433,27 @@ int _mi_prim_commit(void* start, size_t size, bool* is_zero) {
433433
return err;
434434
}
435435

436+
int _mi_prim_reuse(void* start, size_t size) {
437+
#if defined(__APPLE__) && defined(MADV_FREE_REUSE)
438+
return unix_madvise(start, size, MADV_FREE_REUSE);
439+
#endif
440+
return 0;
441+
}
442+
436443
int _mi_prim_decommit(void* start, size_t size, bool* needs_recommit) {
437444
int err = 0;
438-
// decommit: use MADV_DONTNEED as it decreases rss immediately (unlike MADV_FREE)
439-
err = unix_madvise(start, size, MADV_DONTNEED);
440445
#if !MI_DEBUG && MI_SECURE<=2
441446
*needs_recommit = false;
447+
#if defined(__APPLE__) && defined(MADV_FREE_REUSABLE)
448+
// decommit on macOS: use MADV_FREE_REUSABLE as it does immediate rss accounting (issue #1097)
449+
err = unix_madvise(start, size, MADV_FREE_REUSABLE);
450+
#else
451+
// decommit: use MADV_DONTNEED as it decreases rss immediately (unlike MADV_FREE)
452+
err = unix_madvise(start, size, MADV_DONTNEED);
453+
#endif
442454
#else
455+
// note: don't use MADV_FREE_REUSABLE as the range may contain protected areas
456+
err = unix_madvise(start, size, MADV_DONTNEED);
443457
*needs_recommit = true;
444458
mprotect(start, size, PROT_NONE);
445459
#endif
@@ -454,22 +468,29 @@ int _mi_prim_decommit(void* start, size_t size, bool* needs_recommit) {
454468
}
455469

456470
int _mi_prim_reset(void* start, size_t size) {
457-
// We try to use `MADV_FREE` as that is the fastest. A drawback though is that it
471+
int err = 0;
472+
#if defined(__APPLE__) && defined(MADV_FREE_REUSABLE)
473+
// on macOS we try to use MADV_FREE_REUSABLE as it seems the fastest
474+
err = unix_madvise(start, size, MADV_FREE_REUSABLE);
475+
if (err == 0) return 0;
476+
// fall through
477+
#endif
478+
479+
#if defined(MADV_FREE)
480+
// Otherwise, we try to use `MADV_FREE` as that is the fastest. A drawback though is that it
458481
// will not reduce the `rss` stats in tools like `top` even though the memory is available
459482
// to other processes. With the default `MIMALLOC_PURGE_DECOMMITS=1` we ensure that by
460483
// default `MADV_DONTNEED` is used though.
461-
#if defined(MADV_FREE)
462484
static _Atomic(size_t) advice = MI_ATOMIC_VAR_INIT(MADV_FREE);
463485
int oadvice = (int)mi_atomic_load_relaxed(&advice);
464-
int err;
465486
while ((err = unix_madvise(start, size, oadvice)) != 0 && errno == EAGAIN) { errno = 0; };
466487
if (err != 0 && errno == EINVAL && oadvice == MADV_FREE) {
467488
// if MADV_FREE is not supported, fall back to MADV_DONTNEED from now on
468489
mi_atomic_store_release(&advice, (size_t)MADV_DONTNEED);
469490
err = unix_madvise(start, size, MADV_DONTNEED);
470491
}
471492
#else
472-
int err = unix_madvise(start, size, MADV_DONTNEED);
493+
err = unix_madvise(start, size, MADV_DONTNEED);
473494
#endif
474495
return err;
475496
}

src/prim/wasi/prim.c

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -149,6 +149,11 @@ int _mi_prim_reset(void* addr, size_t size) {
149149
return 0;
150150
}
151151

152+
int _mi_prim_reuse(void* addr, size_t size) {
153+
MI_UNUSED(addr); MI_UNUSED(size);
154+
return 0;
155+
}
156+
152157
int _mi_prim_protect(void* addr, size_t size, bool protect) {
153158
MI_UNUSED(addr); MI_UNUSED(size); MI_UNUSED(protect);
154159
return 0;

src/prim/windows/prim.c

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -352,6 +352,11 @@ int _mi_prim_reset(void* addr, size_t size) {
352352
return (p != NULL ? 0 : (int)GetLastError());
353353
}
354354

355+
int _mi_prim_reuse(void* addr, size_t size) {
356+
MI_UNUSED(addr); MI_UNUSED(size);
357+
return 0;
358+
}
359+
355360
int _mi_prim_protect(void* addr, size_t size, bool protect) {
356361
DWORD oldprotect = 0;
357362
BOOL ok = VirtualProtect(addr, size, protect ? PAGE_NOACCESS : PAGE_READWRITE, &oldprotect);

0 commit comments

Comments
 (0)