fix to avoid potential linear behavior in page collect
This commit is contained in:
parent
d3375f2bac
commit
6d11e59250
@ -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
|
||||||
|
@ -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;
|
||||||
|
|
||||||
|
50
src/page.c
50
src/page.c
@ -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
|
||||||
|
Loading…
Reference in New Issue
Block a user