mirror of
https://github.com/microsoft/mimalloc.git
synced 2025-01-15 09:38:00 +08:00
fix assertions for huge segment free
This commit is contained in:
parent
dad3be3c64
commit
dcb3574cf0
@ -82,6 +82,7 @@ void _mi_segment_page_free(mi_page_t* page, bool force, mi_segments_tld_t*
|
|||||||
void _mi_segment_page_abandon(mi_page_t* page, mi_segments_tld_t* tld);
|
void _mi_segment_page_abandon(mi_page_t* page, mi_segments_tld_t* tld);
|
||||||
bool _mi_segment_try_reclaim_abandoned( mi_heap_t* heap, bool try_all, mi_segments_tld_t* tld);
|
bool _mi_segment_try_reclaim_abandoned( mi_heap_t* heap, bool try_all, mi_segments_tld_t* tld);
|
||||||
void _mi_segment_thread_collect(mi_segments_tld_t* tld);
|
void _mi_segment_thread_collect(mi_segments_tld_t* tld);
|
||||||
|
void _mi_segment_huge_page_free(mi_segment_t* segment, mi_page_t* page, mi_block_t* block);
|
||||||
|
|
||||||
uint8_t* _mi_segment_page_start(const mi_segment_t* segment, const mi_page_t* page, size_t* page_size); // page start for any page
|
uint8_t* _mi_segment_page_start(const mi_segment_t* segment, const mi_page_t* page, size_t* page_size); // page start for any page
|
||||||
void _mi_abandoned_reclaim_all(mi_heap_t* heap, mi_segments_tld_t* tld);
|
void _mi_abandoned_reclaim_all(mi_heap_t* heap, mi_segments_tld_t* tld);
|
||||||
|
31
src/alloc.c
31
src/alloc.c
@ -286,35 +286,6 @@ static void mi_padding_shrink(const mi_page_t* page, const mi_block_t* block, co
|
|||||||
// Free
|
// Free
|
||||||
// ------------------------------------------------------
|
// ------------------------------------------------------
|
||||||
|
|
||||||
// free huge block from another thread
|
|
||||||
static mi_decl_noinline void mi_free_huge_block_mt(mi_segment_t* segment, mi_page_t* page, mi_block_t* block) {
|
|
||||||
// huge page segments are always abandoned and can be freed immediately
|
|
||||||
mi_assert_internal(segment->kind==MI_SEGMENT_HUGE);
|
|
||||||
mi_assert_internal(segment == _mi_page_segment(page));
|
|
||||||
mi_assert_internal(mi_atomic_read_relaxed(&segment->thread_id)==0);
|
|
||||||
|
|
||||||
// claim it and free
|
|
||||||
mi_heap_t* heap = mi_get_default_heap();
|
|
||||||
// paranoia: if this it the last reference, the cas should always succeed
|
|
||||||
if (mi_atomic_cas_strong(&segment->thread_id, heap->thread_id, 0)) {
|
|
||||||
mi_block_set_next(page, block, page->free);
|
|
||||||
page->free = block;
|
|
||||||
page->used--;
|
|
||||||
page->is_zero = false;
|
|
||||||
mi_assert(page->used == 0);
|
|
||||||
mi_tld_t* tld = heap->tld;
|
|
||||||
const size_t bsize = mi_page_block_size(page);
|
|
||||||
if (bsize <= MI_LARGE_OBJ_SIZE_MAX) {
|
|
||||||
mi_assert_internal(false);
|
|
||||||
_mi_stat_decrease(&tld->stats.large, bsize);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
_mi_stat_decrease(&tld->stats.huge, bsize);
|
|
||||||
}
|
|
||||||
_mi_segment_page_free(page, true, &tld->segments);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// multi-threaded free
|
// multi-threaded free
|
||||||
static mi_decl_noinline void _mi_free_block_mt(mi_page_t* page, mi_block_t* block)
|
static mi_decl_noinline void _mi_free_block_mt(mi_page_t* page, mi_block_t* block)
|
||||||
{
|
{
|
||||||
@ -329,7 +300,7 @@ static mi_decl_noinline void _mi_free_block_mt(mi_page_t* page, mi_block_t* bloc
|
|||||||
// huge page segments are always abandoned and can be freed immediately
|
// huge page segments are always abandoned and can be freed immediately
|
||||||
mi_segment_t* segment = _mi_page_segment(page);
|
mi_segment_t* segment = _mi_page_segment(page);
|
||||||
if (segment->kind==MI_SEGMENT_HUGE) {
|
if (segment->kind==MI_SEGMENT_HUGE) {
|
||||||
mi_free_huge_block_mt(segment, page, block);
|
_mi_segment_huge_page_free(segment, page, block);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -166,8 +166,8 @@ static bool mi_segment_is_valid(mi_segment_t* segment, mi_segments_tld_t* tld) {
|
|||||||
mi_slice_t* last = &segment->slices[maxindex];
|
mi_slice_t* last = &segment->slices[maxindex];
|
||||||
mi_assert_internal((uint8_t*)slice == (uint8_t*)last - last->slice_offset);
|
mi_assert_internal((uint8_t*)slice == (uint8_t*)last - last->slice_offset);
|
||||||
mi_assert_internal(slice == last || last->slice_count == 0 );
|
mi_assert_internal(slice == last || last->slice_count == 0 );
|
||||||
mi_assert_internal(last->xblock_size == 0);
|
mi_assert_internal(last->xblock_size == 0 || (segment->kind==MI_SEGMENT_HUGE && last->xblock_size==1));
|
||||||
if (segment->kind == MI_SEGMENT_NORMAL && segment->thread_id != 0) { // segment is not huge or abandonded
|
if (segment->kind != MI_SEGMENT_HUGE && segment->thread_id != 0) { // segment is not huge or abandonded
|
||||||
sq = mi_span_queue_for(slice->slice_count,tld);
|
sq = mi_span_queue_for(slice->slice_count,tld);
|
||||||
mi_assert_internal(mi_span_queue_contains(sq,slice));
|
mi_assert_internal(mi_span_queue_contains(sq,slice));
|
||||||
}
|
}
|
||||||
@ -525,8 +525,10 @@ static mi_slice_t* mi_segment_span_free_coalesce(mi_slice_t* slice, mi_segments_
|
|||||||
|
|
||||||
// for huge pages, just mark as free but don't add to the queues
|
// for huge pages, just mark as free but don't add to the queues
|
||||||
if (segment->kind == MI_SEGMENT_HUGE) {
|
if (segment->kind == MI_SEGMENT_HUGE) {
|
||||||
mi_assert_internal(segment->used == 0);
|
mi_assert_internal(segment->used == 1); // decreased right after this call in `mi_segment_page_clear`
|
||||||
slice->xblock_size = 0; // mark as free anyways
|
slice->xblock_size = 0; // mark as free anyways
|
||||||
|
// we should mark the last slice `xblock_size=0` now to maintain invariants but we skip it to
|
||||||
|
// avoid a possible cache miss (and the segment is about to be freed)
|
||||||
return slice;
|
return slice;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1022,8 +1024,8 @@ static void mi_segment_abandon(mi_segment_t* segment, mi_segments_tld_t* tld) {
|
|||||||
slice = slice + slice->slice_count;
|
slice = slice + slice->slice_count;
|
||||||
}
|
}
|
||||||
|
|
||||||
// perform delayed decommits instead
|
// perform delayed decommits
|
||||||
mi_segment_delayed_decommit(segment, mi_option_is_enabled(mi_option_abandoned_page_reset), tld->stats);
|
mi_segment_delayed_decommit(segment, mi_option_is_enabled(mi_option_abandoned_page_reset) /* force? */, tld->stats);
|
||||||
|
|
||||||
// all pages in the segment are abandoned; add it to the abandoned list
|
// all pages in the segment are abandoned; add it to the abandoned list
|
||||||
_mi_stat_increase(&tld->stats->segments_abandoned, 1);
|
_mi_stat_increase(&tld->stats->segments_abandoned, 1);
|
||||||
@ -1297,6 +1299,7 @@ static mi_page_t* mi_segment_huge_page_alloc(size_t size, mi_segments_tld_t* tld
|
|||||||
// free huge block from another thread
|
// free huge block from another thread
|
||||||
void _mi_segment_huge_page_free(mi_segment_t* segment, mi_page_t* page, mi_block_t* block) {
|
void _mi_segment_huge_page_free(mi_segment_t* segment, mi_page_t* page, mi_block_t* block) {
|
||||||
// huge page segments are always abandoned and can be freed immediately by any thread
|
// huge page segments are always abandoned and can be freed immediately by any thread
|
||||||
|
mi_assert_internal(segment->kind==MI_SEGMENT_HUGE);
|
||||||
mi_assert_internal(segment == _mi_page_segment(page));
|
mi_assert_internal(segment == _mi_page_segment(page));
|
||||||
mi_assert_internal(mi_atomic_read_relaxed(&segment->thread_id)==0);
|
mi_assert_internal(mi_atomic_read_relaxed(&segment->thread_id)==0);
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user