fix to avoid potential linear behavior in page collect

This commit is contained in:
daan 2019-08-14 07:46:38 -07:00
parent d3375f2bac
commit 6d11e59250
3 changed files with 30 additions and 26 deletions

View File

@ -77,7 +77,7 @@ void _mi_page_use_delayed_free(mi_page_t* page, mi_delayed_t delay);
size_t _mi_page_queue_append(mi_heap_t* heap, mi_page_queue_t* pq, mi_page_queue_t* append); size_t _mi_page_queue_append(mi_heap_t* heap, mi_page_queue_t* pq, mi_page_queue_t* append);
void _mi_deferred_free(mi_heap_t* heap, bool force); void _mi_deferred_free(mi_heap_t* heap, bool force);
void _mi_page_free_collect(mi_page_t* page); void _mi_page_free_collect(mi_page_t* page,bool force);
void _mi_page_reclaim(mi_heap_t* heap, mi_page_t* page); // callback from segments void _mi_page_reclaim(mi_heap_t* heap, mi_page_t* page); // callback from segments
size_t _mi_bin_size(uint8_t bin); // for stats size_t _mi_bin_size(uint8_t bin); // for stats

View File

@ -85,7 +85,7 @@ static bool mi_heap_page_collect(mi_heap_t* heap, mi_page_queue_t* pq, mi_page_t
UNUSED(arg2); UNUSED(arg2);
UNUSED(heap); UNUSED(heap);
mi_collect_t collect = *((mi_collect_t*)arg_collect); mi_collect_t collect = *((mi_collect_t*)arg_collect);
_mi_page_free_collect(page); _mi_page_free_collect(page, collect >= ABANDON);
if (mi_page_all_free(page)) { if (mi_page_all_free(page)) {
// no more used blocks, free the page. TODO: should we retire here and be less aggressive? // no more used blocks, free the page. TODO: should we retire here and be less aggressive?
_mi_page_free(page, pq, collect != NORMAL); _mi_page_free(page, pq, collect != NORMAL);
@ -428,7 +428,7 @@ static bool mi_heap_area_visit_blocks(const mi_heap_area_ex_t* xarea, mi_block_v
mi_assert(page != NULL); mi_assert(page != NULL);
if (page == NULL) return true; if (page == NULL) return true;
_mi_page_free_collect(page); _mi_page_free_collect(page,true);
mi_assert_internal(page->local_free == NULL); mi_assert_internal(page->local_free == NULL);
if (page->used == 0) return true; if (page->used == 0) return true;

View File

@ -137,7 +137,7 @@ void _mi_page_use_delayed_free(mi_page_t* page, mi_delayed_t delay ) {
// Note: The exchange must be done atomically as this is used right after // Note: The exchange must be done atomically as this is used right after
// moving to the full list in `mi_page_collect_ex` and we need to // moving to the full list in `mi_page_collect_ex` and we need to
// ensure that there was no race where the page became unfull just before the move. // ensure that there was no race where the page became unfull just before the move.
static void mi_page_thread_free_collect(mi_page_t* page) static void _mi_page_thread_free_collect(mi_page_t* page)
{ {
mi_block_t* head; mi_block_t* head;
mi_thread_free_t tfree; mi_thread_free_t tfree;
@ -152,47 +152,51 @@ static void mi_page_thread_free_collect(mi_page_t* page)
if (head == NULL) return; if (head == NULL) return;
// find the tail // find the tail
uint16_t count = 1; uintptr_t count = 1;
mi_block_t* tail = head; mi_block_t* tail = head;
mi_block_t* next; mi_block_t* next;
while ((next = mi_block_next(page,tail)) != NULL) { while ((next = mi_block_next(page,tail)) != NULL) {
count++; count++;
tail = next; tail = next;
} }
// and append the current local free list
// and prepend to the free list mi_block_set_next(page,tail, page->local_free);
mi_block_set_next(page,tail, page->free); page->local_free = head;
page->free = head;
// update counts now // update counts now
mi_atomic_subtract(&page->thread_freed, count); mi_atomic_subtract(&page->thread_freed, count);
page->used -= count; page->used -= count;
} }
void _mi_page_free_collect(mi_page_t* page) { void _mi_page_free_collect(mi_page_t* page, bool force) {
mi_assert_internal(page!=NULL); mi_assert_internal(page!=NULL);
//if (page->free != NULL) return; // avoid expensive append
// free the local free list // collect the thread free list
if (force || mi_tf_block(page->thread_free) != NULL) { // quick test to avoid an atomic operation
_mi_page_thread_free_collect(page);
}
// and the local free list
if (page->local_free != NULL) { if (page->local_free != NULL) {
if (mi_likely(page->free == NULL)) { if (mi_unlikely(page->free == NULL)) {
// usual case // usual case
page->free = page->local_free; page->free = page->local_free;
page->local_free = NULL;
} }
else { else if (force) {
mi_block_t* tail = page->free; // append -- only on shutdown (force) as this is a linear operation
mi_block_t* tail = page->local_free;
mi_block_t* next; mi_block_t* next;
while ((next = mi_block_next(page, tail)) != NULL) { while ((next = mi_block_next(page, tail)) != NULL) {
tail = next; tail = next;
} }
mi_block_set_next(page, tail, page->local_free); mi_block_set_next(page, tail, page->free);
} page->free = page->local_free;
page->local_free = NULL; page->local_free = NULL;
} }
// and the thread free list
if (mi_tf_block(page->thread_free) != NULL) { // quick test to avoid an atomic operation
mi_page_thread_free_collect(page);
} }
mi_assert_internal(!force || page->local_free == NULL);
} }
@ -205,7 +209,7 @@ void _mi_page_free_collect(mi_page_t* page) {
void _mi_page_reclaim(mi_heap_t* heap, mi_page_t* page) { void _mi_page_reclaim(mi_heap_t* heap, mi_page_t* page) {
mi_assert_expensive(mi_page_is_valid_init(page)); mi_assert_expensive(mi_page_is_valid_init(page));
mi_assert_internal(page->heap == NULL); mi_assert_internal(page->heap == NULL);
_mi_page_free_collect(page); _mi_page_free_collect(page,false);
mi_page_queue_t* pq = mi_page_queue(heap, page->block_size); mi_page_queue_t* pq = mi_page_queue(heap, page->block_size);
mi_page_queue_push(heap, pq, page); mi_page_queue_push(heap, pq, page);
mi_assert_expensive(_mi_page_is_valid(page)); mi_assert_expensive(_mi_page_is_valid(page));
@ -304,7 +308,7 @@ static void mi_page_to_full(mi_page_t* page, mi_page_queue_t* pq) {
if (mi_page_is_in_full(page)) return; if (mi_page_is_in_full(page)) return;
mi_page_queue_enqueue_from(&page->heap->pages[MI_BIN_FULL], pq, page); mi_page_queue_enqueue_from(&page->heap->pages[MI_BIN_FULL], pq, page);
mi_page_thread_free_collect(page); // try to collect right away in case another thread freed just before MI_USE_DELAYED_FREE was set _mi_page_free_collect(page,false); // try to collect right away in case another thread freed just before MI_USE_DELAYED_FREE was set
} }
@ -595,7 +599,7 @@ static mi_page_t* mi_page_queue_find_free_ex(mi_heap_t* heap, mi_page_queue_t* p
count++; count++;
// 0. collect freed blocks by us and other threads // 0. collect freed blocks by us and other threads
_mi_page_free_collect(page); _mi_page_free_collect(page,false);
// 1. if the page contains free blocks, we are done // 1. if the page contains free blocks, we are done
if (mi_page_immediate_available(page)) { if (mi_page_immediate_available(page)) {
@ -662,7 +666,7 @@ static inline mi_page_t* mi_find_free_page(mi_heap_t* heap, size_t size) {
mi_assert_internal(mi_page_immediate_available(page)); mi_assert_internal(mi_page_immediate_available(page));
} }
else { else {
_mi_page_free_collect(page); _mi_page_free_collect(page,false);
} }
if (mi_page_immediate_available(page)) { if (mi_page_immediate_available(page)) {
return page; // fast path return page; // fast path