Mono Continuations - после хранения память увеличивается () - PullRequest
1 голос
/ 26 августа 2009

Вот продолжение продолжения Mono Continuations_store (...). Из приведенного ниже кода видно, что store () следует следующим двум ветвям:

  1. cont->saved_stack && num_bytes <= cont->stack_alloc_size
    • использовать память напрямую
  2. еще
    • gc освобождает использованную память и создает новую память.

Однако странно то, что, если я неоднократно использую continueation_store (), использование памяти увеличивается, пока на более позднем этапе не будет выполнена огромная и медленная операция GC. Кто-нибудь может объяснить, почему это происходит?

Спасибо

static int
continuation_store (MonoContinuation *cont, int state, MonoException **e)
{
    MonoLMF *lmf = mono_get_lmf ();
    gsize num_bytes;

    if (!cont->domain)
        *e =  mono_get_exception_argument ("cont", "Continuation not initialized");
    if (cont->domain != mono_domain_get () || cont->thread_id != GetCurrentThreadId ())
        *e = mono_get_exception_argument ("cont", "Continuation from another thread or domain");

    cont->lmf = lmf;
    cont->return_ip = __builtin_return_address (0);
    cont->return_sp = __builtin_frame_address (0);

    num_bytes = (char*)cont->top_sp - (char*)cont->return_sp;

    /*g_print ("store: %d bytes, sp: %p, ip: %p, lmf: %p\n", num_bytes, cont->return_sp, cont->return_ip, lmf);*/

    if (cont->saved_stack && num_bytes <= cont->stack_alloc_size) 
    {
        /* clear to avoid GC retention */
        if (num_bytes < cont->stack_used_size)
            memset ((char*)cont->saved_stack + num_bytes, 0, cont->stack_used_size - num_bytes);
    } 
    else 
    {
        tasklets_lock ();
        internal_init ();
        if (cont->saved_stack) {
            mono_g_hash_table_remove (keepalive_stacks, cont->saved_stack);
            mono_gc_free_fixed (cont->saved_stack);
        }
        cont->stack_used_size = num_bytes;
        cont->stack_alloc_size = num_bytes * 1.1;
        cont->saved_stack = mono_gc_alloc_fixed (cont->stack_alloc_size, NULL);
        mono_g_hash_table_insert (keepalive_stacks, cont->saved_stack, cont->saved_stack);
        tasklets_unlock ();
    }
    memcpy (cont->saved_stack, cont->return_sp, num_bytes);

    return state;
}

1 Ответ

1 голос
/ 19 февраля 2011

Обратите внимание, что вызов mono_gc_free_fixed ничего не делает в сборщике boehm по умолчанию: https://github.com/mono/mono/blob/master/mono/metadata/boehm-gc.c#L528

Ветвь else просто удаляет память из хеш-таблицы keepalive_stacks. Эта хеш-таблица хранит ссылку на выделенную память, поэтому она не является сборщиком мусора. Как только указатель на выделенную память будет удален из этой хеш-таблицы, он будет возвращен во время следующей сборки мусора. Эта коллекция запускается позднее, когда достигается определенный объем выделенной памяти.

...