/* * threads.c: Thread support internal calls * * Author: * Dick Porter (dick@ximian.com) * Paolo Molaro (lupus@ximian.com) * Patrik Torstensson (patrik.torstensson@labs2.com) * * Copyright 2001-2003 Ximian, Inc (http://www.ximian.com) * Copyright 2004-2009 Novell, Inc (http://www.novell.com) * Copyright 2011 Xamarin, Inc (http://www.xamarin.com) */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef HAVE_SIGNAL_H #include #endif #if defined(PLATFORM_ANDROID) && !defined(TARGET_ARM64) && !defined(TARGET_AMD64) #define USE_TKILL_ON_ANDROID 1 #endif #ifdef PLATFORM_ANDROID #include #ifdef USE_TKILL_ON_ANDROID extern int tkill (pid_t tid, int signal); #endif #endif /*#define THREAD_DEBUG(a) do { a; } while (0)*/ #define THREAD_DEBUG(a) /*#define THREAD_WAIT_DEBUG(a) do { a; } while (0)*/ #define THREAD_WAIT_DEBUG(a) /*#define LIBGC_DEBUG(a) do { a; } while (0)*/ #define LIBGC_DEBUG(a) #define SPIN_TRYLOCK(i) (InterlockedCompareExchange (&(i), 1, 0) == 0) #define SPIN_LOCK(i) do { \ if (SPIN_TRYLOCK (i)) \ break; \ } while (1) #define SPIN_UNLOCK(i) i = 0 #define LOCK_THREAD(thread) lock_thread((thread)) #define UNLOCK_THREAD(thread) unlock_thread((thread)) /* Provide this for systems with glib < 2.6 */ #ifndef G_GSIZE_FORMAT # if GLIB_SIZEOF_LONG == 8 # define G_GSIZE_FORMAT "lu" # else # define G_GSIZE_FORMAT "u" # endif #endif typedef struct { guint32 (*func)(void *); MonoThread *obj; MonoObject *delegate; void *start_arg; } StartInfo; typedef union { gint32 ival; gfloat fval; } IntFloatUnion; typedef union { gint64 ival; gdouble fval; } LongDoubleUnion; typedef struct _StaticDataFreeList StaticDataFreeList; struct _StaticDataFreeList { StaticDataFreeList *next; guint32 offset; guint32 size; }; typedef struct { int idx; int offset; StaticDataFreeList *freelist; } StaticDataInfo; /* Number of cached culture objects in the MonoThread->cached_culture_info array * (per-type): we use the first NUM entries for CultureInfo and the last for * UICultureInfo. So the size of the array is really NUM_CACHED_CULTURES * 2. */ #define NUM_CACHED_CULTURES 4 #define CULTURES_START_IDX 0 #define UICULTURES_START_IDX NUM_CACHED_CULTURES /* Controls access to the 'threads' hash table */ static void mono_threads_lock (void); static void mono_threads_unlock (void); static mono_mutex_t threads_mutex; /* Controls access to the 'joinable_threads' hash table */ #define joinable_threads_lock() mono_mutex_lock (&joinable_threads_mutex) #define joinable_threads_unlock() mono_mutex_unlock (&joinable_threads_mutex) static mono_mutex_t joinable_threads_mutex; /* Holds current status of static data heap */ static StaticDataInfo thread_static_info; static StaticDataInfo context_static_info; /* The hash of existing threads (key is thread ID, value is * MonoInternalThread*) that need joining before exit */ static MonoGHashTable *threads=NULL; /* List of app context GC handles. * Added to from ves_icall_System_Runtime_Remoting_Contexts_Context_RegisterContext (). */ static GHashTable *contexts = NULL; /* * Threads which are starting up and they are not in the 'threads' hash yet. * When handle_store is called for a thread, it will be removed from this hash table. * Protected by mono_threads_lock (). */ static MonoGHashTable *threads_starting_up = NULL; /* The TLS key that holds the MonoObject assigned to each thread */ static MonoNativeTlsKey current_object_key; /* Contains tids */ /* Protected by the threads lock */ static GHashTable *joinable_threads; static int joinable_thread_count; #ifdef MONO_HAVE_FAST_TLS /* we need to use both the Tls* functions and __thread because * the gc needs to see all the threads */ MONO_FAST_TLS_DECLARE(tls_current_object); #define SET_CURRENT_OBJECT(x) do { \ MONO_FAST_TLS_SET (tls_current_object, x); \ mono_native_tls_set_value (current_object_key, x); \ } while (FALSE) #define GET_CURRENT_OBJECT() ((MonoInternalThread*) MONO_FAST_TLS_GET (tls_current_object)) #else #define SET_CURRENT_OBJECT(x) mono_native_tls_set_value (current_object_key, x) #define GET_CURRENT_OBJECT() (MonoInternalThread*) mono_native_tls_get_value (current_object_key) #endif /* function called at thread start */ static MonoThreadStartCB mono_thread_start_cb = NULL; /* function called at thread attach */ static MonoThreadAttachCB mono_thread_attach_cb = NULL; /* function called at thread cleanup */ static MonoThreadCleanupFunc mono_thread_cleanup_fn = NULL; /* function called to notify the runtime about a pending exception on the current thread */ static MonoThreadNotifyPendingExcFunc mono_thread_notify_pending_exc_fn = NULL; /* The default stack size for each thread */ static guint32 default_stacksize = 0; #define default_stacksize_for_thread(thread) ((thread)->stack_size? (thread)->stack_size: default_stacksize) static void thread_adjust_static_data (MonoInternalThread *thread); static void context_adjust_static_data (MonoAppContext *ctx); static void mono_free_static_data (gpointer* static_data); static void mono_init_static_data_info (StaticDataInfo *static_data); static guint32 mono_alloc_static_data_slot (StaticDataInfo *static_data, guint32 size, guint32 align); static gboolean mono_thread_resume (MonoInternalThread* thread); static void abort_thread_internal (MonoInternalThread *thread, gboolean can_raise_exception, gboolean install_async_abort); static void suspend_thread_internal (MonoInternalThread *thread, gboolean interrupt); static void self_suspend_internal (MonoInternalThread *thread); static gboolean resume_thread_internal (MonoInternalThread *thread); static MonoException* mono_thread_execute_interruption (); static void ref_stack_destroy (gpointer rs); /* Spin lock for InterlockedXXX 64 bit functions */ #define mono_interlocked_lock() mono_mutex_lock (&interlocked_mutex) #define mono_interlocked_unlock() mono_mutex_unlock (&interlocked_mutex) static mono_mutex_t interlocked_mutex; /* global count of thread interruptions requested */ static gint32 thread_interruption_requested = 0; /* Event signaled when a thread changes its background mode */ static HANDLE background_change_event; static gboolean shutting_down = FALSE; static gint32 managed_thread_id_counter = 0; static void mono_threads_lock (void) { MONO_TRY_BLOCKING; mono_locks_acquire (&threads_mutex, ThreadsLock); MONO_FINISH_TRY_BLOCKING; } static void mono_threads_unlock (void) { mono_locks_release (&threads_mutex, ThreadsLock); } static guint32 get_next_managed_thread_id (void) { return InterlockedIncrement (&managed_thread_id_counter); } MonoNativeTlsKey mono_thread_get_tls_key (void) { return current_object_key; } gint32 mono_thread_get_tls_offset (void) { int offset; MONO_THREAD_VAR_OFFSET (tls_current_object,offset); return offset; } static inline MonoNativeThreadId thread_get_tid (MonoInternalThread *thread) { /* We store the tid as a guint64 to keep the object layout constant between platforms */ return MONO_UINT_TO_NATIVE_THREAD_ID (thread->tid); } /* handle_store() and handle_remove() manage the array of threads that * still need to be waited for when the main thread exits. * * If handle_store() returns FALSE the thread must not be started * because Mono is shutting down. */ static gboolean handle_store(MonoThread *thread, gboolean force_attach) { mono_threads_lock (); THREAD_DEBUG (g_message ("%s: thread %p ID %"G_GSIZE_FORMAT, __func__, thread, (gsize)thread->internal_thread->tid)); if (threads_starting_up) mono_g_hash_table_remove (threads_starting_up, thread); if (shutting_down && !force_attach) { mono_threads_unlock (); return FALSE; } if(threads==NULL) { MONO_GC_REGISTER_ROOT_FIXED (threads, MONO_ROOT_SOURCE_THREADING, "threads table"); threads=mono_g_hash_table_new_type (NULL, NULL, MONO_HASH_VALUE_GC, MONO_ROOT_SOURCE_THREADING, "threads table"); } /* We don't need to duplicate thread->handle, because it is * only closed when the thread object is finalized by the GC. */ g_assert (thread->internal_thread); mono_g_hash_table_insert(threads, (gpointer)(gsize)(thread->internal_thread->tid), thread->internal_thread); mono_threads_unlock (); return TRUE; } static gboolean handle_remove(MonoInternalThread *thread) { gboolean ret; gsize tid = thread->tid; THREAD_DEBUG (g_message ("%s: thread ID %"G_GSIZE_FORMAT, __func__, tid)); mono_threads_lock (); if (threads) { /* We have to check whether the thread object for the * tid is still the same in the table because the * thread might have been destroyed and the tid reused * in the meantime, in which case the tid would be in * the table, but with another thread object. */ if (mono_g_hash_table_lookup (threads, (gpointer)tid) == thread) { mono_g_hash_table_remove (threads, (gpointer)tid); ret = TRUE; } else { ret = FALSE; } } else ret = FALSE; mono_threads_unlock (); /* Don't close the handle here, wait for the object finalizer * to do it. Otherwise, the following race condition applies: * * 1) Thread exits (and handle_remove() closes the handle) * * 2) Some other handle is reassigned the same slot * * 3) Another thread tries to join the first thread, and * blocks waiting for the reassigned handle to be signalled * (which might never happen). This is possible, because the * thread calling Join() still has a reference to the first * thread's object. */ return ret; } static void ensure_synch_cs_set (MonoInternalThread *thread) { mono_mutex_t *synch_cs; if (thread->synch_cs != NULL) { return; } synch_cs = g_new0 (mono_mutex_t, 1); mono_mutex_init_recursive (synch_cs); if (InterlockedCompareExchangePointer ((gpointer *)&thread->synch_cs, synch_cs, NULL) != NULL) { /* Another thread must have installed this CS */ mono_mutex_destroy (synch_cs); g_free (synch_cs); } } static inline void lock_thread (MonoInternalThread *thread) { if (!thread->synch_cs) ensure_synch_cs_set (thread); g_assert (thread->synch_cs); MONO_TRY_BLOCKING; mono_mutex_lock (thread->synch_cs); MONO_FINISH_TRY_BLOCKING; } static inline void unlock_thread (MonoInternalThread *thread) { mono_mutex_unlock (thread->synch_cs); } /* * NOTE: this function can be called also for threads different from the current one: * make sure no code called from it will ever assume it is run on the thread that is * getting cleaned up. */ static void thread_cleanup (MonoInternalThread *thread) { g_assert (thread != NULL); if (thread->abort_state_handle) { mono_gchandle_free (thread->abort_state_handle); thread->abort_state_handle = 0; } thread->abort_exc = NULL; thread->current_appcontext = NULL; /* * This is necessary because otherwise we might have * cross-domain references which will not get cleaned up when * the target domain is unloaded. */ if (thread->cached_culture_info) { int i; for (i = 0; i < NUM_CACHED_CULTURES * 2; ++i) mono_array_set (thread->cached_culture_info, MonoObject*, i, NULL); } /* * thread->synch_cs can be NULL if this was called after * ves_icall_System_Threading_InternalThread_Thread_free_internal. * This can happen only during shutdown. * The shutting_down flag is not always set, so we can't assert on it. */ if (thread->synch_cs) LOCK_THREAD (thread); thread->state |= ThreadState_Stopped; thread->state &= ~ThreadState_Background; if (thread->synch_cs) UNLOCK_THREAD (thread); /* An interruption request has leaked to cleanup. Adjust the global counter. This can happen is the abort source thread finds the abortee (this) thread in unmanaged code. If this thread never trips back to managed code or check the local flag it will be left set and positively unbalance the global counter. Leaving the counter unbalanced will cause a performance degradation since all threads will now keep checking their local flags all the time. */ if (InterlockedExchange (&thread->interruption_requested, 0)) InterlockedDecrement (&thread_interruption_requested); /* if the thread is not in the hash it has been removed already */ if (!handle_remove (thread)) { if (thread == mono_thread_internal_current ()) { mono_domain_unset (); mono_memory_barrier (); } /* This needs to be called even if handle_remove () fails */ if (mono_thread_cleanup_fn) mono_thread_cleanup_fn (thread_get_tid (thread)); return; } mono_release_type_locks (thread); mono_profiler_thread_end (thread->tid); if (thread == mono_thread_internal_current ()) { /* * This will signal async signal handlers that the thread has exited. * The profiler callback needs this to be set, so it cannot be done earlier. */ mono_domain_unset (); mono_memory_barrier (); } if (thread == mono_thread_internal_current ()) mono_thread_pop_appdomain_ref (); thread->cached_culture_info = NULL; mono_free_static_data (thread->static_data); thread->static_data = NULL; ref_stack_destroy (thread->appdomain_refs); thread->appdomain_refs = NULL; if (mono_thread_cleanup_fn) mono_thread_cleanup_fn (thread_get_tid (thread)); if (mono_gc_is_moving ()) { MONO_GC_UNREGISTER_ROOT (thread->thread_pinning_ref); thread->thread_pinning_ref = NULL; } } /* * A special static data offset (guint32) consists of 3 parts: * * [0] 6-bit index into the array of chunks. * [6] 25-bit offset into the array. * [31] Bit indicating thread or context static. */ typedef union { struct { #if G_BYTE_ORDER != G_LITTLE_ENDIAN guint32 type : 1; guint32 offset : 25; guint32 index : 6; #else guint32 index : 6; guint32 offset : 25; guint32 type : 1; #endif } fields; guint32 raw; } SpecialStaticOffset; #define SPECIAL_STATIC_OFFSET_TYPE_THREAD 0 #define SPECIAL_STATIC_OFFSET_TYPE_CONTEXT 1 #define MAKE_SPECIAL_STATIC_OFFSET(index, offset, type) \ ((SpecialStaticOffset) { .fields = { (index), (offset), (type) } }.raw) #define ACCESS_SPECIAL_STATIC_OFFSET(x,f) \ (((SpecialStaticOffset *) &(x))->fields.f) static gpointer get_thread_static_data (MonoInternalThread *thread, guint32 offset) { g_assert (ACCESS_SPECIAL_STATIC_OFFSET (offset, type) == SPECIAL_STATIC_OFFSET_TYPE_THREAD); int idx = ACCESS_SPECIAL_STATIC_OFFSET (offset, index); int off = ACCESS_SPECIAL_STATIC_OFFSET (offset, offset); return ((char *) thread->static_data [idx]) + off; } static gpointer get_context_static_data (MonoAppContext *ctx, guint32 offset) { g_assert (ACCESS_SPECIAL_STATIC_OFFSET (offset, type) == SPECIAL_STATIC_OFFSET_TYPE_CONTEXT); int idx = ACCESS_SPECIAL_STATIC_OFFSET (offset, index); int off = ACCESS_SPECIAL_STATIC_OFFSET (offset, offset); return ((char *) ctx->static_data [idx]) + off; } static MonoThread** get_current_thread_ptr_for_domain (MonoDomain *domain, MonoInternalThread *thread) { static MonoClassField *current_thread_field = NULL; guint32 offset; if (!current_thread_field) { current_thread_field = mono_class_get_field_from_name (mono_defaults.thread_class, "current_thread"); g_assert (current_thread_field); } mono_class_vtable (domain, mono_defaults.thread_class); mono_domain_lock (domain); offset = GPOINTER_TO_UINT (g_hash_table_lookup (domain->special_static_fields, current_thread_field)); mono_domain_unlock (domain); g_assert (offset); return get_thread_static_data (thread, offset); } static void set_current_thread_for_domain (MonoDomain *domain, MonoInternalThread *thread, MonoThread *current) { MonoThread **current_thread_ptr = get_current_thread_ptr_for_domain (domain, thread); g_assert (current->obj.vtable->domain == domain); g_assert (!*current_thread_ptr); *current_thread_ptr = current; } static MonoThread* create_thread_object (MonoDomain *domain) { MonoVTable *vt = mono_class_vtable (domain, mono_defaults.thread_class); return (MonoThread*)mono_gc_alloc_mature (vt); } static MonoThread* new_thread_with_internal (MonoDomain *domain, MonoInternalThread *internal) { MonoThread *thread = create_thread_object (domain); MONO_OBJECT_SETREF (thread, internal_thread, internal); return thread; } static MonoInternalThread* create_internal_thread (void) { MonoInternalThread *thread; MonoVTable *vt; vt = mono_class_vtable (mono_get_root_domain (), mono_defaults.internal_thread_class); thread = (MonoInternalThread*)mono_gc_alloc_mature (vt); thread->synch_cs = g_new0 (mono_mutex_t, 1); mono_mutex_init_recursive (thread->synch_cs); thread->apartment_state = ThreadApartmentState_Unknown; thread->managed_id = get_next_managed_thread_id (); if (mono_gc_is_moving ()) { thread->thread_pinning_ref = thread; MONO_GC_REGISTER_ROOT_PINNING (thread->thread_pinning_ref, MONO_ROOT_SOURCE_THREADING, "thread pinning reference"); } return thread; } static void init_root_domain_thread (MonoInternalThread *thread, MonoThread *candidate) { MonoDomain *domain = mono_get_root_domain (); if (!candidate || candidate->obj.vtable->domain != domain) candidate = new_thread_with_internal (domain, thread); set_current_thread_for_domain (domain, thread, candidate); g_assert (!thread->root_domain_thread); MONO_OBJECT_SETREF (thread, root_domain_thread, candidate); } static guint32 WINAPI start_wrapper_internal(void *data) { MonoThreadInfo *info; StartInfo *start_info = (StartInfo *)data; guint32 (*start_func)(void *); void *start_arg; gsize tid; /* * We don't create a local to hold start_info->obj, so hopefully it won't get pinned during a * GC stack walk. */ MonoInternalThread *internal = start_info->obj->internal_thread; MonoObject *start_delegate = start_info->delegate; MonoDomain *domain = start_info->obj->obj.vtable->domain; THREAD_DEBUG (g_message ("%s: (%"G_GSIZE_FORMAT") Start wrapper", __func__, GetCurrentThreadId ())); /* We can be sure start_info->obj->tid and * start_info->obj->handle have been set, because the thread * was created suspended, and these values were set before the * thread resumed */ info = mono_thread_info_current (); g_assert (info); internal->thread_info = info; internal->small_id = info->small_id; tid = internal->tid; SET_CURRENT_OBJECT (internal); /* Every thread references the appdomain which created it */ mono_thread_push_appdomain_ref (domain); if (!mono_domain_set (domain, FALSE)) { /* No point in raising an appdomain_unloaded exception here */ /* FIXME: Cleanup here */ mono_thread_pop_appdomain_ref (); return 0; } start_func = start_info->func; start_arg = start_info->obj->start_obj; if (!start_arg) start_arg = start_info->start_arg; /* We have to do this here because mono_thread_new_init() requires that root_domain_thread is set up. */ thread_adjust_static_data (internal); init_root_domain_thread (internal, start_info->obj); /* This MUST be called before any managed code can be * executed, as it calls the callback function that (for the * jit) sets the lmf marker. */ mono_thread_new_init (tid, &tid, start_func); internal->stack_ptr = &tid; LIBGC_DEBUG (g_message ("%s: (%"G_GSIZE_FORMAT",%d) Setting thread stack to %p", __func__, GetCurrentThreadId (), getpid (), thread->stack_ptr)); THREAD_DEBUG (g_message ("%s: (%"G_GSIZE_FORMAT") Setting current_object_key to %p", __func__, GetCurrentThreadId (), internal)); /* On 2.0 profile (and higher), set explicitly since state might have been Unknown */ if (internal->apartment_state == ThreadApartmentState_Unknown) internal->apartment_state = ThreadApartmentState_MTA; mono_thread_init_apartment_state (); if(internal->start_notify!=NULL) { /* Let the thread that called Start() know we're * ready */ ReleaseSemaphore (internal->start_notify, 1, NULL); } g_free (start_info); THREAD_DEBUG (g_message ("%s: start_wrapper for %"G_GSIZE_FORMAT, __func__, internal->tid)); /* * Call this after calling start_notify, since the profiler callback might want * to lock the thread, and the lock is held by thread_start () which waits for * start_notify. */ mono_profiler_thread_start (tid); /* if the name was set before starting, we didn't invoke the profiler callback */ if (internal->name && (internal->flags & MONO_THREAD_FLAG_NAME_SET)) { char *tname = g_utf16_to_utf8 (internal->name, internal->name_len, NULL, NULL, NULL); mono_profiler_thread_name (internal->tid, tname); g_free (tname); } /* start_func is set only for unmanaged start functions */ if (start_func) { start_func (start_arg); } else { void *args [1]; g_assert (start_delegate != NULL); args [0] = start_arg; /* we may want to handle the exception here. See comment below on unhandled exceptions */ mono_runtime_delegate_invoke (start_delegate, args, NULL); } /* If the thread calls ExitThread at all, this remaining code * will not be executed, but the main thread will eventually * call thread_cleanup() on this thread's behalf. */ THREAD_DEBUG (g_message ("%s: (%"G_GSIZE_FORMAT") Start wrapper terminating", __func__, GetCurrentThreadId ())); /* Do any cleanup needed for apartment state. This * cannot be done in thread_cleanup since thread_cleanup could be * called for a thread other than the current thread. * mono_thread_cleanup_apartment_state cleans up apartment * for the current thead */ mono_thread_cleanup_apartment_state (); thread_cleanup (internal); internal->tid = 0; /* Remove the reference to the thread object in the TLS data, * so the thread object can be finalized. This won't be * reached if the thread threw an uncaught exception, so those * thread handles will stay referenced :-( (This is due to * missing support for scanning thread-specific data in the * Boehm GC - the io-layer keeps a GC-visible hash of pointers * to TLS data.) */ SET_CURRENT_OBJECT (NULL); return(0); } static guint32 WINAPI start_wrapper(void *data) { volatile int dummy; /* Avoid scanning the frames above this frame during a GC */ mono_gc_set_stack_end ((void*)&dummy); return start_wrapper_internal (data); } /* * create_thread: * * Common thread creation code. * LOCKING: Acquires the threads lock. */ static gboolean create_thread (MonoThread *thread, MonoInternalThread *internal, StartInfo *start_info, gboolean threadpool_thread, guint32 stack_size, gboolean throw_on_failure) { HANDLE thread_handle; MonoNativeThreadId tid; guint32 create_flags; /* * Join joinable threads to prevent running out of threads since the finalizer * thread might be blocked/backlogged. */ mono_threads_join_threads (); mono_threads_lock (); if (shutting_down) { g_free (start_info); mono_threads_unlock (); return FALSE; } if (threads_starting_up == NULL) { MONO_GC_REGISTER_ROOT_FIXED (threads_starting_up, MONO_ROOT_SOURCE_THREADING, "starting threads table"); threads_starting_up = mono_g_hash_table_new_type (NULL, NULL, MONO_HASH_KEY_VALUE_GC, MONO_ROOT_SOURCE_THREADING, "starting threads table"); } mono_g_hash_table_insert (threads_starting_up, thread, thread); mono_threads_unlock (); internal->start_notify = CreateSemaphore (NULL, 0, 0x7fffffff, NULL); if (!internal->start_notify) { mono_threads_lock (); mono_g_hash_table_remove (threads_starting_up, thread); mono_threads_unlock (); g_warning ("%s: CreateSemaphore error 0x%x", __func__, GetLastError ()); g_free (start_info); return FALSE; } if (stack_size == 0) stack_size = default_stacksize_for_thread (internal); /* Create suspended, so we can do some housekeeping before the thread * starts */ create_flags = CREATE_SUSPENDED; MONO_PREPARE_BLOCKING; thread_handle = mono_threads_create_thread ((LPTHREAD_START_ROUTINE)start_wrapper, start_info, stack_size, create_flags, &tid); MONO_FINISH_BLOCKING; if (thread_handle == NULL) { /* The thread couldn't be created, so throw an exception */ mono_threads_lock (); mono_g_hash_table_remove (threads_starting_up, thread); mono_threads_unlock (); g_free (start_info); if (throw_on_failure) mono_raise_exception (mono_get_exception_execution_engine ("Couldn't create thread")); else g_warning ("%s: CreateThread error 0x%x", __func__, GetLastError ()); return FALSE; } THREAD_DEBUG (g_message ("%s: Started thread ID %"G_GSIZE_FORMAT" (handle %p)", __func__, tid, thread_handle)); internal->handle = thread_handle; internal->tid = MONO_NATIVE_THREAD_ID_TO_UINT (tid); internal->threadpool_thread = threadpool_thread; if (threadpool_thread) mono_thread_set_state (internal, ThreadState_Background); THREAD_DEBUG (g_message ("%s: (%"G_GSIZE_FORMAT") Launching thread %p (%"G_GSIZE_FORMAT")", __func__, GetCurrentThreadId (), internal, (gsize)internal->tid)); /* Only store the handle when the thread is about to be * launched, to avoid the main thread deadlocking while trying * to clean up a thread that will never be signalled. */ if (!handle_store (thread, FALSE)) return FALSE; MONO_PREPARE_BLOCKING; mono_thread_info_resume (tid); MONO_FINISH_BLOCKING; if (internal->start_notify) { /* * Wait for the thread to set up its TLS data etc, so * theres no potential race condition if someone tries * to look up the data believing the thread has * started */ THREAD_DEBUG (g_message ("%s: (%"G_GSIZE_FORMAT") waiting for thread %p (%"G_GSIZE_FORMAT") to start", __func__, GetCurrentThreadId (), internal, (gsize)internal->tid)); MONO_PREPARE_BLOCKING; WaitForSingleObjectEx (internal->start_notify, INFINITE, FALSE); MONO_FINISH_BLOCKING; CloseHandle (internal->start_notify); internal->start_notify = NULL; } THREAD_DEBUG (g_message ("%s: (%"G_GSIZE_FORMAT") Done launching thread %p (%"G_GSIZE_FORMAT")", __func__, GetCurrentThreadId (), internal, (gsize)internal->tid)); return TRUE; } void mono_thread_new_init (intptr_t tid, gpointer stack_start, gpointer func) { if (mono_thread_start_cb) { mono_thread_start_cb (tid, stack_start, func); } } void mono_threads_set_default_stacksize (guint32 stacksize) { default_stacksize = stacksize; } guint32 mono_threads_get_default_stacksize (void) { return default_stacksize; } /* * mono_thread_create_internal: * * ARG should not be a GC reference. */ MonoInternalThread* mono_thread_create_internal (MonoDomain *domain, gpointer func, gpointer arg, gboolean threadpool_thread, guint32 stack_size) { MonoThread *thread; MonoInternalThread *internal; StartInfo *start_info; gboolean res; thread = create_thread_object (domain); internal = create_internal_thread (); MONO_OBJECT_SETREF (thread, internal_thread, internal); start_info = g_new0 (StartInfo, 1); start_info->func = func; start_info->obj = thread; start_info->start_arg = arg; res = create_thread (thread, internal, start_info, threadpool_thread, stack_size, TRUE); if (!res) return NULL; /* Check that the managed and unmanaged layout of MonoInternalThread matches */ if (mono_check_corlib_version () == NULL) g_assert (((char*)&internal->unused2 - (char*)internal) == mono_defaults.internal_thread_class->fields [mono_defaults.internal_thread_class->field.count - 1].offset); return internal; } void mono_thread_create (MonoDomain *domain, gpointer func, gpointer arg) { mono_thread_create_internal (domain, func, arg, FALSE, 0); } MonoThread * mono_thread_attach (MonoDomain *domain) { return mono_thread_attach_full (domain, FALSE); } MonoThread * mono_thread_attach_full (MonoDomain *domain, gboolean force_attach) { MonoThreadInfo *info; MonoInternalThread *thread; MonoThread *current_thread; HANDLE thread_handle; gsize tid; if ((thread = mono_thread_internal_current ())) { if (domain != mono_domain_get ()) mono_domain_set (domain, TRUE); /* Already attached */ return mono_thread_current (); } if (!mono_gc_register_thread (&domain)) { g_error ("Thread %"G_GSIZE_FORMAT" calling into managed code is not registered with the GC. On UNIX, this can be fixed by #include-ing before in the file containing the thread creation code.", GetCurrentThreadId ()); } thread = create_internal_thread (); thread_handle = mono_thread_info_open_handle (); g_assert (thread_handle); tid=GetCurrentThreadId (); thread->handle = thread_handle; thread->tid = tid; thread->stack_ptr = &tid; THREAD_DEBUG (g_message ("%s: Attached thread ID %"G_GSIZE_FORMAT" (handle %p)", __func__, tid, thread_handle)); info = mono_thread_info_current (); g_assert (info); thread->thread_info = info; thread->small_id = info->small_id; current_thread = new_thread_with_internal (domain, thread); if (!handle_store (current_thread, force_attach)) { /* Mono is shutting down, so just wait for the end */ for (;;) Sleep (10000); } THREAD_DEBUG (g_message ("%s: (%"G_GSIZE_FORMAT") Setting current_object_key to %p", __func__, GetCurrentThreadId (), thread)); SET_CURRENT_OBJECT (thread); mono_domain_set (domain, TRUE); thread_adjust_static_data (thread); init_root_domain_thread (thread, current_thread); if (domain != mono_get_root_domain ()) set_current_thread_for_domain (domain, thread, current_thread); if (mono_thread_attach_cb) { guint8 *staddr; size_t stsize; mono_thread_info_get_stack_bounds (&staddr, &stsize); if (staddr == NULL) mono_thread_attach_cb (tid, &tid); else mono_thread_attach_cb (tid, staddr + stsize); } // FIXME: Need a separate callback mono_profiler_thread_start (tid); return current_thread; } void mono_thread_detach_internal (MonoInternalThread *thread) { g_return_if_fail (thread != NULL); THREAD_DEBUG (g_message ("%s: mono_thread_detach for %p (%"G_GSIZE_FORMAT")", __func__, thread, (gsize)thread->tid)); thread_cleanup (thread); SET_CURRENT_OBJECT (NULL); mono_domain_unset (); /* Don't need to CloseHandle this thread, even though we took a * reference in mono_thread_attach (), because the GC will do it * when the Thread object is finalised. */ } void mono_thread_detach (MonoThread *thread) { if (thread) mono_thread_detach_internal (thread->internal_thread); } /* * mono_thread_detach_if_exiting: * * Detach the current thread from the runtime if it is exiting, i.e. it is running pthread dtors. * This should be used at the end of embedding code which calls into managed code, and which * can be called from pthread dtors, like dealloc: implementations in objective-c. */ void mono_thread_detach_if_exiting (void) { if (mono_thread_info_is_exiting ()) { MonoInternalThread *thread; thread = mono_thread_internal_current (); if (thread) { mono_thread_detach_internal (thread); mono_thread_info_detach (); } } } void mono_thread_exit () { MonoInternalThread *thread = mono_thread_internal_current (); THREAD_DEBUG (g_message ("%s: mono_thread_exit for %p (%"G_GSIZE_FORMAT")", __func__, thread, (gsize)thread->tid)); thread_cleanup (thread); SET_CURRENT_OBJECT (NULL); mono_domain_unset (); /* we could add a callback here for embedders to use. */ if (mono_thread_get_main () && (thread == mono_thread_get_main ()->internal_thread)) exit (mono_environment_exitcode_get ()); mono_thread_info_exit (); } void ves_icall_System_Threading_Thread_ConstructInternalThread (MonoThread *this_obj) { MonoInternalThread *internal = create_internal_thread (); internal->state = ThreadState_Unstarted; InterlockedCompareExchangePointer ((gpointer)&this_obj->internal_thread, internal, NULL); } HANDLE ves_icall_System_Threading_Thread_Thread_internal (MonoThread *this_obj, MonoObject *start) { StartInfo *start_info; MonoInternalThread *internal; gboolean res; THREAD_DEBUG (g_message("%s: Trying to start a new thread: this (%p) start (%p)", __func__, this_obj, start)); if (!this_obj->internal_thread) ves_icall_System_Threading_Thread_ConstructInternalThread (this_obj); internal = this_obj->internal_thread; LOCK_THREAD (internal); if ((internal->state & ThreadState_Unstarted) == 0) { UNLOCK_THREAD (internal); mono_set_pending_exception (mono_get_exception_thread_state ("Thread has already been started.")); return NULL; } if ((internal->state & ThreadState_Aborted) != 0) { UNLOCK_THREAD (internal); return this_obj; } /* This is freed in start_wrapper */ start_info = g_new0 (StartInfo, 1); start_info->func = NULL; start_info->start_arg = NULL; start_info->delegate = start; start_info->obj = this_obj; g_assert (this_obj->obj.vtable->domain == mono_domain_get ()); res = create_thread (this_obj, internal, start_info, FALSE, 0, FALSE); if (!res) { UNLOCK_THREAD (internal); return NULL; } internal->state &= ~ThreadState_Unstarted; THREAD_DEBUG (g_message ("%s: Started thread ID %"G_GSIZE_FORMAT" (handle %p)", __func__, tid, thread)); UNLOCK_THREAD (internal); return internal->handle; } /* * This is called from the finalizer of the internal thread object. */ void ves_icall_System_Threading_InternalThread_Thread_free_internal (MonoInternalThread *this_obj, HANDLE thread) { THREAD_DEBUG (g_message ("%s: Closing thread %p, handle %p", __func__, this, thread)); /* * Since threads keep a reference to their thread object while running, by the time this function is called, * the thread has already exited/detached, i.e. thread_cleanup () has ran. The exception is during shutdown, * when thread_cleanup () can be called after this. */ if (thread) CloseHandle (thread); if (this_obj->synch_cs) { mono_mutex_t *synch_cs = this_obj->synch_cs; this_obj->synch_cs = NULL; mono_mutex_destroy (synch_cs); g_free (synch_cs); } if (this_obj->name) { void *name = this_obj->name; this_obj->name = NULL; g_free (name); } } void ves_icall_System_Threading_Thread_Sleep_internal(gint32 ms) { guint32 res; MonoInternalThread *thread = mono_thread_internal_current (); THREAD_DEBUG (g_message ("%s: Sleeping for %d ms", __func__, ms)); mono_thread_current_check_pending_interrupt (); while (TRUE) { mono_thread_set_state (thread, ThreadState_WaitSleepJoin); MONO_PREPARE_BLOCKING; res = SleepEx(ms,TRUE); MONO_FINISH_BLOCKING; mono_thread_clr_state (thread, ThreadState_WaitSleepJoin); if (res == WAIT_IO_COMPLETION) { /* we might have been interrupted */ MonoException* exc = mono_thread_execute_interruption (); if (exc) { mono_raise_exception (exc); } else { // FIXME: !INFINITE if (ms != INFINITE) break; } } else { break; } } } void ves_icall_System_Threading_Thread_SpinWait_nop (void) { } gint32 ves_icall_System_Threading_Thread_GetDomainID (void) { return mono_domain_get()->domain_id; } gboolean ves_icall_System_Threading_Thread_Yield (void) { return mono_thread_info_yield (); } /* * mono_thread_get_name: * * Return the name of the thread. NAME_LEN is set to the length of the name. * Return NULL if the thread has no name. The returned memory is owned by the * caller. */ gunichar2* mono_thread_get_name (MonoInternalThread *this_obj, guint32 *name_len) { gunichar2 *res; LOCK_THREAD (this_obj); if (!this_obj->name) { *name_len = 0; res = NULL; } else { *name_len = this_obj->name_len; res = g_new (gunichar2, this_obj->name_len); memcpy (res, this_obj->name, sizeof (gunichar2) * this_obj->name_len); } UNLOCK_THREAD (this_obj); return res; } MonoString* ves_icall_System_Threading_Thread_GetName_internal (MonoInternalThread *this_obj) { MonoString* str; LOCK_THREAD (this_obj); if (!this_obj->name) str = NULL; else str = mono_string_new_utf16 (mono_domain_get (), this_obj->name, this_obj->name_len); UNLOCK_THREAD (this_obj); return str; } void mono_thread_set_name_internal (MonoInternalThread *this_obj, MonoString *name, gboolean managed) { LOCK_THREAD (this_obj); if ((this_obj->flags & MONO_THREAD_FLAG_NAME_SET) && !this_obj->threadpool_thread) { UNLOCK_THREAD (this_obj); mono_raise_exception (mono_get_exception_invalid_operation ("Thread.Name can only be set once.")); return; } if (this_obj->name) { g_free (this_obj->name); this_obj->name_len = 0; } if (name) { this_obj->name = g_new (gunichar2, mono_string_length (name)); memcpy (this_obj->name, mono_string_chars (name), mono_string_length (name) * 2); this_obj->name_len = mono_string_length (name); } else this_obj->name = NULL; if (managed) this_obj->flags |= MONO_THREAD_FLAG_NAME_SET; UNLOCK_THREAD (this_obj); if (this_obj->name && this_obj->tid) { char *tname = mono_string_to_utf8 (name); mono_profiler_thread_name (this_obj->tid, tname); mono_thread_info_set_name (thread_get_tid (this_obj), tname); mono_free (tname); } } void ves_icall_System_Threading_Thread_SetName_internal (MonoInternalThread *this_obj, MonoString *name) { mono_thread_set_name_internal (this_obj, name, TRUE); } int ves_icall_System_Threading_Thread_GetPriority (MonoThread *this) { return ThreadPriority_Lowest; } void ves_icall_System_Threading_Thread_SetPriority (MonoThread *this_obj, int priority) { } /* If the array is already in the requested domain, we just return it, otherwise we return a copy in that domain. */ static MonoArray* byte_array_to_domain (MonoArray *arr, MonoDomain *domain) { MonoArray *copy; if (!arr) return NULL; if (mono_object_domain (arr) == domain) return arr; copy = mono_array_new (domain, mono_defaults.byte_class, arr->max_length); memmove (mono_array_addr (copy, guint8, 0), mono_array_addr (arr, guint8, 0), arr->max_length); return copy; } MonoArray* ves_icall_System_Threading_Thread_ByteArrayToRootDomain (MonoArray *arr) { return byte_array_to_domain (arr, mono_get_root_domain ()); } MonoArray* ves_icall_System_Threading_Thread_ByteArrayToCurrentDomain (MonoArray *arr) { return byte_array_to_domain (arr, mono_domain_get ()); } MonoThread * mono_thread_current (void) { MonoDomain *domain = mono_domain_get (); MonoInternalThread *internal = mono_thread_internal_current (); MonoThread **current_thread_ptr; g_assert (internal); current_thread_ptr = get_current_thread_ptr_for_domain (domain, internal); if (!*current_thread_ptr) { g_assert (domain != mono_get_root_domain ()); *current_thread_ptr = new_thread_with_internal (domain, internal); } return *current_thread_ptr; } MonoInternalThread* mono_thread_internal_current (void) { MonoInternalThread *res = GET_CURRENT_OBJECT (); THREAD_DEBUG (g_message ("%s: returning %p", __func__, res)); return res; } gboolean ves_icall_System_Threading_Thread_Join_internal(MonoThread *this_obj, int ms) { MonoInternalThread *thread = this_obj->internal_thread; HANDLE handle = thread->handle; MonoInternalThread *cur_thread = mono_thread_internal_current (); gboolean ret; mono_thread_current_check_pending_interrupt (); LOCK_THREAD (thread); if ((thread->state & ThreadState_Unstarted) != 0) { UNLOCK_THREAD (thread); mono_set_pending_exception (mono_get_exception_thread_state ("Thread has not been started.")); return FALSE; } UNLOCK_THREAD (thread); if(ms== -1) { ms=INFINITE; } THREAD_DEBUG (g_message ("%s: joining thread handle %p, %d ms", __func__, handle, ms)); mono_thread_set_state (cur_thread, ThreadState_WaitSleepJoin); MONO_PREPARE_BLOCKING; ret=WaitForSingleObjectEx (handle, ms, TRUE); MONO_FINISH_BLOCKING; mono_thread_clr_state (cur_thread, ThreadState_WaitSleepJoin); if(ret==WAIT_OBJECT_0) { THREAD_DEBUG (g_message ("%s: join successful", __func__)); return(TRUE); } THREAD_DEBUG (g_message ("%s: join failed", __func__)); return(FALSE); } static gint32 mono_wait_uninterrupted (MonoInternalThread *thread, gboolean multiple, guint32 numhandles, gpointer *handles, gboolean waitall, gint32 ms, gboolean alertable) { MonoException *exc; guint32 ret; gint64 start; gint32 diff_ms; gint32 wait = ms; start = (ms == -1) ? 0 : mono_100ns_ticks (); do { MONO_PREPARE_BLOCKING; if (multiple) ret = WaitForMultipleObjectsEx (numhandles, handles, waitall, wait, alertable); else ret = WaitForSingleObjectEx (handles [0], ms, alertable); MONO_FINISH_BLOCKING; if (ret != WAIT_IO_COMPLETION) break; exc = mono_thread_execute_interruption (); if (exc) mono_raise_exception (exc); if (ms == -1) continue; /* Re-calculate ms according to the time passed */ diff_ms = (gint32)((mono_100ns_ticks () - start) / 10000); if (diff_ms >= ms) { ret = WAIT_TIMEOUT; break; } wait = ms - diff_ms; } while (TRUE); return ret; } /* FIXME: exitContext isnt documented */ gboolean ves_icall_System_Threading_WaitHandle_WaitAll_internal(MonoArray *mono_handles, gint32 ms, gboolean exitContext) { HANDLE *handles; guint32 numhandles; guint32 ret; guint32 i; MonoObject *waitHandle; MonoInternalThread *thread = mono_thread_internal_current (); /* Do this WaitSleepJoin check before creating objects */ mono_thread_current_check_pending_interrupt (); /* We fail in managed if the array has more than 64 elements */ numhandles = (guint32)mono_array_length(mono_handles); handles = g_new0(HANDLE, numhandles); for(i = 0; i < numhandles; i++) { waitHandle = mono_array_get(mono_handles, MonoObject*, i); handles [i] = mono_wait_handle_get_handle ((MonoWaitHandle *) waitHandle); } if(ms== -1) { ms=INFINITE; } mono_thread_set_state (thread, ThreadState_WaitSleepJoin); ret = mono_wait_uninterrupted (thread, TRUE, numhandles, handles, TRUE, ms, TRUE); mono_thread_clr_state (thread, ThreadState_WaitSleepJoin); g_free(handles); if(ret==WAIT_FAILED) { THREAD_WAIT_DEBUG (g_message ("%s: (%"G_GSIZE_FORMAT") Wait failed", __func__, GetCurrentThreadId ())); return(FALSE); } else if(ret==WAIT_TIMEOUT) { THREAD_WAIT_DEBUG (g_message ("%s: (%"G_GSIZE_FORMAT") Wait timed out", __func__, GetCurrentThreadId ())); return(FALSE); } return(TRUE); } /* FIXME: exitContext isnt documented */ gint32 ves_icall_System_Threading_WaitHandle_WaitAny_internal(MonoArray *mono_handles, gint32 ms, gboolean exitContext) { HANDLE handles [MAXIMUM_WAIT_OBJECTS]; uintptr_t numhandles; guint32 ret; guint32 i; MonoObject *waitHandle; MonoInternalThread *thread = mono_thread_internal_current (); /* Do this WaitSleepJoin check before creating objects */ mono_thread_current_check_pending_interrupt (); numhandles = mono_array_length(mono_handles); if (numhandles > MAXIMUM_WAIT_OBJECTS) return WAIT_FAILED; for(i = 0; i < numhandles; i++) { waitHandle = mono_array_get(mono_handles, MonoObject*, i); handles [i] = mono_wait_handle_get_handle ((MonoWaitHandle *) waitHandle); } if(ms== -1) { ms=INFINITE; } mono_thread_set_state (thread, ThreadState_WaitSleepJoin); ret = mono_wait_uninterrupted (thread, TRUE, numhandles, handles, FALSE, ms, TRUE); mono_thread_clr_state (thread, ThreadState_WaitSleepJoin); THREAD_WAIT_DEBUG (g_message ("%s: (%"G_GSIZE_FORMAT") returning %d", __func__, GetCurrentThreadId (), ret)); /* * These need to be here. See MSDN dos on WaitForMultipleObjects. */ if (ret >= WAIT_OBJECT_0 && ret <= WAIT_OBJECT_0 + numhandles - 1) { return ret - WAIT_OBJECT_0; } else if (ret >= WAIT_ABANDONED_0 && ret <= WAIT_ABANDONED_0 + numhandles - 1) { return ret - WAIT_ABANDONED_0; } else { return ret; } } /* FIXME: exitContext isnt documented */ gboolean ves_icall_System_Threading_WaitHandle_WaitOne_internal(MonoObject *this_obj, HANDLE handle, gint32 ms, gboolean exitContext) { guint32 ret; MonoInternalThread *thread = mono_thread_internal_current (); THREAD_WAIT_DEBUG (g_message ("%s: (%"G_GSIZE_FORMAT") waiting for %p, %d ms", __func__, GetCurrentThreadId (), handle, ms)); if(ms== -1) { ms=INFINITE; } mono_thread_current_check_pending_interrupt (); mono_thread_set_state (thread, ThreadState_WaitSleepJoin); ret = mono_wait_uninterrupted (thread, FALSE, 1, &handle, FALSE, ms, TRUE); mono_thread_clr_state (thread, ThreadState_WaitSleepJoin); if(ret==WAIT_FAILED) { THREAD_WAIT_DEBUG (g_message ("%s: (%"G_GSIZE_FORMAT") Wait failed", __func__, GetCurrentThreadId ())); return(FALSE); } else if(ret==WAIT_TIMEOUT) { THREAD_WAIT_DEBUG (g_message ("%s: (%"G_GSIZE_FORMAT") Wait timed out", __func__, GetCurrentThreadId ())); return(FALSE); } return(TRUE); } gboolean ves_icall_System_Threading_WaitHandle_SignalAndWait_Internal (HANDLE toSignal, HANDLE toWait, gint32 ms, gboolean exitContext) { guint32 ret; MonoInternalThread *thread = mono_thread_internal_current (); if (ms == -1) ms = INFINITE; mono_thread_current_check_pending_interrupt (); mono_thread_set_state (thread, ThreadState_WaitSleepJoin); MONO_PREPARE_BLOCKING; ret = SignalObjectAndWait (toSignal, toWait, ms, TRUE); MONO_FINISH_BLOCKING; mono_thread_clr_state (thread, ThreadState_WaitSleepJoin); return (!(ret == WAIT_TIMEOUT || ret == WAIT_IO_COMPLETION || ret == WAIT_FAILED)); } HANDLE ves_icall_System_Threading_Mutex_CreateMutex_internal (MonoBoolean owned, MonoString *name, MonoBoolean *created) { HANDLE mutex; *created = TRUE; if (name == NULL) { mutex = CreateMutex (NULL, owned, NULL); } else { mutex = CreateMutex (NULL, owned, mono_string_chars (name)); if (GetLastError () == ERROR_ALREADY_EXISTS) { *created = FALSE; } } return(mutex); } MonoBoolean ves_icall_System_Threading_Mutex_ReleaseMutex_internal (HANDLE handle ) { return(ReleaseMutex (handle)); } HANDLE ves_icall_System_Threading_Mutex_OpenMutex_internal (MonoString *name, gint32 rights, gint32 *error) { HANDLE ret; *error = ERROR_SUCCESS; ret = OpenMutex (rights, FALSE, mono_string_chars (name)); if (ret == NULL) { *error = GetLastError (); } return(ret); } HANDLE ves_icall_System_Threading_Semaphore_CreateSemaphore_internal (gint32 initialCount, gint32 maximumCount, MonoString *name, MonoBoolean *created) { HANDLE sem; *created = TRUE; if (name == NULL) { sem = CreateSemaphore (NULL, initialCount, maximumCount, NULL); } else { sem = CreateSemaphore (NULL, initialCount, maximumCount, mono_string_chars (name)); if (GetLastError () == ERROR_ALREADY_EXISTS) { *created = FALSE; } } return(sem); } gint32 ves_icall_System_Threading_Semaphore_ReleaseSemaphore_internal (HANDLE handle, gint32 releaseCount, MonoBoolean *fail) { gint32 prevcount; *fail = !ReleaseSemaphore (handle, releaseCount, &prevcount); return (prevcount); } HANDLE ves_icall_System_Threading_Semaphore_OpenSemaphore_internal (MonoString *name, gint32 rights, gint32 *error) { HANDLE ret; *error = ERROR_SUCCESS; ret = OpenSemaphore (rights, FALSE, mono_string_chars (name)); if (ret == NULL) { *error = GetLastError (); } return(ret); } HANDLE ves_icall_System_Threading_Events_CreateEvent_internal (MonoBoolean manual, MonoBoolean initial, MonoString *name, MonoBoolean *created) { HANDLE event; *created = TRUE; if (name == NULL) { event = CreateEvent (NULL, manual, initial, NULL); } else { event = CreateEvent (NULL, manual, initial, mono_string_chars (name)); if (GetLastError () == ERROR_ALREADY_EXISTS) { *created = FALSE; } } return(event); } gboolean ves_icall_System_Threading_Events_SetEvent_internal (HANDLE handle) { return (SetEvent(handle)); } gboolean ves_icall_System_Threading_Events_ResetEvent_internal (HANDLE handle) { return (ResetEvent(handle)); } void ves_icall_System_Threading_Events_CloseEvent_internal (HANDLE handle) { CloseHandle (handle); } HANDLE ves_icall_System_Threading_Events_OpenEvent_internal (MonoString *name, gint32 rights, gint32 *error) { HANDLE ret; *error = ERROR_SUCCESS; ret = OpenEvent (rights, FALSE, mono_string_chars (name)); if (ret == NULL) { *error = GetLastError (); } return(ret); } gint32 ves_icall_System_Threading_Interlocked_Increment_Int (gint32 *location) { return InterlockedIncrement (location); } gint64 ves_icall_System_Threading_Interlocked_Increment_Long (gint64 *location) { #if SIZEOF_VOID_P == 4 if (G_UNLIKELY ((size_t)location & 0x7)) { gint64 ret; mono_interlocked_lock (); (*location)++; ret = *location; mono_interlocked_unlock (); return ret; } #endif return InterlockedIncrement64 (location); } gint32 ves_icall_System_Threading_Interlocked_Decrement_Int (gint32 *location) { return InterlockedDecrement(location); } gint64 ves_icall_System_Threading_Interlocked_Decrement_Long (gint64 * location) { #if SIZEOF_VOID_P == 4 if (G_UNLIKELY ((size_t)location & 0x7)) { gint64 ret; mono_interlocked_lock (); (*location)--; ret = *location; mono_interlocked_unlock (); return ret; } #endif return InterlockedDecrement64 (location); } gint32 ves_icall_System_Threading_Interlocked_Exchange_Int (gint32 *location, gint32 value) { return InterlockedExchange(location, value); } MonoObject * ves_icall_System_Threading_Interlocked_Exchange_Object (MonoObject **location, MonoObject *value) { MonoObject *res; res = (MonoObject *) InterlockedExchangePointer((gpointer *) location, value); mono_gc_wbarrier_generic_nostore (location); return res; } gpointer ves_icall_System_Threading_Interlocked_Exchange_IntPtr (gpointer *location, gpointer value) { return InterlockedExchangePointer(location, value); } gfloat ves_icall_System_Threading_Interlocked_Exchange_Single (gfloat *location, gfloat value) { IntFloatUnion val, ret; val.fval = value; ret.ival = InterlockedExchange((gint32 *) location, val.ival); return ret.fval; } gint64 ves_icall_System_Threading_Interlocked_Exchange_Long (gint64 *location, gint64 value) { #if SIZEOF_VOID_P == 4 if (G_UNLIKELY ((size_t)location & 0x7)) { gint64 ret; mono_interlocked_lock (); ret = *location; *location = value; mono_interlocked_unlock (); return ret; } #endif return InterlockedExchange64 (location, value); } gdouble ves_icall_System_Threading_Interlocked_Exchange_Double (gdouble *location, gdouble value) { LongDoubleUnion val, ret; val.fval = value; ret.ival = (gint64)InterlockedExchange64((gint64 *) location, val.ival); return ret.fval; } gint32 ves_icall_System_Threading_Interlocked_CompareExchange_Int(gint32 *location, gint32 value, gint32 comparand) { return InterlockedCompareExchange(location, value, comparand); } gint32 ves_icall_System_Threading_Interlocked_CompareExchange_Int_Success(gint32 *location, gint32 value, gint32 comparand, MonoBoolean *success) { gint32 r = InterlockedCompareExchange(location, value, comparand); *success = r == comparand; return r; } MonoObject * ves_icall_System_Threading_Interlocked_CompareExchange_Object (MonoObject **location, MonoObject *value, MonoObject *comparand) { MonoObject *res; res = (MonoObject *) InterlockedCompareExchangePointer((gpointer *) location, value, comparand); mono_gc_wbarrier_generic_nostore (location); return res; } gpointer ves_icall_System_Threading_Interlocked_CompareExchange_IntPtr(gpointer *location, gpointer value, gpointer comparand) { return InterlockedCompareExchangePointer(location, value, comparand); } gfloat ves_icall_System_Threading_Interlocked_CompareExchange_Single (gfloat *location, gfloat value, gfloat comparand) { IntFloatUnion val, ret, cmp; val.fval = value; cmp.fval = comparand; ret.ival = InterlockedCompareExchange((gint32 *) location, val.ival, cmp.ival); return ret.fval; } gdouble ves_icall_System_Threading_Interlocked_CompareExchange_Double (gdouble *location, gdouble value, gdouble comparand) { #if SIZEOF_VOID_P == 8 LongDoubleUnion val, comp, ret; val.fval = value; comp.fval = comparand; ret.ival = (gint64)InterlockedCompareExchangePointer((gpointer *) location, (gpointer)val.ival, (gpointer)comp.ival); return ret.fval; #else gdouble old; mono_interlocked_lock (); old = *location; if (old == comparand) *location = value; mono_interlocked_unlock (); return old; #endif } gint64 ves_icall_System_Threading_Interlocked_CompareExchange_Long (gint64 *location, gint64 value, gint64 comparand) { #if SIZEOF_VOID_P == 4 if (G_UNLIKELY ((size_t)location & 0x7)) { gint64 old; mono_interlocked_lock (); old = *location; if (old == comparand) *location = value; mono_interlocked_unlock (); return old; } #endif return InterlockedCompareExchange64 (location, value, comparand); } MonoObject* ves_icall_System_Threading_Interlocked_CompareExchange_T (MonoObject **location, MonoObject *value, MonoObject *comparand) { MonoObject *res; res = InterlockedCompareExchangePointer ((gpointer *)location, value, comparand); mono_gc_wbarrier_generic_nostore (location); return res; } MonoObject* ves_icall_System_Threading_Interlocked_Exchange_T (MonoObject **location, MonoObject *value) { MonoObject *res; res = InterlockedExchangePointer ((gpointer *)location, value); mono_gc_wbarrier_generic_nostore (location); return res; } gint32 ves_icall_System_Threading_Interlocked_Add_Int (gint32 *location, gint32 value) { return InterlockedAdd (location, value); } gint64 ves_icall_System_Threading_Interlocked_Add_Long (gint64 *location, gint64 value) { #if SIZEOF_VOID_P == 4 if (G_UNLIKELY ((size_t)location & 0x7)) { gint64 ret; mono_interlocked_lock (); *location += value; ret = *location; mono_interlocked_unlock (); return ret; } #endif return InterlockedAdd64 (location, value); } gint64 ves_icall_System_Threading_Interlocked_Read_Long (gint64 *location) { #if SIZEOF_VOID_P == 4 if (G_UNLIKELY ((size_t)location & 0x7)) { gint64 ret; mono_interlocked_lock (); ret = *location; mono_interlocked_unlock (); return ret; } #endif return InterlockedRead64 (location); } void ves_icall_System_Threading_Thread_MemoryBarrier (void) { mono_memory_barrier (); } void ves_icall_System_Threading_Thread_ClrState (MonoInternalThread* this, guint32 state) { mono_thread_clr_state (this, state); if (state & ThreadState_Background) { /* If the thread changes the background mode, the main thread has to * be notified, since it has to rebuild the list of threads to * wait for. */ SetEvent (background_change_event); } } void ves_icall_System_Threading_Thread_SetState (MonoInternalThread* this, guint32 state) { mono_thread_set_state (this, state); if (state & ThreadState_Background) { /* If the thread changes the background mode, the main thread has to * be notified, since it has to rebuild the list of threads to * wait for. */ SetEvent (background_change_event); } } guint32 ves_icall_System_Threading_Thread_GetState (MonoInternalThread* this) { guint32 state; LOCK_THREAD (this); state = this->state; UNLOCK_THREAD (this); return state; } void ves_icall_System_Threading_Thread_Interrupt_internal (MonoThread *this_obj) { MonoInternalThread *current; gboolean throw; MonoInternalThread *thread = this_obj->internal_thread; LOCK_THREAD (thread); current = mono_thread_internal_current (); thread->thread_interrupt_requested = TRUE; throw = current != thread && (thread->state & ThreadState_WaitSleepJoin); UNLOCK_THREAD (thread); if (throw) { abort_thread_internal (thread, TRUE, FALSE); } } void mono_thread_current_check_pending_interrupt () { MonoInternalThread *thread = mono_thread_internal_current (); gboolean throw = FALSE; LOCK_THREAD (thread); if (thread->thread_interrupt_requested) { throw = TRUE; thread->thread_interrupt_requested = FALSE; } UNLOCK_THREAD (thread); if (throw) { mono_raise_exception (mono_get_exception_thread_interrupted ()); } } void ves_icall_System_Threading_Thread_Abort (MonoInternalThread *thread, MonoObject *state) { LOCK_THREAD (thread); if ((thread->state & ThreadState_AbortRequested) != 0 || (thread->state & ThreadState_StopRequested) != 0 || (thread->state & ThreadState_Stopped) != 0) { UNLOCK_THREAD (thread); return; } if ((thread->state & ThreadState_Unstarted) != 0) { thread->state |= ThreadState_Aborted; UNLOCK_THREAD (thread); return; } thread->state |= ThreadState_AbortRequested; if (thread->abort_state_handle) mono_gchandle_free (thread->abort_state_handle); if (state) { thread->abort_state_handle = mono_gchandle_new (state, FALSE); g_assert (thread->abort_state_handle); } else { thread->abort_state_handle = 0; } thread->abort_exc = NULL; UNLOCK_THREAD (thread); THREAD_DEBUG (g_message ("%s: (%"G_GSIZE_FORMAT") Abort requested for %p (%"G_GSIZE_FORMAT")", __func__, GetCurrentThreadId (), thread, (gsize)thread->tid)); /* During shutdown, we can't wait for other threads */ if (!shutting_down) /* Make sure the thread is awake */ mono_thread_resume (thread); abort_thread_internal (thread, TRUE, TRUE); } void ves_icall_System_Threading_Thread_ResetAbort (MonoThread *this_obj) { MonoInternalThread *thread = mono_thread_internal_current (); gboolean was_aborting; LOCK_THREAD (thread); was_aborting = thread->state & ThreadState_AbortRequested; thread->state &= ~ThreadState_AbortRequested; UNLOCK_THREAD (thread); if (!was_aborting) { const char *msg = "Unable to reset abort because no abort was requested"; mono_set_pending_exception (mono_get_exception_thread_state (msg)); return; } thread->abort_exc = NULL; if (thread->abort_state_handle) { mono_gchandle_free (thread->abort_state_handle); /* This is actually not necessary - the handle only counts if the exception is set */ thread->abort_state_handle = 0; } } void mono_thread_internal_reset_abort (MonoInternalThread *thread) { LOCK_THREAD (thread); thread->state &= ~ThreadState_AbortRequested; if (thread->abort_exc) { thread->abort_exc = NULL; if (thread->abort_state_handle) { mono_gchandle_free (thread->abort_state_handle); /* This is actually not necessary - the handle only counts if the exception is set */ thread->abort_state_handle = 0; } } UNLOCK_THREAD (thread); } MonoObject* ves_icall_System_Threading_Thread_GetAbortExceptionState (MonoThread *this_obj) { MonoInternalThread *thread = this_obj->internal_thread; MonoObject *state, *deserialized = NULL, *exc; MonoDomain *domain; if (!thread->abort_state_handle) return NULL; state = mono_gchandle_get_target (thread->abort_state_handle); g_assert (state); domain = mono_domain_get (); if (mono_object_domain (state) == domain) return state; deserialized = mono_object_xdomain_representation (state, domain, &exc); if (!deserialized) { MonoException *invalid_op_exc = mono_get_exception_invalid_operation ("Thread.ExceptionState cannot access an ExceptionState from a different AppDomain"); if (exc) MONO_OBJECT_SETREF (invalid_op_exc, inner_ex, exc); mono_set_pending_exception (invalid_op_exc); return NULL; } return deserialized; } static gboolean mono_thread_suspend (MonoInternalThread *thread) { LOCK_THREAD (thread); if ((thread->state & ThreadState_Unstarted) != 0 || (thread->state & ThreadState_Aborted) != 0 || (thread->state & ThreadState_Stopped) != 0) { UNLOCK_THREAD (thread); return FALSE; } if ((thread->state & ThreadState_Suspended) != 0 || (thread->state & ThreadState_SuspendRequested) != 0 || (thread->state & ThreadState_StopRequested) != 0) { UNLOCK_THREAD (thread); return TRUE; } thread->state |= ThreadState_SuspendRequested; UNLOCK_THREAD (thread); suspend_thread_internal (thread, FALSE); return TRUE; } void ves_icall_System_Threading_Thread_Suspend (MonoThread *this_obj) { if (!mono_thread_suspend (this_obj->internal_thread)) { mono_set_pending_exception (mono_get_exception_thread_state ("Thread has not been started, or is dead.")); return; } } static gboolean mono_thread_resume (MonoInternalThread *thread) { LOCK_THREAD (thread); if ((thread->state & ThreadState_SuspendRequested) != 0) { thread->state &= ~ThreadState_SuspendRequested; UNLOCK_THREAD (thread); return TRUE; } if ((thread->state & ThreadState_Suspended) == 0 || (thread->state & ThreadState_Unstarted) != 0 || (thread->state & ThreadState_Aborted) != 0 || (thread->state & ThreadState_Stopped) != 0) { UNLOCK_THREAD (thread); return FALSE; } return resume_thread_internal (thread); } void ves_icall_System_Threading_Thread_Resume (MonoThread *thread) { if (!thread->internal_thread || !mono_thread_resume (thread->internal_thread)) { mono_set_pending_exception (mono_get_exception_thread_state ("Thread has not been started, or is dead.")); return; } } static gboolean mono_threads_is_critical_method (MonoMethod *method) { switch (method->wrapper_type) { case MONO_WRAPPER_RUNTIME_INVOKE: case MONO_WRAPPER_XDOMAIN_INVOKE: case MONO_WRAPPER_XDOMAIN_DISPATCH: return TRUE; } return FALSE; } static gboolean find_wrapper (MonoMethod *m, gint no, gint ilo, gboolean managed, gpointer data) { if (managed) return TRUE; if (mono_threads_is_critical_method (m)) { *((gboolean*)data) = TRUE; return TRUE; } return FALSE; } static gboolean is_running_protected_wrapper (void) { gboolean found = FALSE; mono_stack_walk (find_wrapper, &found); return found; } void mono_thread_internal_stop (MonoInternalThread *thread) { LOCK_THREAD (thread); if ((thread->state & ThreadState_StopRequested) != 0 || (thread->state & ThreadState_Stopped) != 0) { UNLOCK_THREAD (thread); return; } /* Make sure the thread is awake */ mono_thread_resume (thread); thread->state |= ThreadState_StopRequested; thread->state &= ~ThreadState_AbortRequested; UNLOCK_THREAD (thread); abort_thread_internal (thread, TRUE, TRUE); } void mono_thread_stop (MonoThread *thread) { mono_thread_internal_stop (thread->internal_thread); } gint8 ves_icall_System_Threading_Thread_VolatileRead1 (void *ptr) { gint8 tmp; mono_atomic_load_acquire (tmp, gint8, (volatile gint8 *) ptr); return tmp; } gint16 ves_icall_System_Threading_Thread_VolatileRead2 (void *ptr) { gint16 tmp; mono_atomic_load_acquire (tmp, gint16, (volatile gint16 *) ptr); return tmp; } gint32 ves_icall_System_Threading_Thread_VolatileRead4 (void *ptr) { gint32 tmp; mono_atomic_load_acquire (tmp, gint32, (volatile gint32 *) ptr); return tmp; } gint64 ves_icall_System_Threading_Thread_VolatileRead8 (void *ptr) { gint64 tmp; mono_atomic_load_acquire (tmp, gint64, (volatile gint64 *) ptr); return tmp; } void * ves_icall_System_Threading_Thread_VolatileReadIntPtr (void *ptr) { volatile void *tmp; mono_atomic_load_acquire (tmp, volatile void *, (volatile void **) ptr); return (void *) tmp; } void * ves_icall_System_Threading_Thread_VolatileReadObject (void *ptr) { volatile MonoObject *tmp; mono_atomic_load_acquire (tmp, volatile MonoObject *, (volatile MonoObject **) ptr); return (MonoObject *) tmp; } double ves_icall_System_Threading_Thread_VolatileReadDouble (void *ptr) { double tmp; mono_atomic_load_acquire (tmp, double, (volatile double *) ptr); return tmp; } float ves_icall_System_Threading_Thread_VolatileReadFloat (void *ptr) { float tmp; mono_atomic_load_acquire (tmp, float, (volatile float *) ptr); return tmp; } gint8 ves_icall_System_Threading_Volatile_Read1 (void *ptr) { return InterlockedRead8 (ptr); } gint16 ves_icall_System_Threading_Volatile_Read2 (void *ptr) { return InterlockedRead16 (ptr); } gint32 ves_icall_System_Threading_Volatile_Read4 (void *ptr) { return InterlockedRead (ptr); } gint64 ves_icall_System_Threading_Volatile_Read8 (void *ptr) { #if SIZEOF_VOID_P == 4 if (G_UNLIKELY ((size_t)ptr & 0x7)) { gint64 val; mono_interlocked_lock (); val = *(gint64*)ptr; mono_interlocked_unlock (); return val; } #endif return InterlockedRead64 (ptr); } void * ves_icall_System_Threading_Volatile_ReadIntPtr (void *ptr) { return InterlockedReadPointer (ptr); } double ves_icall_System_Threading_Volatile_ReadDouble (void *ptr) { LongDoubleUnion u; #if SIZEOF_VOID_P == 4 if (G_UNLIKELY ((size_t)ptr & 0x7)) { double val; mono_interlocked_lock (); val = *(double*)ptr; mono_interlocked_unlock (); return val; } #endif u.ival = InterlockedRead64 (ptr); return u.fval; } float ves_icall_System_Threading_Volatile_ReadFloat (void *ptr) { IntFloatUnion u; u.ival = InterlockedRead (ptr); return u.fval; } MonoObject* ves_icall_System_Threading_Volatile_Read_T (void *ptr) { return InterlockedReadPointer (ptr); } void ves_icall_System_Threading_Thread_VolatileWrite1 (void *ptr, gint8 value) { mono_atomic_store_release ((volatile gint8 *) ptr, value); } void ves_icall_System_Threading_Thread_VolatileWrite2 (void *ptr, gint16 value) { mono_atomic_store_release ((volatile gint16 *) ptr, value); } void ves_icall_System_Threading_Thread_VolatileWrite4 (void *ptr, gint32 value) { mono_atomic_store_release ((volatile gint32 *) ptr, value); } void ves_icall_System_Threading_Thread_VolatileWrite8 (void *ptr, gint64 value) { mono_atomic_store_release ((volatile gint64 *) ptr, value); } void ves_icall_System_Threading_Thread_VolatileWriteIntPtr (void *ptr, void *value) { mono_atomic_store_release ((volatile void **) ptr, value); } void ves_icall_System_Threading_Thread_VolatileWriteObject (void *ptr, MonoObject *value) { mono_gc_wbarrier_generic_store_atomic (ptr, value); } void ves_icall_System_Threading_Thread_VolatileWriteDouble (void *ptr, double value) { mono_atomic_store_release ((volatile double *) ptr, value); } void ves_icall_System_Threading_Thread_VolatileWriteFloat (void *ptr, float value) { mono_atomic_store_release ((volatile float *) ptr, value); } void ves_icall_System_Threading_Volatile_Write1 (void *ptr, gint8 value) { InterlockedWrite8 (ptr, value); } void ves_icall_System_Threading_Volatile_Write2 (void *ptr, gint16 value) { InterlockedWrite16 (ptr, value); } void ves_icall_System_Threading_Volatile_Write4 (void *ptr, gint32 value) { InterlockedWrite (ptr, value); } void ves_icall_System_Threading_Volatile_Write8 (void *ptr, gint64 value) { #if SIZEOF_VOID_P == 4 if (G_UNLIKELY ((size_t)ptr & 0x7)) { mono_interlocked_lock (); *(gint64*)ptr = value; mono_interlocked_unlock (); return; } #endif InterlockedWrite64 (ptr, value); } void ves_icall_System_Threading_Volatile_WriteIntPtr (void *ptr, void *value) { InterlockedWritePointer (ptr, value); } void ves_icall_System_Threading_Volatile_WriteDouble (void *ptr, double value) { LongDoubleUnion u; #if SIZEOF_VOID_P == 4 if (G_UNLIKELY ((size_t)ptr & 0x7)) { mono_interlocked_lock (); *(double*)ptr = value; mono_interlocked_unlock (); return; } #endif u.fval = value; InterlockedWrite64 (ptr, u.ival); } void ves_icall_System_Threading_Volatile_WriteFloat (void *ptr, float value) { IntFloatUnion u; u.fval = value; InterlockedWrite (ptr, u.ival); } void ves_icall_System_Threading_Volatile_Write_T (void *ptr, MonoObject *value) { mono_gc_wbarrier_generic_store_atomic (ptr, value); } void ves_icall_System_Runtime_Remoting_Contexts_Context_RegisterContext (MonoAppContext *ctx) { mono_threads_lock (); //g_print ("Registering context %d in domain %d\n", ctx->context_id, ctx->domain_id); if (!contexts) contexts = g_hash_table_new (NULL, NULL); context_adjust_static_data (ctx); gpointer gch = GUINT_TO_POINTER (mono_gchandle_new_weakref (&ctx->obj, FALSE)); g_hash_table_insert (contexts, gch, gch); mono_threads_unlock (); mono_profiler_context_loaded (ctx); } void ves_icall_System_Runtime_Remoting_Contexts_Context_ReleaseContext (MonoAppContext *ctx) { /* * NOTE: Since finalizers are unreliable for the purposes of ensuring * cleanup in exceptional circumstances, we don't actually do any * cleanup work here. We instead do this when we iterate the `contexts` * hash table. The only purpose of this finalizer, at the moment, is to * notify the profiler. */ //g_print ("Releasing context %d in domain %d\n", ctx->context_id, ctx->domain_id); mono_profiler_context_unloaded (ctx); } void mono_thread_init_tls (void) { MONO_FAST_TLS_INIT (tls_current_object); mono_native_tls_alloc (¤t_object_key, NULL); } void mono_thread_init (MonoThreadStartCB start_cb, MonoThreadAttachCB attach_cb) { mono_mutex_init_recursive(&threads_mutex); mono_mutex_init_recursive(&interlocked_mutex); mono_mutex_init_recursive(&joinable_threads_mutex); background_change_event = CreateEvent (NULL, TRUE, FALSE, NULL); g_assert(background_change_event != NULL); mono_init_static_data_info (&thread_static_info); mono_init_static_data_info (&context_static_info); THREAD_DEBUG (g_message ("%s: Allocated current_object_key %d", __func__, current_object_key)); mono_thread_start_cb = start_cb; mono_thread_attach_cb = attach_cb; /* Get a pseudo handle to the current process. This is just a * kludge so that wapi can build a process handle if needed. * As a pseudo handle is returned, we don't need to clean * anything up. */ GetCurrentProcess (); } void mono_thread_cleanup (void) { #if !defined(HOST_WIN32) && !defined(RUN_IN_SUBTHREAD) MonoThreadInfo *info; /* The main thread must abandon any held mutexes (particularly * important for named mutexes as they are shared across * processes, see bug 74680.) This will happen when the * thread exits, but if it's not running in a subthread it * won't exit in time. */ info = mono_thread_info_current (); wapi_thread_handle_set_exited (info->handle, mono_environment_exitcode_get ()); #endif #if 0 /* This stuff needs more testing, it seems one of these * critical sections can be locked when mono_thread_cleanup is * called. */ mono_mutex_destroy (&threads_mutex); mono_mutex_destroy (&interlocked_mutex); mono_mutex_destroy (&delayed_free_table_mutex); mono_mutex_destroy (&small_id_mutex); CloseHandle (background_change_event); #endif mono_native_tls_free (current_object_key); } void mono_threads_install_cleanup (MonoThreadCleanupFunc func) { mono_thread_cleanup_fn = func; } void mono_thread_set_manage_callback (MonoThread *thread, MonoThreadManageCallback func) { thread->internal_thread->manage_callback = func; } void mono_threads_install_notify_pending_exc (MonoThreadNotifyPendingExcFunc func) { mono_thread_notify_pending_exc_fn = func; } G_GNUC_UNUSED static void print_tids (gpointer key, gpointer value, gpointer user) { /* GPOINTER_TO_UINT breaks horribly if sizeof(void *) > * sizeof(uint) and a cast to uint would overflow */ /* Older versions of glib don't have G_GSIZE_FORMAT, so just * print this as a pointer. */ g_message ("Waiting for: %p", key); } struct wait_data { HANDLE handles[MAXIMUM_WAIT_OBJECTS]; MonoInternalThread *threads[MAXIMUM_WAIT_OBJECTS]; guint32 num; }; static void wait_for_tids (struct wait_data *wait, guint32 timeout) { guint32 i, ret; THREAD_DEBUG (g_message("%s: %d threads to wait for in this batch", __func__, wait->num)); MONO_PREPARE_BLOCKING; ret=WaitForMultipleObjectsEx(wait->num, wait->handles, TRUE, timeout, TRUE); MONO_FINISH_BLOCKING; if(ret==WAIT_FAILED) { /* See the comment in build_wait_tids() */ THREAD_DEBUG (g_message ("%s: Wait failed", __func__)); return; } for(i=0; inum; i++) CloseHandle (wait->handles[i]); if (ret == WAIT_TIMEOUT) return; for(i=0; inum; i++) { gsize tid = wait->threads[i]->tid; /* * On !win32, when the thread handle becomes signalled, it just means the thread has exited user code, * it can still run io-layer etc. code. So wait for it to really exit. * FIXME: This won't join threads which are not in the joinable_hash yet. */ mono_thread_join ((gpointer)tid); mono_threads_lock (); if(mono_g_hash_table_lookup (threads, (gpointer)tid)!=NULL) { /* This thread must have been killed, because * it hasn't cleaned itself up. (It's just * possible that the thread exited before the * parent thread had a chance to store the * handle, and now there is another pointer to * the already-exited thread stored. In this * case, we'll just get two * mono_profiler_thread_end() calls for the * same thread.) */ mono_threads_unlock (); THREAD_DEBUG (g_message ("%s: cleaning up after thread %p (%"G_GSIZE_FORMAT")", __func__, wait->threads[i], tid)); thread_cleanup (wait->threads[i]); } else { mono_threads_unlock (); } } } static void wait_for_tids_or_state_change (struct wait_data *wait, guint32 timeout) { guint32 i, ret, count; THREAD_DEBUG (g_message("%s: %d threads to wait for in this batch", __func__, wait->num)); /* Add the thread state change event, so it wakes up if a thread changes * to background mode. */ count = wait->num; if (count < MAXIMUM_WAIT_OBJECTS) { wait->handles [count] = background_change_event; count++; } MONO_PREPARE_BLOCKING; ret=WaitForMultipleObjectsEx (count, wait->handles, FALSE, timeout, TRUE); MONO_FINISH_BLOCKING; if(ret==WAIT_FAILED) { /* See the comment in build_wait_tids() */ THREAD_DEBUG (g_message ("%s: Wait failed", __func__)); return; } for(i=0; inum; i++) CloseHandle (wait->handles[i]); if (ret == WAIT_TIMEOUT) return; if (ret < wait->num) { gsize tid = wait->threads[ret]->tid; mono_threads_lock (); if (mono_g_hash_table_lookup (threads, (gpointer)tid)!=NULL) { /* See comment in wait_for_tids about thread cleanup */ mono_threads_unlock (); THREAD_DEBUG (g_message ("%s: cleaning up after thread %"G_GSIZE_FORMAT, __func__, tid)); thread_cleanup (wait->threads [ret]); } else mono_threads_unlock (); } } static void build_wait_tids (gpointer key, gpointer value, gpointer user) { struct wait_data *wait=(struct wait_data *)user; if(wait->numstate & ThreadState_Background) { THREAD_DEBUG (g_message ("%s: ignoring background thread %"G_GSIZE_FORMAT, __func__, (gsize)thread->tid)); return; /* just leave, ignore */ } if (mono_gc_is_finalizer_internal_thread (thread)) { THREAD_DEBUG (g_message ("%s: ignoring finalizer thread %"G_GSIZE_FORMAT, __func__, (gsize)thread->tid)); return; } if (thread == mono_thread_internal_current ()) { THREAD_DEBUG (g_message ("%s: ignoring current thread %"G_GSIZE_FORMAT, __func__, (gsize)thread->tid)); return; } if (mono_thread_get_main () && (thread == mono_thread_get_main ()->internal_thread)) { THREAD_DEBUG (g_message ("%s: ignoring main thread %"G_GSIZE_FORMAT, __func__, (gsize)thread->tid)); return; } if (thread->flags & MONO_THREAD_FLAG_DONT_MANAGE) { THREAD_DEBUG (g_message ("%s: ignoring thread %" G_GSIZE_FORMAT "with DONT_MANAGE flag set.", __func__, (gsize)thread->tid)); return; } handle = mono_threads_open_thread_handle (thread->handle, thread_get_tid (thread)); if (handle == NULL) { THREAD_DEBUG (g_message ("%s: ignoring unopenable thread %"G_GSIZE_FORMAT, __func__, (gsize)thread->tid)); return; } THREAD_DEBUG (g_message ("%s: Invoking mono_thread_manage callback on thread %p", __func__, thread)); if ((thread->manage_callback == NULL) || (thread->manage_callback (thread->root_domain_thread) == TRUE)) { wait->handles[wait->num]=handle; wait->threads[wait->num]=thread; wait->num++; THREAD_DEBUG (g_message ("%s: adding thread %"G_GSIZE_FORMAT, __func__, (gsize)thread->tid)); } else { THREAD_DEBUG (g_message ("%s: ignoring (because of callback) thread %"G_GSIZE_FORMAT, __func__, (gsize)thread->tid)); } } else { /* Just ignore the rest, we can't do anything with * them yet */ } } static gboolean remove_and_abort_threads (gpointer key, gpointer value, gpointer user) { struct wait_data *wait=(struct wait_data *)user; gsize self = GetCurrentThreadId (); MonoInternalThread *thread = value; HANDLE handle; if (wait->num >= MAXIMUM_WAIT_OBJECTS) return FALSE; /* The finalizer thread is not a background thread */ if (thread->tid != self && (thread->state & ThreadState_Background) != 0 && !(thread->flags & MONO_THREAD_FLAG_DONT_MANAGE)) { handle = mono_threads_open_thread_handle (thread->handle, thread_get_tid (thread)); if (handle == NULL) return FALSE; /* printf ("A: %d\n", wait->num); */ wait->handles[wait->num]=thread->handle; wait->threads[wait->num]=thread; wait->num++; THREAD_DEBUG (g_print ("%s: Aborting id: %"G_GSIZE_FORMAT"\n", __func__, (gsize)thread->tid)); mono_thread_internal_stop (thread); return TRUE; } return (thread->tid != self && !mono_gc_is_finalizer_internal_thread (thread)); } /** * mono_threads_set_shutting_down: * * Is called by a thread that wants to shut down Mono. If the runtime is already * shutting down, the calling thread is suspended/stopped, and this function never * returns. */ void mono_threads_set_shutting_down (void) { MonoInternalThread *current_thread = mono_thread_internal_current (); mono_threads_lock (); if (shutting_down) { mono_threads_unlock (); /* Make sure we're properly suspended/stopped */ LOCK_THREAD (current_thread); if ((current_thread->state & ThreadState_SuspendRequested) || (current_thread->state & ThreadState_AbortRequested) || (current_thread->state & ThreadState_StopRequested)) { UNLOCK_THREAD (current_thread); mono_thread_execute_interruption (); } else { current_thread->state |= ThreadState_Stopped; UNLOCK_THREAD (current_thread); } /*since we're killing the thread, unset the current domain.*/ mono_domain_unset (); /* Wake up other threads potentially waiting for us */ mono_thread_info_exit (); } else { shutting_down = TRUE; /* Not really a background state change, but this will * interrupt the main thread if it is waiting for all * the other threads. */ SetEvent (background_change_event); mono_threads_unlock (); } } void mono_thread_manage (void) { struct wait_data wait_data; struct wait_data *wait = &wait_data; memset (wait, 0, sizeof (struct wait_data)); /* join each thread that's still running */ THREAD_DEBUG (g_message ("%s: Joining each running thread...", __func__)); mono_threads_lock (); if(threads==NULL) { THREAD_DEBUG (g_message("%s: No threads", __func__)); mono_threads_unlock (); return; } mono_threads_unlock (); do { mono_threads_lock (); if (shutting_down) { /* somebody else is shutting down */ mono_threads_unlock (); break; } THREAD_DEBUG (g_message ("%s: There are %d threads to join", __func__, mono_g_hash_table_size (threads)); mono_g_hash_table_foreach (threads, print_tids, NULL)); ResetEvent (background_change_event); wait->num=0; /*We must zero all InternalThread pointers to avoid making the GC unhappy.*/ memset (wait->threads, 0, MAXIMUM_WAIT_OBJECTS * SIZEOF_VOID_P); mono_g_hash_table_foreach (threads, build_wait_tids, wait); mono_threads_unlock (); if(wait->num>0) { /* Something to wait for */ wait_for_tids_or_state_change (wait, INFINITE); } THREAD_DEBUG (g_message ("%s: I have %d threads after waiting.", __func__, wait->num)); } while(wait->num>0); /* Mono is shutting down, so just wait for the end */ if (!mono_runtime_try_shutdown ()) { /*FIXME mono_thread_suspend probably should call mono_thread_execute_interruption when self interrupting. */ mono_thread_suspend (mono_thread_internal_current ()); mono_thread_execute_interruption (); } /* * Remove everything but the finalizer thread and self. * Also abort all the background threads * */ do { mono_threads_lock (); wait->num = 0; /*We must zero all InternalThread pointers to avoid making the GC unhappy.*/ memset (wait->threads, 0, MAXIMUM_WAIT_OBJECTS * SIZEOF_VOID_P); mono_g_hash_table_foreach_remove (threads, remove_and_abort_threads, wait); mono_threads_unlock (); THREAD_DEBUG (g_message ("%s: wait->num is now %d", __func__, wait->num)); if(wait->num>0) { /* Something to wait for */ wait_for_tids (wait, INFINITE); } } while (wait->num > 0); /* * give the subthreads a chance to really quit (this is mainly needed * to get correct user and system times from getrusage/wait/time(1)). * This could be removed if we avoid pthread_detach() and use pthread_join(). */ mono_thread_info_yield (); } static void terminate_thread (gpointer key, gpointer value, gpointer user) { MonoInternalThread *thread=(MonoInternalThread *)value; if(thread->tid != (gsize)user) { /*TerminateThread (thread->handle, -1);*/ } } void mono_thread_abort_all_other_threads (void) { gsize self = GetCurrentThreadId (); mono_threads_lock (); THREAD_DEBUG (g_message ("%s: There are %d threads to abort", __func__, mono_g_hash_table_size (threads)); mono_g_hash_table_foreach (threads, print_tids, NULL)); mono_g_hash_table_foreach (threads, terminate_thread, (gpointer)self); mono_threads_unlock (); } static void collect_threads_for_suspend (gpointer key, gpointer value, gpointer user_data) { MonoInternalThread *thread = (MonoInternalThread*)value; struct wait_data *wait = (struct wait_data*)user_data; HANDLE handle; /* * We try to exclude threads early, to avoid running into the MAXIMUM_WAIT_OBJECTS * limitation. * This needs no locking. */ if ((thread->state & ThreadState_Suspended) != 0 || (thread->state & ThreadState_Stopped) != 0) return; if (wait->numhandle, thread_get_tid (thread)); if (handle == NULL) return; wait->handles [wait->num] = handle; wait->threads [wait->num] = thread; wait->num++; } } /* * mono_thread_suspend_all_other_threads: * * Suspend all managed threads except the finalizer thread and this thread. It is * not possible to resume them later. */ void mono_thread_suspend_all_other_threads (void) { struct wait_data wait_data; struct wait_data *wait = &wait_data; int i; gsize self = GetCurrentThreadId (); guint32 eventidx = 0; gboolean starting, finished; memset (wait, 0, sizeof (struct wait_data)); /* * The other threads could be in an arbitrary state at this point, i.e. * they could be starting up, shutting down etc. This means that there could be * threads which are not even in the threads hash table yet. */ /* * First we set a barrier which will be checked by all threads before they * are added to the threads hash table, and they will exit if the flag is set. * This ensures that no threads could be added to the hash later. * We will use shutting_down as the barrier for now. */ g_assert (shutting_down); /* * We make multiple calls to WaitForMultipleObjects since: * - we can only wait for MAXIMUM_WAIT_OBJECTS threads * - some threads could exit without becoming suspended */ finished = FALSE; while (!finished) { /* * Make a copy of the hashtable since we can't do anything with * threads while threads_mutex is held. */ wait->num = 0; /*We must zero all InternalThread pointers to avoid making the GC unhappy.*/ memset (wait->threads, 0, MAXIMUM_WAIT_OBJECTS * SIZEOF_VOID_P); mono_threads_lock (); mono_g_hash_table_foreach (threads, collect_threads_for_suspend, wait); mono_threads_unlock (); eventidx = 0; /* Get the suspended events that we'll be waiting for */ for (i = 0; i < wait->num; ++i) { MonoInternalThread *thread = wait->threads [i]; if ((thread->tid == self) || mono_gc_is_finalizer_internal_thread (thread) || (thread->flags & MONO_THREAD_FLAG_DONT_MANAGE)) { //CloseHandle (wait->handles [i]); wait->threads [i] = NULL; /* ignore this thread in next loop */ continue; } LOCK_THREAD (thread); if ((thread->state & ThreadState_Suspended) != 0 || (thread->state & ThreadState_StopRequested) != 0 || (thread->state & ThreadState_Stopped) != 0) { UNLOCK_THREAD (thread); CloseHandle (wait->handles [i]); wait->threads [i] = NULL; /* ignore this thread in next loop */ continue; } ++eventidx; /* Convert abort requests into suspend requests */ if ((thread->state & ThreadState_AbortRequested) != 0) thread->state &= ~ThreadState_AbortRequested; thread->state |= ThreadState_SuspendRequested; UNLOCK_THREAD (thread); /* Signal the thread to suspend */ suspend_thread_internal (thread, TRUE); } if (eventidx <= 0) { /* * If there are threads which are starting up, we wait until they * are suspended when they try to register in the threads hash. * This is guaranteed to finish, since the threads which can create new * threads get suspended after a while. * FIXME: The finalizer thread can still create new threads. */ mono_threads_lock (); if (threads_starting_up) starting = mono_g_hash_table_size (threads_starting_up) > 0; else starting = FALSE; mono_threads_unlock (); if (starting) Sleep (100); else finished = TRUE; } } } static gboolean thread_dump_requested; static G_GNUC_UNUSED gboolean print_stack_frame_to_string (MonoStackFrameInfo *frame, MonoContext *ctx, gpointer data) { GString *p = (GString*)data; MonoMethod *method = NULL; if (frame->type == FRAME_TYPE_MANAGED) method = mono_jit_info_get_method (frame->ji); if (method) { gchar *location = mono_debug_print_stack_frame (method, frame->native_offset, frame->domain); g_string_append_printf (p, " %s\n", location); g_free (location); } else g_string_append_printf (p, " at <0x%05x>\n", frame->native_offset); return FALSE; } static SuspendThreadResult print_thread_dump (MonoThreadInfo *info, gpointer ud) { MonoInternalThread *thread = ud; GString* text = g_string_new (0); char *name; GError *error = NULL; if (thread->name) { name = g_utf16_to_utf8 (thread->name, thread->name_len, NULL, NULL, &error); g_assert (!error); g_string_append_printf (text, "\n\"%s\"", name); g_free (name); } else if (thread->threadpool_thread) g_string_append (text, "\n\"\""); else g_string_append (text, "\n\"\""); #if 0 /* This no longer works with remote unwinding */ #ifndef HOST_WIN32 wapi_desc = wapi_current_thread_desc (); g_string_append_printf (text, " tid=0x%p this=0x%p %s\n", (gpointer)(gsize)thread->tid, thread, wapi_desc); free (wapi_desc); #endif #endif mono_get_eh_callbacks ()->mono_walk_stack_with_state (print_stack_frame_to_string, mono_thread_info_get_suspend_state (info), MONO_UNWIND_SIGNAL_SAFE, text); fprintf (stdout, "%s", text->str); #if PLATFORM_WIN32 && TARGET_WIN32 && _DEBUG OutputDebugStringA(text->str); #endif g_string_free (text, TRUE); fflush (stdout); return MonoResumeThread; } static void dump_thread (gpointer key, gpointer value, gpointer user) { MonoInternalThread *thread = (MonoInternalThread *)value; if (thread == mono_thread_internal_current ()) return; /* FIXME This still can hang if we stop a thread during malloc. FIXME This can hang if we suspend on a critical method and the GC kicks in. A fix might be to have function that takes a callback and runs it with the target suspended. We probably should loop a bit around trying to get it to either managed code or WSJ state. */ mono_thread_info_safe_suspend_and_run (thread_get_tid (thread), FALSE, print_thread_dump, thread); } void mono_threads_perform_thread_dump (void) { if (!thread_dump_requested) return; printf ("Full thread dump:\n"); /* We take the loader lock and the root domain lock as to increase our odds of not deadlocking if something needs then in the process. */ mono_loader_lock (); mono_domain_lock (mono_get_root_domain ()); mono_threads_lock (); mono_g_hash_table_foreach (threads, dump_thread, NULL); mono_threads_unlock (); mono_domain_unlock (mono_get_root_domain ()); mono_loader_unlock (); thread_dump_requested = FALSE; } /** * mono_threads_request_thread_dump: * * Ask all threads except the current to print their stacktrace to stdout. */ void mono_threads_request_thread_dump (void) { /*The new thread dump code runs out of the finalizer thread. */ thread_dump_requested = TRUE; mono_gc_finalize_notify (); } struct ref_stack { gpointer *refs; gint allocated; /* +1 so that refs [allocated] == NULL */ gint bottom; }; typedef struct ref_stack RefStack; static RefStack * ref_stack_new (gint initial_size) { RefStack *rs; initial_size = MAX (initial_size, 16) + 1; rs = g_new0 (RefStack, 1); rs->refs = g_new0 (gpointer, initial_size); rs->allocated = initial_size; return rs; } static void ref_stack_destroy (gpointer ptr) { RefStack *rs = ptr; if (rs != NULL) { g_free (rs->refs); g_free (rs); } } static void ref_stack_push (RefStack *rs, gpointer ptr) { g_assert (rs != NULL); if (rs->bottom >= rs->allocated) { rs->refs = g_realloc (rs->refs, rs->allocated * 2 * sizeof (gpointer) + 1); rs->allocated <<= 1; rs->refs [rs->allocated] = NULL; } rs->refs [rs->bottom++] = ptr; } static void ref_stack_pop (RefStack *rs) { if (rs == NULL || rs->bottom == 0) return; rs->bottom--; rs->refs [rs->bottom] = NULL; } static gboolean ref_stack_find (RefStack *rs, gpointer ptr) { gpointer *refs; if (rs == NULL) return FALSE; for (refs = rs->refs; refs && *refs; refs++) { if (*refs == ptr) return TRUE; } return FALSE; } /* * mono_thread_push_appdomain_ref: * * Register that the current thread may have references to objects in domain * @domain on its stack. Each call to this function should be paired with a * call to pop_appdomain_ref. */ void mono_thread_push_appdomain_ref (MonoDomain *domain) { MonoInternalThread *thread = mono_thread_internal_current (); if (thread) { /* printf ("PUSH REF: %"G_GSIZE_FORMAT" -> %s.\n", (gsize)thread->tid, domain->friendly_name); */ SPIN_LOCK (thread->lock_thread_id); if (thread->appdomain_refs == NULL) thread->appdomain_refs = ref_stack_new (16); ref_stack_push (thread->appdomain_refs, domain); SPIN_UNLOCK (thread->lock_thread_id); } } void mono_thread_pop_appdomain_ref (void) { MonoInternalThread *thread = mono_thread_internal_current (); if (thread) { /* printf ("POP REF: %"G_GSIZE_FORMAT" -> %s.\n", (gsize)thread->tid, ((MonoDomain*)(thread->appdomain_refs->data))->friendly_name); */ SPIN_LOCK (thread->lock_thread_id); ref_stack_pop (thread->appdomain_refs); SPIN_UNLOCK (thread->lock_thread_id); } } gboolean mono_thread_internal_has_appdomain_ref (MonoInternalThread *thread, MonoDomain *domain) { gboolean res; SPIN_LOCK (thread->lock_thread_id); res = ref_stack_find (thread->appdomain_refs, domain); SPIN_UNLOCK (thread->lock_thread_id); return res; } gboolean mono_thread_has_appdomain_ref (MonoThread *thread, MonoDomain *domain) { return mono_thread_internal_has_appdomain_ref (thread->internal_thread, domain); } typedef struct abort_appdomain_data { struct wait_data wait; MonoDomain *domain; } abort_appdomain_data; static void collect_appdomain_thread (gpointer key, gpointer value, gpointer user_data) { MonoInternalThread *thread = (MonoInternalThread*)value; abort_appdomain_data *data = (abort_appdomain_data*)user_data; MonoDomain *domain = data->domain; if (mono_thread_internal_has_appdomain_ref (thread, domain)) { /* printf ("ABORTING THREAD %p BECAUSE IT REFERENCES DOMAIN %s.\n", thread->tid, domain->friendly_name); */ if(data->wait.numhandle, thread_get_tid (thread)); if (handle == NULL) return; data->wait.handles [data->wait.num] = handle; data->wait.threads [data->wait.num] = thread; data->wait.num++; } else { /* Just ignore the rest, we can't do anything with * them yet */ } } } /* * mono_threads_abort_appdomain_threads: * * Abort threads which has references to the given appdomain. */ gboolean mono_threads_abort_appdomain_threads (MonoDomain *domain, int timeout) { #ifdef __native_client__ return FALSE; #endif abort_appdomain_data user_data; guint32 start_time; int orig_timeout = timeout; int i; THREAD_DEBUG (g_message ("%s: starting abort", __func__)); start_time = mono_msec_ticks (); do { mono_threads_lock (); user_data.domain = domain; user_data.wait.num = 0; /* This shouldn't take any locks */ mono_g_hash_table_foreach (threads, collect_appdomain_thread, &user_data); mono_threads_unlock (); if (user_data.wait.num > 0) { /* Abort the threads outside the threads lock */ for (i = 0; i < user_data.wait.num; ++i) ves_icall_System_Threading_Thread_Abort (user_data.wait.threads [i], NULL); /* * We should wait for the threads either to abort, or to leave the * domain. We can't do the latter, so we wait with a timeout. */ wait_for_tids (&user_data.wait, 100); } /* Update remaining time */ timeout -= mono_msec_ticks () - start_time; start_time = mono_msec_ticks (); if (orig_timeout != -1 && timeout < 0) return FALSE; } while (user_data.wait.num > 0); THREAD_DEBUG (g_message ("%s: abort done", __func__)); return TRUE; } static void clear_cached_culture (gpointer key, gpointer value, gpointer user_data) { MonoInternalThread *thread = (MonoInternalThread*)value; MonoDomain *domain = (MonoDomain*)user_data; int i; /* No locking needed here */ /* FIXME: why no locking? writes to the cache are protected with synch_cs above */ if (thread->cached_culture_info) { for (i = 0; i < NUM_CACHED_CULTURES * 2; ++i) { MonoObject *obj = mono_array_get (thread->cached_culture_info, MonoObject*, i); if (obj && obj->vtable->domain == domain) mono_array_set (thread->cached_culture_info, MonoObject*, i, NULL); } } } /* * mono_threads_clear_cached_culture: * * Clear the cached_current_culture from all threads if it is in the * given appdomain. */ void mono_threads_clear_cached_culture (MonoDomain *domain) { mono_threads_lock (); mono_g_hash_table_foreach (threads, clear_cached_culture, domain); mono_threads_unlock (); } /* * mono_thread_get_undeniable_exception: * * Return an exception which needs to be raised when leaving a catch clause. * This is used for undeniable exception propagation. */ MonoException* mono_thread_get_undeniable_exception (void) { MonoInternalThread *thread = mono_thread_internal_current (); if (thread && thread->abort_exc && !is_running_protected_wrapper ()) { /* * FIXME: Clear the abort exception and return an AppDomainUnloaded * exception if the thread no longer references a dying appdomain. */ thread->abort_exc->trace_ips = NULL; thread->abort_exc->stack_trace = NULL; return thread->abort_exc; } return NULL; } #if MONO_SMALL_CONFIG #define NUM_STATIC_DATA_IDX 4 static const int static_data_size [NUM_STATIC_DATA_IDX] = { 64, 256, 1024, 4096 }; #else #define NUM_STATIC_DATA_IDX 8 static const int static_data_size [NUM_STATIC_DATA_IDX] = { 1024, 4096, 16384, 65536, 262144, 1048576, 4194304, 16777216 }; #endif static MonoBitSet *thread_reference_bitmaps [NUM_STATIC_DATA_IDX]; static MonoBitSet *context_reference_bitmaps [NUM_STATIC_DATA_IDX]; static void mark_slots (void *addr, MonoBitSet **bitmaps, MonoGCMarkFunc mark_func, void *gc_data) { gpointer *static_data = addr; for (int i = 0; i < NUM_STATIC_DATA_IDX; ++i) { void **ptr = static_data [i]; if (!ptr) continue; MONO_BITSET_FOREACH (bitmaps [i], idx, { void **p = ptr + idx; if (*p) mark_func ((MonoObject**)p, gc_data); }); } } static void mark_tls_slots (void *addr, MonoGCMarkFunc mark_func, void *gc_data) { mark_slots (addr, thread_reference_bitmaps, mark_func, gc_data); } static void mark_ctx_slots (void *addr, MonoGCMarkFunc mark_func, void *gc_data) { mark_slots (addr, context_reference_bitmaps, mark_func, gc_data); } /* * mono_alloc_static_data * * Allocate memory blocks for storing threads or context static data */ static void mono_alloc_static_data (gpointer **static_data_ptr, guint32 offset, gboolean threadlocal) { guint idx = ACCESS_SPECIAL_STATIC_OFFSET (offset, index); int i; gpointer* static_data = *static_data_ptr; if (!static_data) { static MonoGCDescriptor tls_desc = MONO_GC_DESCRIPTOR_NULL; static MonoGCDescriptor ctx_desc = MONO_GC_DESCRIPTOR_NULL; if (mono_gc_user_markers_supported ()) { if (tls_desc == MONO_GC_DESCRIPTOR_NULL) tls_desc = mono_gc_make_root_descr_user (mark_tls_slots); if (ctx_desc == MONO_GC_DESCRIPTOR_NULL) ctx_desc = mono_gc_make_root_descr_user (mark_ctx_slots); } static_data = mono_gc_alloc_fixed (static_data_size [0], threadlocal ? tls_desc : ctx_desc, threadlocal ? MONO_ROOT_SOURCE_THREAD_STATIC : MONO_ROOT_SOURCE_CONTEXT_STATIC, threadlocal ? "managed thread-static variables" : "managed context-static variables"); *static_data_ptr = static_data; static_data [0] = static_data; } for (i = 1; i <= idx; ++i) { if (static_data [i]) continue; if (mono_gc_user_markers_supported ()) static_data [i] = g_malloc0 (static_data_size [i]); else static_data [i] = mono_gc_alloc_fixed (static_data_size [i], MONO_GC_DESCRIPTOR_NULL, threadlocal ? MONO_ROOT_SOURCE_THREAD_STATIC : MONO_ROOT_SOURCE_CONTEXT_STATIC, threadlocal ? "managed thread-static variables" : "managed context-static variables"); } } static void mono_free_static_data (gpointer* static_data) { int i; for (i = 1; i < NUM_STATIC_DATA_IDX; ++i) { gpointer p = static_data [i]; if (!p) continue; /* * At this point, the static data pointer array is still registered with the * GC, so must ensure that mark_tls_slots() will not encounter any invalid * data. Freeing the individual arrays without first nulling their slots * would make it possible for mark_tls/ctx_slots() to encounter a pointer to * such an already freed array. See bug #13813. */ static_data [i] = NULL; mono_memory_write_barrier (); if (mono_gc_user_markers_supported ()) g_free (p); else mono_gc_free_fixed (p); } mono_gc_free_fixed (static_data); } /* * mono_init_static_data_info * * Initializes static data counters */ static void mono_init_static_data_info (StaticDataInfo *static_data) { static_data->idx = 0; static_data->offset = 0; static_data->freelist = NULL; } /* * mono_alloc_static_data_slot * * Generates an offset for static data. static_data contains the counters * used to generate it. */ static guint32 mono_alloc_static_data_slot (StaticDataInfo *static_data, guint32 size, guint32 align) { if (!static_data->idx && !static_data->offset) { /* * we use the first chunk of the first allocation also as * an array for the rest of the data */ static_data->offset = sizeof (gpointer) * NUM_STATIC_DATA_IDX; } static_data->offset += align - 1; static_data->offset &= ~(align - 1); if (static_data->offset + size >= static_data_size [static_data->idx]) { static_data->idx ++; g_assert (size <= static_data_size [static_data->idx]); g_assert (static_data->idx < NUM_STATIC_DATA_IDX); static_data->offset = 0; } guint32 offset = MAKE_SPECIAL_STATIC_OFFSET (static_data->idx, static_data->offset, 0); static_data->offset += size; return offset; } /* * ensure thread static fields already allocated are valid for thread * This function is called when a thread is created or on thread attach. */ static void thread_adjust_static_data (MonoInternalThread *thread) { mono_threads_lock (); if (thread_static_info.offset || thread_static_info.idx > 0) { /* get the current allocated size */ guint32 offset = MAKE_SPECIAL_STATIC_OFFSET (thread_static_info.idx, thread_static_info.offset, 0); mono_alloc_static_data (&thread->static_data, offset, TRUE); } mono_threads_unlock (); } /* * LOCKING: requires that threads_mutex is held */ static void context_adjust_static_data (MonoAppContext *ctx) { if (context_static_info.offset || context_static_info.idx > 0) { guint32 offset = MAKE_SPECIAL_STATIC_OFFSET (context_static_info.idx, context_static_info.offset, 0); mono_alloc_static_data (&ctx->static_data, offset, FALSE); } } /* * LOCKING: requires that threads_mutex is held */ static void alloc_thread_static_data_helper (gpointer key, gpointer value, gpointer user) { MonoInternalThread *thread = value; guint32 offset = GPOINTER_TO_UINT (user); mono_alloc_static_data (&(thread->static_data), offset, TRUE); } /* * LOCKING: requires that threads_mutex is held */ static gboolean alloc_context_static_data_helper (gpointer key, gpointer value, gpointer user) { uint32_t gch = GPOINTER_TO_INT (key); MonoAppContext *ctx = (MonoAppContext *) mono_gchandle_get_target (gch); if (!ctx) { mono_gchandle_free (gch); return TRUE; // Remove this key/value pair } guint32 offset = GPOINTER_TO_UINT (user); mono_alloc_static_data (&ctx->static_data, offset, FALSE); return FALSE; // Don't remove it } static StaticDataFreeList* search_slot_in_freelist (StaticDataInfo *static_data, guint32 size, guint32 align) { StaticDataFreeList* prev = NULL; StaticDataFreeList* tmp = static_data->freelist; while (tmp) { if (tmp->size == size) { if (prev) prev->next = tmp->next; else static_data->freelist = tmp->next; return tmp; } prev = tmp; tmp = tmp->next; } return NULL; } #if SIZEOF_VOID_P == 4 #define ONE_P 1 #else #define ONE_P 1ll #endif static void update_reference_bitmap (MonoBitSet **sets, guint32 offset, uintptr_t *bitmap, int numbits) { int idx = ACCESS_SPECIAL_STATIC_OFFSET (offset, index); if (!sets [idx]) sets [idx] = mono_bitset_new (static_data_size [idx] / sizeof (uintptr_t), 0); MonoBitSet *rb = sets [idx]; offset = ACCESS_SPECIAL_STATIC_OFFSET (offset, offset); offset /= sizeof (uintptr_t); /* offset is now the bitmap offset */ for (int i = 0; i < numbits; ++i) { if (bitmap [i / sizeof (uintptr_t)] & (ONE_P << (i & (sizeof (uintptr_t) * 8 -1)))) mono_bitset_set_fast (rb, offset + i); } } static void clear_reference_bitmap (MonoBitSet **sets, guint32 offset, guint32 size) { int idx = ACCESS_SPECIAL_STATIC_OFFSET (offset, index); MonoBitSet *rb = sets [idx]; offset = ACCESS_SPECIAL_STATIC_OFFSET (offset, offset); offset /= sizeof (uintptr_t); /* offset is now the bitmap offset */ for (int i = 0; i < size / sizeof (uintptr_t); i++) mono_bitset_clear_fast (rb, offset + i); } guint32 mono_alloc_special_static_data (guint32 static_type, guint32 size, guint32 align, uintptr_t *bitmap, int numbits) { g_assert (static_type == SPECIAL_STATIC_THREAD || static_type == SPECIAL_STATIC_CONTEXT); StaticDataInfo *info; MonoBitSet **sets; if (static_type == SPECIAL_STATIC_THREAD) { info = &thread_static_info; sets = thread_reference_bitmaps; } else { info = &context_static_info; sets = context_reference_bitmaps; } mono_threads_lock (); StaticDataFreeList *item = search_slot_in_freelist (info, size, align); guint32 offset; if (item) { offset = item->offset; g_free (item); } else { offset = mono_alloc_static_data_slot (info, size, align); } update_reference_bitmap (sets, offset, bitmap, numbits); if (static_type == SPECIAL_STATIC_THREAD) { /* This can be called during startup */ if (threads != NULL) mono_g_hash_table_foreach (threads, alloc_thread_static_data_helper, GUINT_TO_POINTER (offset)); } else { if (contexts != NULL) g_hash_table_foreach_remove (contexts, alloc_context_static_data_helper, GUINT_TO_POINTER (offset)); ACCESS_SPECIAL_STATIC_OFFSET (offset, type) = SPECIAL_STATIC_OFFSET_TYPE_CONTEXT; } mono_threads_unlock (); return offset; } gpointer mono_get_special_static_data_for_thread (MonoInternalThread *thread, guint32 offset) { guint32 static_type = ACCESS_SPECIAL_STATIC_OFFSET (offset, type); if (static_type == SPECIAL_STATIC_OFFSET_TYPE_THREAD) { return get_thread_static_data (thread, offset); } else { return get_context_static_data (thread->current_appcontext, offset); } } gpointer mono_get_special_static_data (guint32 offset) { return mono_get_special_static_data_for_thread (mono_thread_internal_current (), offset); } typedef struct { guint32 offset; guint32 size; } OffsetSize; /* * LOCKING: requires that threads_mutex is held */ static void free_thread_static_data_helper (gpointer key, gpointer value, gpointer user) { MonoInternalThread *thread = value; OffsetSize *data = user; int idx = ACCESS_SPECIAL_STATIC_OFFSET (data->offset, index); int off = ACCESS_SPECIAL_STATIC_OFFSET (data->offset, offset); char *ptr; if (!thread->static_data || !thread->static_data [idx]) return; ptr = ((char*) thread->static_data [idx]) + off; mono_gc_bzero_atomic (ptr, data->size); } /* * LOCKING: requires that threads_mutex is held */ static gboolean free_context_static_data_helper (gpointer key, gpointer value, gpointer user) { uint32_t gch = GPOINTER_TO_INT (key); MonoAppContext *ctx = (MonoAppContext *) mono_gchandle_get_target (gch); if (!ctx) { mono_gchandle_free (gch); return TRUE; // Remove this key/value pair } OffsetSize *data = user; int idx = ACCESS_SPECIAL_STATIC_OFFSET (data->offset, index); int off = ACCESS_SPECIAL_STATIC_OFFSET (data->offset, offset); char *ptr; if (!ctx->static_data || !ctx->static_data [idx]) return FALSE; // Don't remove this key/value pair ptr = ((char*) ctx->static_data [idx]) + off; mono_gc_bzero_atomic (ptr, data->size); return FALSE; // Don't remove this key/value pair } static void do_free_special_slot (guint32 offset, guint32 size) { guint32 static_type = ACCESS_SPECIAL_STATIC_OFFSET (offset, type); MonoBitSet **sets; StaticDataInfo *info; if (static_type == SPECIAL_STATIC_OFFSET_TYPE_THREAD) { info = &thread_static_info; sets = thread_reference_bitmaps; } else { info = &context_static_info; sets = context_reference_bitmaps; } guint32 data_offset = offset; ACCESS_SPECIAL_STATIC_OFFSET (data_offset, type) = 0; OffsetSize data = { data_offset, size }; clear_reference_bitmap (sets, data.offset, data.size); if (static_type == SPECIAL_STATIC_OFFSET_TYPE_THREAD) { if (threads != NULL) mono_g_hash_table_foreach (threads, free_thread_static_data_helper, &data); } else { if (contexts != NULL) g_hash_table_foreach_remove (contexts, free_context_static_data_helper, &data); } if (!mono_runtime_is_shutting_down ()) { StaticDataFreeList *item = g_new0 (StaticDataFreeList, 1); item->offset = offset; item->size = size; item->next = info->freelist; info->freelist = item; } } static void do_free_special (gpointer key, gpointer value, gpointer data) { MonoClassField *field = key; guint32 offset = GPOINTER_TO_UINT (value); gint32 align; guint32 size; size = mono_type_size (field->type, &align); do_free_special_slot (offset, size); } void mono_alloc_special_static_data_free (GHashTable *special_static_fields) { mono_threads_lock (); g_hash_table_foreach (special_static_fields, do_free_special, NULL); mono_threads_unlock (); } static void mono_special_static_data_free_slot (guint32 offset, guint32 size) { /* Only ever called for ThreadLocal instances */ g_assert (ACCESS_SPECIAL_STATIC_OFFSET (offset, type) == SPECIAL_STATIC_OFFSET_TYPE_THREAD); mono_threads_lock (); do_free_special_slot (offset, size); mono_threads_unlock (); } #ifdef HOST_WIN32 static void CALLBACK dummy_apc (ULONG_PTR param) { } #endif /* * mono_thread_execute_interruption * * Performs the operation that the requested thread state requires (abort, * suspend or stop) */ static MonoException* mono_thread_execute_interruption (void) { MonoInternalThread *thread = mono_thread_internal_current (); LOCK_THREAD (thread); /* MonoThread::interruption_requested can only be changed with atomics */ if (InterlockedCompareExchange (&thread->interruption_requested, FALSE, TRUE)) { /* this will consume pending APC calls */ #ifdef HOST_WIN32 WaitForSingleObjectEx (GetCurrentThread(), 0, TRUE); #endif InterlockedDecrement (&thread_interruption_requested); /* Clear the interrupted flag of the thread so it can wait again */ mono_thread_info_clear_self_interrupt (); } if ((thread->state & ThreadState_AbortRequested) != 0) { UNLOCK_THREAD (thread); if (thread->abort_exc == NULL) { /* * This might be racy, but it has to be called outside the lock * since it calls managed code. */ MONO_OBJECT_SETREF (thread, abort_exc, mono_get_exception_thread_abort ()); } return thread->abort_exc; } else if ((thread->state & ThreadState_SuspendRequested) != 0) { self_suspend_internal (thread); return NULL; } else if ((thread->state & ThreadState_StopRequested) != 0) { /* FIXME: do this through the JIT? */ UNLOCK_THREAD (thread); mono_thread_exit (); return NULL; } else if (thread->pending_exception) { MonoException *exc; exc = thread->pending_exception; thread->pending_exception = NULL; UNLOCK_THREAD (thread); return exc; } else if (thread->thread_interrupt_requested) { thread->thread_interrupt_requested = FALSE; UNLOCK_THREAD (thread); return(mono_get_exception_thread_interrupted ()); } UNLOCK_THREAD (thread); return NULL; } /* * mono_thread_request_interruption * * A signal handler can call this method to request the interruption of a * thread. The result of the interruption will depend on the current state of * the thread. If the result is an exception that needs to be throw, it is * provided as return value. */ MonoException* mono_thread_request_interruption (gboolean running_managed) { MonoInternalThread *thread = mono_thread_internal_current (); /* The thread may already be stopping */ if (thread == NULL) return NULL; #ifdef HOST_WIN32 if (thread->interrupt_on_stop && thread->state & ThreadState_StopRequested && thread->state & ThreadState_Background) ExitThread (1); #endif if (InterlockedCompareExchange (&thread->interruption_requested, 1, 0) == 1) return NULL; InterlockedIncrement (&thread_interruption_requested); if (!running_managed || is_running_protected_wrapper ()) { /* Can't stop while in unmanaged code. Increase the global interruption request count. When exiting the unmanaged method the count will be checked and the thread will be interrupted. */ if (mono_thread_notify_pending_exc_fn && !running_managed) /* The JIT will notify the thread about the interruption */ /* This shouldn't take any locks */ mono_thread_notify_pending_exc_fn (NULL); /* this will awake the thread if it is in WaitForSingleObject or similar */ /* Our implementation of this function ignores the func argument */ #ifdef HOST_WIN32 QueueUserAPC ((PAPCFUNC)dummy_apc, thread->handle, (ULONG_PTR)NULL); #else mono_thread_info_self_interrupt (); #endif return NULL; } else { return mono_thread_execute_interruption (); } } /*This function should be called by a thread after it has exited all of * its handle blocks at interruption time.*/ MonoException* mono_thread_resume_interruption (void) { MonoInternalThread *thread = mono_thread_internal_current (); gboolean still_aborting; /* The thread may already be stopping */ if (thread == NULL) return NULL; LOCK_THREAD (thread); still_aborting = (thread->state & ThreadState_AbortRequested) != 0; UNLOCK_THREAD (thread); /*This can happen if the protected block called Thread::ResetAbort*/ if (!still_aborting) return FALSE; if (InterlockedCompareExchange (&thread->interruption_requested, 1, 0) == 1) return NULL; InterlockedIncrement (&thread_interruption_requested); mono_thread_info_self_interrupt (); return mono_thread_execute_interruption (); } gboolean mono_thread_interruption_requested () { if (thread_interruption_requested) { MonoInternalThread *thread = mono_thread_internal_current (); /* The thread may already be stopping */ if (thread != NULL) return (thread->interruption_requested); } return FALSE; } static MonoException* mono_thread_interruption_checkpoint_request (gboolean bypass_abort_protection) { MonoInternalThread *thread = mono_thread_internal_current (); /* The thread may already be stopping */ if (thread == NULL) return NULL; if (thread->interruption_requested && (bypass_abort_protection || !is_running_protected_wrapper ())) { MonoException* exc = mono_thread_execute_interruption (); if (exc) return exc; } return NULL; } /* * Performs the interruption of the current thread, if one has been requested, * and the thread is not running a protected wrapper. * Return the exception which needs to be thrown, if any. */ MonoException* mono_thread_interruption_checkpoint (void) { return mono_thread_interruption_checkpoint_request (FALSE); } /* * Performs the interruption of the current thread, if one has been requested. * Return the exception which needs to be thrown, if any. */ MonoException* mono_thread_force_interruption_checkpoint_noraise (void) { return mono_thread_interruption_checkpoint_request (TRUE); } /* * Performs the interruption of the current thread, if one has been requested. * Throw the exception which needs to be thrown, if any. */ void mono_thread_force_interruption_checkpoint (void) { MonoException *ex; ex = mono_thread_interruption_checkpoint_request (TRUE); if (ex) mono_raise_exception (ex); } /* * mono_thread_get_and_clear_pending_exception: * * Return any pending exceptions for the current thread and clear it as a side effect. */ MonoException* mono_thread_get_and_clear_pending_exception (void) { MonoInternalThread *thread = mono_thread_internal_current (); /* The thread may already be stopping */ if (thread == NULL) return NULL; if (thread->interruption_requested && !is_running_protected_wrapper ()) { return mono_thread_execute_interruption (); } if (thread->pending_exception) { MonoException *exc = thread->pending_exception; thread->pending_exception = NULL; return exc; } return NULL; } /* * mono_set_pending_exception: * * Set the pending exception of the current thread to EXC. * The exception will be thrown when execution returns to managed code. */ void mono_set_pending_exception (MonoException *exc) { MonoInternalThread *thread = mono_thread_internal_current (); /* The thread may already be stopping */ if (thread == NULL) return; MONO_OBJECT_SETREF (thread, pending_exception, exc); mono_thread_request_interruption (FALSE); } /** * mono_thread_interruption_request_flag: * * Returns the address of a flag that will be non-zero if an interruption has * been requested for a thread. The thread to interrupt may not be the current * thread, so an additional call to mono_thread_interruption_requested() or * mono_thread_interruption_checkpoint() is allways needed if the flag is not * zero. */ gint32* mono_thread_interruption_request_flag () { return &thread_interruption_requested; } void mono_thread_init_apartment_state (void) { #ifdef HOST_WIN32 MonoInternalThread* thread = mono_thread_internal_current (); /* Positive return value indicates success, either * S_OK if this is first CoInitialize call, or * S_FALSE if CoInitialize already called, but with same * threading model. A negative value indicates failure, * probably due to trying to change the threading model. */ if (CoInitializeEx(NULL, (thread->apartment_state == ThreadApartmentState_STA) ? COINIT_APARTMENTTHREADED : COINIT_MULTITHREADED) < 0) { thread->apartment_state = ThreadApartmentState_Unknown; } #endif } void mono_thread_cleanup_apartment_state (void) { #ifdef HOST_WIN32 MonoInternalThread* thread = mono_thread_internal_current (); if (thread && thread->apartment_state != ThreadApartmentState_Unknown) { CoUninitialize (); } #endif } void mono_thread_set_state (MonoInternalThread *thread, MonoThreadState state) { LOCK_THREAD (thread); thread->state |= state; UNLOCK_THREAD (thread); } void mono_thread_clr_state (MonoInternalThread *thread, MonoThreadState state) { LOCK_THREAD (thread); thread->state &= ~state; UNLOCK_THREAD (thread); } gboolean mono_thread_test_state (MonoInternalThread *thread, MonoThreadState test) { gboolean ret = FALSE; LOCK_THREAD (thread); if ((thread->state & test) != 0) { ret = TRUE; } UNLOCK_THREAD (thread); return ret; } static gboolean has_tls_get = FALSE; void mono_runtime_set_has_tls_get (gboolean val) { has_tls_get = val; } gboolean mono_runtime_has_tls_get (void) { return has_tls_get; } static void self_interrupt_thread (void *_unused) { MonoThreadInfo *info = mono_thread_info_current (); MonoException *exc = mono_thread_execute_interruption (); if (exc) /*We must use _with_context since we didn't trampoline into the runtime*/ mono_raise_exception_with_context (exc, &info->thread_saved_state [ASYNC_SUSPEND_STATE_INDEX].ctx); /* FIXME using thread_saved_state [ASYNC_SUSPEND_STATE_INDEX] can race with another suspend coming in. */ g_assert_not_reached (); /*this MUST not happen since we can't resume from an async call*/ } static gboolean mono_jit_info_match (MonoJitInfo *ji, gpointer ip) { if (!ji) return FALSE; return ji->code_start <= ip && (char*)ip < (char*)ji->code_start + ji->code_size; } static gboolean last_managed (MonoStackFrameInfo *frame, MonoContext *ctx, gpointer data) { MonoJitInfo **dest = data; *dest = frame->ji; return TRUE; } static MonoJitInfo* mono_thread_info_get_last_managed (MonoThreadInfo *info) { MonoJitInfo *ji = NULL; if (!info) return NULL; mono_get_eh_callbacks ()->mono_walk_stack_with_state (last_managed, mono_thread_info_get_suspend_state (info), MONO_UNWIND_SIGNAL_SAFE, &ji); return ji; } typedef struct { MonoInternalThread *thread; gboolean install_async_abort; MonoThreadInfoInterruptToken *interrupt_token; } AbortThreadData; static SuspendThreadResult abort_thread_critical (MonoThreadInfo *info, gpointer ud) { AbortThreadData *data = ud; MonoInternalThread *thread = data->thread; MonoJitInfo *ji = NULL; gboolean protected_wrapper; gboolean running_managed; if (mono_get_eh_callbacks ()->mono_install_handler_block_guard (mono_thread_info_get_suspend_state (info))) return MonoResumeThread; /*someone is already interrupting it*/ if (InterlockedCompareExchange (&thread->interruption_requested, 1, 0) == 1) return MonoResumeThread; InterlockedIncrement (&thread_interruption_requested); ji = mono_thread_info_get_last_managed (info); protected_wrapper = ji && !ji->is_trampoline && !ji->async && mono_threads_is_critical_method (mono_jit_info_get_method (ji)); running_managed = mono_jit_info_match (ji, MONO_CONTEXT_GET_IP (&mono_thread_info_get_suspend_state (info)->ctx)); if (!protected_wrapper && running_managed) { /*We are in managed code*/ /*Set the thread to call */ if (data->install_async_abort) mono_thread_info_setup_async_call (info, self_interrupt_thread, NULL); return MonoResumeThread; } else { if (mono_thread_notify_pending_exc_fn) /* The JIT will notify the thread about the interruption */ mono_thread_notify_pending_exc_fn (info); /* * This will cause waits to be broken. * It will also prevent the thread from entering a wait, so if the thread returns * from the wait before it receives the abort signal, it will just spin in the wait * functions in the io-layer until the signal handler calls QueueUserAPC which will * make it return. */ data->interrupt_token = mono_thread_info_prepare_interrupt (info); return MonoResumeThread; } } static void abort_thread_internal (MonoInternalThread *thread, gboolean can_raise_exception, gboolean install_async_abort) { AbortThreadData data = { 0 }; data.thread = thread; data.install_async_abort = install_async_abort; /* FIXME this is insanely broken, it doesn't cause interruption to happen synchronously since passing FALSE to mono_thread_request_interruption makes sure it returns NULL */ if (thread == mono_thread_internal_current ()) { /* Do it synchronously */ MonoException *exc = mono_thread_request_interruption (can_raise_exception); if (exc) mono_raise_exception (exc); mono_thread_info_self_interrupt (); return; } mono_thread_info_safe_suspend_and_run (thread_get_tid (thread), TRUE, abort_thread_critical, &data); if (data.interrupt_token) mono_thread_info_finish_interrupt (data.interrupt_token); /*FIXME we need to wait for interruption to complete -- figure out how much into interruption we should wait for here*/ } typedef struct{ MonoInternalThread *thread; gboolean interrupt; MonoThreadInfoInterruptToken *interrupt_token; } SuspendThreadData; static SuspendThreadResult suspend_thread_critical (MonoThreadInfo *info, gpointer ud) { SuspendThreadData *data = ud; MonoInternalThread *thread = data->thread; MonoJitInfo *ji = NULL; gboolean protected_wrapper; gboolean running_managed; ji = mono_thread_info_get_last_managed (info); protected_wrapper = ji && !ji->is_trampoline && !ji->async && mono_threads_is_critical_method (mono_jit_info_get_method (ji)); running_managed = mono_jit_info_match (ji, MONO_CONTEXT_GET_IP (&mono_thread_info_get_suspend_state (info)->ctx)); if (running_managed && !protected_wrapper) { thread->state &= ~ThreadState_SuspendRequested; thread->state |= ThreadState_Suspended; return KeepSuspended; } else { if (InterlockedCompareExchange (&thread->interruption_requested, 1, 0) == 0) InterlockedIncrement (&thread_interruption_requested); if (data->interrupt) data->interrupt_token = mono_thread_info_prepare_interrupt (thread->thread_info); if (mono_thread_notify_pending_exc_fn && !running_managed) /* The JIT will notify the thread about the interruption */ mono_thread_notify_pending_exc_fn (info); return MonoResumeThread; } } static void suspend_thread_internal (MonoInternalThread *thread, gboolean interrupt) { LOCK_THREAD (thread); if (thread == mono_thread_internal_current ()) { mono_thread_info_begin_self_suspend (); //XXX replace this with better named functions thread->state &= ~ThreadState_SuspendRequested; thread->state |= ThreadState_Suspended; UNLOCK_THREAD (thread); mono_thread_info_end_self_suspend (); } else { SuspendThreadData data = { 0 }; data.thread = thread; data.interrupt = interrupt; mono_thread_info_safe_suspend_and_run (thread_get_tid (thread), interrupt, suspend_thread_critical, &data); if (data.interrupt_token) mono_thread_info_finish_interrupt (data.interrupt_token); UNLOCK_THREAD (thread); } } /*This is called with @thread synch_cs held and it must release it*/ static void self_suspend_internal (MonoInternalThread *thread) { mono_thread_info_begin_self_suspend (); thread->state &= ~ThreadState_SuspendRequested; thread->state |= ThreadState_Suspended; UNLOCK_THREAD (thread); mono_thread_info_end_self_suspend (); } /*This is called with @thread synch_cs held and it must release it*/ static gboolean resume_thread_internal (MonoInternalThread *thread) { UNLOCK_THREAD (thread); /* Awake the thread */ if (!mono_thread_info_resume (thread_get_tid (thread))) return FALSE; LOCK_THREAD (thread); thread->state &= ~ThreadState_Suspended; UNLOCK_THREAD (thread); return TRUE; } /* * mono_thread_is_foreign: * @thread: the thread to query * * This function allows one to determine if a thread was created by the mono runtime and has * a well defined lifecycle or it's a foreigh one, created by the native environment. * * Returns: true if @thread was not created by the runtime. */ mono_bool mono_thread_is_foreign (MonoThread *thread) { MonoThreadInfo *info = thread->internal_thread->thread_info; return info->runtime_thread == FALSE; } /* * mono_add_joinable_thread: * * Add TID to the list of joinable threads. * LOCKING: Acquires the threads lock. */ void mono_threads_add_joinable_thread (gpointer tid) { #ifndef HOST_WIN32 /* * We cannot detach from threads because it causes problems like * 2fd16f60/r114307. So we collect them and join them when * we have time (in he finalizer thread). */ joinable_threads_lock (); if (!joinable_threads) joinable_threads = g_hash_table_new (NULL, NULL); g_hash_table_insert (joinable_threads, tid, tid); joinable_thread_count ++; joinable_threads_unlock (); mono_gc_finalize_notify (); #endif } /* * mono_threads_join_threads: * * Join all joinable threads. This is called from the finalizer thread. * LOCKING: Acquires the threads lock. */ void mono_threads_join_threads (void) { #ifndef HOST_WIN32 GHashTableIter iter; gpointer key; gpointer tid; pthread_t thread; gboolean found; /* Fastpath */ if (!joinable_thread_count) return; while (TRUE) { joinable_threads_lock (); found = FALSE; if (g_hash_table_size (joinable_threads)) { g_hash_table_iter_init (&iter, joinable_threads); g_hash_table_iter_next (&iter, &key, (void**)&tid); thread = (pthread_t)tid; g_hash_table_remove (joinable_threads, key); joinable_thread_count --; found = TRUE; } joinable_threads_unlock (); if (found) { if (thread != pthread_self ()) /* This shouldn't block */ pthread_join (thread, NULL); } else { break; } } #endif } /* * mono_thread_join: * * Wait for thread TID to exit. * LOCKING: Acquires the threads lock. */ void mono_thread_join (gpointer tid) { #ifndef HOST_WIN32 pthread_t thread; gboolean found = FALSE; joinable_threads_lock (); if (!joinable_threads) joinable_threads = g_hash_table_new (NULL, NULL); if (g_hash_table_lookup (joinable_threads, tid)) { g_hash_table_remove (joinable_threads, tid); joinable_thread_count --; found = TRUE; } joinable_threads_unlock (); if (!found) return; thread = (pthread_t)tid; pthread_join (thread, NULL); #endif } void mono_thread_internal_check_for_interruption_critical (MonoInternalThread *thread) { if ((thread->state & (ThreadState_StopRequested | ThreadState_SuspendRequested)) != 0) mono_thread_interruption_checkpoint (); } static inline gboolean is_appdomainunloaded_exception (MonoClass *klass) { static MonoClass *app_domain_unloaded_exception_klass = NULL; if (!app_domain_unloaded_exception_klass) app_domain_unloaded_exception_klass = mono_class_from_name (mono_defaults.corlib, "System", "AppDomainUnloadedException"); g_assert (app_domain_unloaded_exception_klass); return klass == app_domain_unloaded_exception_klass; } static inline gboolean is_threadabort_exception (MonoClass *klass) { return klass == mono_defaults.threadabortexception_class; } void mono_thread_internal_unhandled_exception (MonoObject* exc) { if (mono_runtime_unhandled_exception_policy_get () == MONO_UNHANDLED_POLICY_CURRENT) { MonoClass *klass = exc->vtable->klass; if (is_threadabort_exception (klass)) { mono_thread_internal_reset_abort (mono_thread_internal_current ()); } else if (!is_appdomainunloaded_exception (klass)) { mono_unhandled_exception (exc); if (mono_environment_exitcode_get () == 1) exit (255); } } }