int sgen_thread_handshake (BOOL suspend) { int count, result; SgenThreadInfo *info; int signum = suspend ? suspend_signal_num : restart_signal_num; MonoNativeThreadId me = mono_native_thread_id_get (); count = 0; mono_thread_info_current ()->client_info.suspend_done = TRUE; FOREACH_THREAD_SAFE (info) { if (mono_native_thread_id_equals (mono_thread_info_get_tid (info), me)) { continue; } info->client_info.suspend_done = FALSE; if (info->client_info.gc_disabled) continue; /*if (signum == suspend_signal_num && info->stop_count == global_stop_count) continue;*/ result = mono_threads_pthread_kill (info, signum); if (result == 0) { count++; } else { info->client_info.skip = 1; } } END_FOREACH_THREAD_SAFE sgen_wait_for_suspend_ack (count); SGEN_LOG (4, "%s handshake for %d threads\n", suspend ? "suspend" : "resume", count); return count; }
int mono_sgen_thread_handshake (BOOL suspend) { int count, result; SgenThreadInfo *info; int signum = suspend ? suspend_signal_num : restart_signal_num; MonoNativeThreadId me = mono_native_thread_id_get (); count = 0; FOREACH_THREAD_SAFE (info) { if (mono_native_thread_id_equals (mono_thread_info_get_tid (info), me)) { continue; } /*if (signum == suspend_signal_num && info->stop_count == global_stop_count) continue;*/ if (suspend) { g_assert (!info->doing_handshake); info->doing_handshake = TRUE; } else { g_assert (info->doing_handshake); info->doing_handshake = FALSE; } result = pthread_kill (mono_thread_info_get_tid (info), signum); if (result == 0) { count++; } else { info->skip = 1; } } END_FOREACH_THREAD_SAFE mono_sgen_wait_for_suspend_ack (count); return count; }
static MonoException* continuation_restore (MonoContinuation *cont, int state) { MonoLMF **lmf_addr = mono_get_lmf_addr (); MonoContinuationRestore restore_state = mono_tasklets_arch_restore (); if (!cont->domain || !cont->return_sp) return mono_get_exception_argument ("cont", "Continuation not initialized"); if (cont->domain != mono_domain_get () || !mono_native_thread_id_equals (cont->thread_id, mono_native_thread_id_get ())) return mono_get_exception_argument ("cont", "Continuation from another thread or domain"); /*g_print ("restore: %p, state: %d\n", cont, state);*/ *lmf_addr = cont->lmf; restore_state (cont, state, lmf_addr); g_assert_not_reached (); }
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"); return 0; } if (cont->domain != mono_domain_get () || !mono_native_thread_id_equals (cont->thread_id, mono_native_thread_id_get ())) { *e = mono_get_exception_argument ("cont", "Continuation from another thread or domain"); return 0; } cont->lmf = lmf; cont->return_ip = __builtin_extract_return_addr (__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); } cont->stack_used_size = num_bytes; } else { tasklets_lock (); internal_init (); if (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_ROOT_SOURCE_THREADING, "saved tasklet stack"); tasklets_unlock (); } memcpy (cont->saved_stack, cont->return_sp, num_bytes); return state; }