/* * threads.c: Thread handles * * Author: * Dick Porter (dick@ximian.com) * * (C) 2002-2006 Ximian, Inc. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #if HAVE_VALGRIND_MEMCHECK_H #include #endif #undef DEBUG #undef TLS_DEBUG /* Hash threads with tids. I thought of using TLS for this, but that * would have to set the data in the new thread, which is more hassle */ static mono_once_t thread_hash_once = MONO_ONCE_INIT; static pthread_key_t thread_hash_key; /* This key is used with attached threads and a destructor to signal * when attached threads exit, as they don't have the thread_exit() * infrastructure */ static pthread_key_t thread_attached_key; struct _WapiHandleOps _wapi_thread_ops = { NULL, /* close */ NULL, /* signal */ NULL, /* own */ NULL, /* is_owned */ NULL, /* special_wait */ NULL /* prewait */ }; static mono_once_t thread_ops_once=MONO_ONCE_INIT; static void thread_ops_init (void) { _wapi_handle_register_capabilities (WAPI_HANDLE_THREAD, WAPI_HANDLE_CAP_WAIT); } /* Called by thread_exit(), but maybe indirectly by * mono_thread_manage() via mono_thread_signal_self() too */ static void _wapi_thread_abandon_mutexes (gpointer handle) { struct _WapiHandle_thread *thread_handle; gboolean ok; int i; pid_t pid = _wapi_getpid (); pthread_t tid = pthread_self (); #ifdef DEBUG g_message ("%s: Thread %p abandoning held mutexes", __func__, handle); #endif if (handle == NULL) { handle = _wapi_thread_handle_from_id (pthread_self ()); if (handle == NULL) { /* Something gone badly wrong... */ return; } } ok = _wapi_lookup_handle (handle, WAPI_HANDLE_THREAD, (gpointer *)&thread_handle); if (ok == FALSE) { g_warning ("%s: error looking up thread handle %p", __func__, handle); return; } if (thread_handle->owner_pid != pid || !pthread_equal (thread_handle->id, tid)) { return; } for (i = 0; i < thread_handle->owned_mutexes->len; i++) { gpointer mutex = g_ptr_array_index (thread_handle->owned_mutexes, i); _wapi_mutex_abandon (mutex, pid, tid); _wapi_thread_disown_mutex (mutex); } } void _wapi_thread_set_termination_details (gpointer handle, guint32 exitstatus) { struct _WapiHandle_thread *thread_handle; gboolean ok; int thr_ret; if (_wapi_handle_issignalled (handle) || _wapi_handle_type (handle) == WAPI_HANDLE_UNUSED) { /* We must have already deliberately finished with * this thread, so don't do any more now */ return; } #ifdef DEBUG g_message ("%s: Thread %p terminating", __func__, handle); #endif thr_ret = _wapi_handle_lock_shared_handles (); g_assert (thr_ret == 0); _wapi_thread_abandon_mutexes (handle); ok = _wapi_lookup_handle (handle, WAPI_HANDLE_THREAD, (gpointer *)&thread_handle); if (ok == FALSE) { g_warning ("%s: error looking up thread handle %p", __func__, handle); _wapi_handle_unlock_shared_handles (); return; } thread_handle->exitstatus = exitstatus; thread_handle->state = THREAD_STATE_EXITED; MONO_SEM_DESTROY (&thread_handle->suspend_sem); g_ptr_array_free (thread_handle->owned_mutexes, TRUE); _wapi_shared_handle_set_signal_state (handle, TRUE); _wapi_handle_unlock_shared_handles (); #ifdef DEBUG g_message("%s: Recording thread handle %p id %ld status as %d", __func__, handle, thread_handle->id, exitstatus); #endif /* The thread is no longer active, so unref it */ _wapi_handle_unref (handle); } void _wapi_thread_signal_self (guint32 exitstatus) { gpointer handle; handle = _wapi_thread_handle_from_id (pthread_self ()); if (handle == NULL) { /* Something gone badly wrong... */ return; } _wapi_thread_set_termination_details (handle, exitstatus); } /* Called by the thread creation code as a thread is finishing up, and * by ExitThread() */ static void thread_exit (guint32 exitstatus, gpointer handle) G_GNUC_NORETURN; static void thread_exit (guint32 exitstatus, gpointer handle) { _wapi_thread_set_termination_details (handle, exitstatus); /* Call pthread_exit() to call destructors and really exit the * thread */ pthread_exit (NULL); } static void thread_attached_exit (gpointer handle) { /* Drop the extra reference we take in thread_attach, now this * thread is dead */ _wapi_thread_set_termination_details (handle, 0); } static void thread_hash_init(void) { int thr_ret; thr_ret = pthread_key_create (&thread_hash_key, NULL); g_assert (thr_ret == 0); thr_ret = pthread_key_create (&thread_attached_key, thread_attached_exit); g_assert (thr_ret == 0); } static void _wapi_thread_suspend (struct _WapiHandle_thread *thread) { g_assert (thread->owner_pid == _wapi_getpid ()); g_assert (pthread_equal (thread->id, pthread_self ())); while (MONO_SEM_WAIT (&thread->suspend_sem) != 0 && errno == EINTR); } static void _wapi_thread_resume (struct _WapiHandle_thread *thread) { if (thread->owner_pid != _wapi_getpid ()) { return; } MONO_SEM_POST (&thread->suspend_sem); } static void *thread_start_routine (gpointer args) G_GNUC_NORETURN; static void *thread_start_routine (gpointer args) { struct _WapiHandle_thread *thread = (struct _WapiHandle_thread *)args; int thr_ret; thr_ret = pthread_detach (pthread_self ()); g_assert (thr_ret == 0); thr_ret = pthread_setspecific (thread_hash_key, (void *)thread->handle); g_assert (thr_ret == 0); thread->id = pthread_self(); #ifdef DEBUG g_message ("%s: started thread id %ld", __func__, thread->id); #endif if (thread->create_flags & CREATE_SUSPENDED) { _wapi_thread_suspend (thread); } thread_exit (thread->start_routine (thread->start_arg), thread->handle); #ifndef __GNUC__ /* Even though we tell gcc that this function doesn't return, * other compilers won't see that. */ return(NULL); #endif } /** * CreateThread: * @security: Ignored for now. * @stacksize: the size in bytes of the new thread's stack. Use 0 to * default to the normal stack size. (Ignored for now). * @start: The function that the new thread should start with * @param: The parameter to give to @start. * @create: If 0, the new thread is ready to run immediately. If * %CREATE_SUSPENDED, the new thread will be in the suspended state, * requiring a ResumeThread() call to continue running. * @tid: If non-NULL, the ID of the new thread is stored here. NB * this is defined as a DWORD (ie 32bit) in the MS API, but we need to * cope with 64 bit IDs for s390x and amd64. * * Creates a new threading handle. * * Return value: a new handle, or NULL */ gpointer CreateThread(WapiSecurityAttributes *security G_GNUC_UNUSED, guint32 stacksize, WapiThreadStart start, gpointer param, guint32 create, gsize *tid) { struct _WapiHandle_thread thread_handle = {0}, *thread_handle_p; pthread_attr_t attr; gpointer handle; gboolean ok; int ret; int thr_ret; int i, unrefs = 0; gpointer ct_ret = NULL; mono_once (&thread_hash_once, thread_hash_init); mono_once (&thread_ops_once, thread_ops_init); if (start == NULL) { return(NULL); } thread_handle.state = THREAD_STATE_START; thread_handle.owner_pid = _wapi_getpid (); thread_handle.owned_mutexes = g_ptr_array_new (); thread_handle.create_flags = create; thread_handle.start_routine = start; thread_handle.start_arg = param; handle = _wapi_handle_new (WAPI_HANDLE_THREAD, &thread_handle); if (handle == _WAPI_HANDLE_INVALID) { g_warning ("%s: error creating thread handle", __func__); SetLastError (ERROR_GEN_FAILURE); return (NULL); } thr_ret = _wapi_handle_lock_shared_handles (); g_assert (thr_ret == 0); ok = _wapi_lookup_handle (handle, WAPI_HANDLE_THREAD, (gpointer *)&thread_handle_p); if (ok == FALSE) { g_warning ("%s: error looking up thread handle %p", __func__, handle); SetLastError (ERROR_GEN_FAILURE); goto cleanup; } /* Hold a reference while the thread is active, because we use * the handle to store thread exit information */ _wapi_handle_ref (handle); /* Set a 2M stack size. This is the default on Linux, but BSD * needs it. (The original bug report from Martin Dvorak * set the size to 2M-4k. I don't know why it's short by 4k, so * I'm leaving it as 2M until I'm told differently.) */ thr_ret = pthread_attr_init(&attr); g_assert (thr_ret == 0); /* defaults of 2Mb for 32bits and 4Mb for 64bits */ /* temporarily changed to use 1 MB: this allows more threads * to be used, as well as using less virtual memory and so * more is available for the GC heap. */ if (stacksize == 0){ #if HAVE_VALGRIND_MEMCHECK_H if (RUNNING_ON_VALGRIND) { stacksize = 1 << 20; } else { stacksize = (SIZEOF_VOID_P / 4) * 1024 * 1024; } #else stacksize = (SIZEOF_VOID_P / 4) * 1024 * 1024; #endif } #ifdef HAVE_PTHREAD_ATTR_SETSTACKSIZE thr_ret = pthread_attr_setstacksize(&attr, stacksize); g_assert (thr_ret == 0); #endif MONO_SEM_INIT (&thread_handle_p->suspend_sem, 0); thread_handle_p->handle = handle; ret = pthread_create (&thread_handle_p->id, &attr, thread_start_routine, (void *)thread_handle_p); if (ret != 0) { #ifdef DEBUG g_message ("%s: Thread create error: %s", __func__, strerror(ret)); #endif /* Two, because of the reference we took above */ unrefs = 2; goto cleanup; } ct_ret = handle; #ifdef DEBUG g_message("%s: Started thread handle %p ID %ld", __func__, handle, thread_handle_p->id); #endif if (tid != NULL) { #ifdef PTHREAD_POINTER_ID /* Don't use GPOINTER_TO_UINT here, it can't cope with * sizeof(void *) > sizeof(uint) when a cast to uint * would overflow */ *tid = (gsize)(thread_handle_p->id); #else *tid = thread_handle_p->id; #endif } cleanup: _wapi_handle_unlock_shared_handles (); /* Must not call _wapi_handle_unref() with the shared handles * already locked */ for (i = 0; i < unrefs; i++) { _wapi_handle_unref (handle); } return(ct_ret); } /* The only time this function is called when tid != pthread_self () * is from OpenThread (), so we can fast-path most cases by just * looking up the handle in TLS. OpenThread () must cope with a NULL * return and do a handle search in that case. */ gpointer _wapi_thread_handle_from_id (pthread_t tid) { gpointer ret; if (pthread_equal (tid, pthread_self ()) && (ret = pthread_getspecific (thread_hash_key)) != NULL) { /* We know the handle */ #ifdef DEBUG g_message ("%s: Returning %p for self thread %ld from TLS", __func__, ret, tid); #endif return(ret); } #ifdef DEBUG g_message ("%s: Returning NULL for unknown or non-self thread %ld", __func__, tid); #endif return(NULL); } static gboolean find_thread_by_id (gpointer handle, gpointer user_data) { pthread_t tid = (pthread_t)user_data; struct _WapiHandle_thread *thread_handle; gboolean ok; /* Ignore threads that have already exited (ie they are signalled) */ if (_wapi_handle_issignalled (handle) == FALSE) { ok = _wapi_lookup_handle (handle, WAPI_HANDLE_THREAD, (gpointer *)&thread_handle); if (ok == FALSE) { /* It's possible that the handle has vanished * during the _wapi_search_handle before it * gets here, so don't spam the console with * warnings. */ return(FALSE); } #ifdef DEBUG g_message ("%s: looking at thread %ld from process %d", __func__, thread_handle->id, thread_handle->owner_pid); #endif if (thread_handle->owner_pid != _wapi_getpid ()) { /* Not sure if ms has this limitation with * OpenThread(), but pthreads IDs are not * unique across processes */ #ifdef DEBUG g_message ("%s: not this process", __func__); #endif return(FALSE); } if (pthread_equal (thread_handle->id, tid)) { #ifdef DEBUG g_message ("%s: found the thread we are looking for", __func__); #endif return(TRUE); } } #ifdef DEBUG g_message ("%s: not found %ld, returning FALSE", __func__, tid); #endif return(FALSE); } /* NB tid is 32bit in MS API, but we need 64bit on amd64 and s390x * (and probably others) */ gpointer OpenThread (guint32 access G_GNUC_UNUSED, gboolean inherit G_GNUC_UNUSED, gsize tid) { gpointer ret=NULL; mono_once (&thread_hash_once, thread_hash_init); mono_once (&thread_ops_once, thread_ops_init); #ifdef DEBUG g_message ("%s: looking up thread %"G_GSIZE_FORMAT, __func__, tid); #endif ret = _wapi_thread_handle_from_id ((pthread_t)tid); if (ret == NULL) { /* We need to search for this thread */ ret = _wapi_search_handle (WAPI_HANDLE_THREAD, find_thread_by_id, (gpointer)tid, NULL, TRUE); /* FIXME: have a proper look at this, me might not need to set search_shared = TRUE */ } else { /* if _wapi_search_handle() returns a found handle, it * refs it itself */ _wapi_handle_ref (ret); } #ifdef DEBUG g_message ("%s: returning thread handle %p", __func__, ret); #endif return(ret); } /** * ExitThread: * @exitcode: Sets the thread's exit code, which can be read from * another thread with GetExitCodeThread(). * * Terminates the calling thread. A thread can also exit by returning * from its start function. When the last thread in a process * terminates, the process itself terminates. */ void ExitThread(guint32 exitcode) { gpointer thread = _wapi_thread_handle_from_id (pthread_self ()); if (thread != NULL) { thread_exit(exitcode, thread); } else { /* Just blow this thread away */ pthread_exit (NULL); } } /** * GetExitCodeThread: * @handle: The thread handle to query * @exitcode: The thread @handle exit code is stored here * * Finds the exit code of @handle, and stores it in @exitcode. If the * thread @handle is still running, the value stored is %STILL_ACTIVE. * * Return value: %TRUE, or %FALSE on error. */ gboolean GetExitCodeThread(gpointer handle, guint32 *exitcode) { struct _WapiHandle_thread *thread_handle; gboolean ok; ok = _wapi_lookup_handle (handle, WAPI_HANDLE_THREAD, (gpointer *)&thread_handle); if (ok == FALSE) { g_warning ("%s: error looking up thread handle %p", __func__, handle); return (FALSE); } #ifdef DEBUG g_message ("%s: Finding exit status for thread handle %p id %ld", __func__, handle, thread_handle->id); #endif if (exitcode == NULL) { #ifdef DEBUG g_message ("%s: Nowhere to store exit code", __func__); #endif return(FALSE); } if (thread_handle->state != THREAD_STATE_EXITED) { #ifdef DEBUG g_message ("%s: Thread still active (state %d, exited is %d)", __func__, thread_handle->state, THREAD_STATE_EXITED); #endif *exitcode = STILL_ACTIVE; return(TRUE); } *exitcode = thread_handle->exitstatus; return(TRUE); } /** * GetCurrentThreadId: * * Looks up the thread ID of the current thread. This ID can be * passed to OpenThread() to create a new handle on this thread. * * Return value: the thread ID. NB this is defined as DWORD (ie 32 * bit) in the MS API, but we need to cope with 64 bit IDs for s390x * and amd64. This doesn't really break the API, it just embraces and * extends it on 64bit platforms :) */ gsize GetCurrentThreadId(void) { pthread_t tid = pthread_self(); #ifdef PTHREAD_POINTER_ID /* Don't use GPOINTER_TO_UINT here, it can't cope with * sizeof(void *) > sizeof(uint) when a cast to uint would * overflow */ return((gsize)tid); #else return(tid); #endif } static gpointer thread_attach(gsize *tid) { struct _WapiHandle_thread thread_handle = {0}, *thread_handle_p; gpointer handle; gboolean ok; int thr_ret; mono_once (&thread_hash_once, thread_hash_init); mono_once (&thread_ops_once, thread_ops_init); thread_handle.state = THREAD_STATE_START; thread_handle.owner_pid = _wapi_getpid (); thread_handle.owned_mutexes = g_ptr_array_new (); handle = _wapi_handle_new (WAPI_HANDLE_THREAD, &thread_handle); if (handle == _WAPI_HANDLE_INVALID) { g_warning ("%s: error creating thread handle", __func__); SetLastError (ERROR_GEN_FAILURE); return (NULL); } thr_ret = _wapi_handle_lock_shared_handles (); g_assert (thr_ret == 0); ok = _wapi_lookup_handle (handle, WAPI_HANDLE_THREAD, (gpointer *)&thread_handle_p); if (ok == FALSE) { g_warning ("%s: error looking up thread handle %p", __func__, handle); SetLastError (ERROR_GEN_FAILURE); goto cleanup; } /* Hold a reference while the thread is active, because we use * the handle to store thread exit information */ _wapi_handle_ref (handle); /* suspend_sem is not used for attached threads, but * thread_exit() might try to destroy it */ MONO_SEM_INIT (&thread_handle_p->suspend_sem, 0); thread_handle_p->handle = handle; thread_handle_p->id = pthread_self (); thr_ret = pthread_setspecific (thread_hash_key, (void *)handle); g_assert (thr_ret == 0); thr_ret = pthread_setspecific (thread_attached_key, (void *)handle); g_assert (thr_ret == 0); #ifdef DEBUG g_message("%s: Attached thread handle %p ID %ld", __func__, handle, thread_handle_p->id); #endif if (tid != NULL) { #ifdef PTHREAD_POINTER_ID /* Don't use GPOINTER_TO_UINT here, it can't cope with * sizeof(void *) > sizeof(uint) when a cast to uint * would overflow */ *tid = (gsize)(thread_handle_p->id); #else *tid = thread_handle_p->id; #endif } cleanup: _wapi_handle_unlock_shared_handles (); return(handle); } gpointer _wapi_thread_duplicate () { gpointer ret = NULL; mono_once (&thread_hash_once, thread_hash_init); mono_once (&thread_ops_once, thread_ops_init); ret = _wapi_thread_handle_from_id (pthread_self ()); if (!ret) { ret = thread_attach (NULL); } else { _wapi_handle_ref (ret); } return(ret); } /** * GetCurrentThread: * * Looks up the handle associated with the current thread. Under * Windows this is a pseudohandle, and must be duplicated with * DuplicateHandle() for some operations. * * Return value: The current thread handle, or %NULL on failure. * (Unknown whether Windows has a possible failure here. It may be * necessary to implement the pseudohandle-constant behaviour). */ gpointer GetCurrentThread(void) { mono_once(&thread_hash_once, thread_hash_init); mono_once (&thread_ops_once, thread_ops_init); return(_WAPI_THREAD_CURRENT); } /** * ResumeThread: * @handle: the thread handle to resume * * Decrements the suspend count of thread @handle. A thread can only * run if its suspend count is zero. * * Return value: the previous suspend count, or 0xFFFFFFFF on error. */ guint32 ResumeThread(gpointer handle) { struct _WapiHandle_thread *thread_handle; gboolean ok; ok = _wapi_lookup_handle (handle, WAPI_HANDLE_THREAD, (gpointer *)&thread_handle); if (ok == FALSE) { g_warning ("%s: error looking up thread handle %p", __func__, handle); return (0xFFFFFFFF); } /* This is still a kludge that only copes with starting a * thread that was suspended on create, so don't bother with * the suspend count crap yet */ _wapi_thread_resume (thread_handle); return(0xFFFFFFFF); } /** * SuspendThread: * @handle: the thread handle to suspend * * Increments the suspend count of thread @handle. A thread can only * run if its suspend count is zero. * * Return value: the previous suspend count, or 0xFFFFFFFF on error. */ guint32 SuspendThread(gpointer handle) { return(0xFFFFFFFF); } /* * We assume here that TLS_MINIMUM_AVAILABLE is less than * PTHREAD_KEYS_MAX, allowing enough overhead for a few TLS keys for * library usage. * * Currently TLS_MINIMUM_AVAILABLE is 64 and _POSIX_THREAD_KEYS_MAX * (the minimum value for PTHREAD_KEYS_MAX) is 128, so we should be * fine. */ static pthread_key_t TLS_keys[TLS_MINIMUM_AVAILABLE]; static gboolean TLS_used[TLS_MINIMUM_AVAILABLE]={FALSE}; static guint32 TLS_spinlock=0; guint32 mono_pthread_key_for_tls (guint32 idx) { return (guint32)TLS_keys [idx]; } /** * TlsAlloc: * * Allocates a Thread Local Storage (TLS) index. Any thread in the * same process can use this index to store and retrieve values that * are local to that thread. * * Return value: The index value, or %TLS_OUT_OF_INDEXES if no index * is available. */ guint32 TlsAlloc(void) { guint32 i; int thr_ret; MONO_SPIN_LOCK (TLS_spinlock); for(i=0; iowner_pid != _wapi_getpid ()) { return; } apc = (ApcInfo *)g_new (ApcInfo, 1); apc->callback = apc_callback; apc->param = param; pthread_cleanup_push ((void(*)(void *))mono_mutex_unlock_in_cleanup, (void *)&apc_mutex); thr_ret = mono_mutex_lock (&apc_mutex); g_assert (thr_ret == 0); thread->apc_queue = g_slist_append (thread->apc_queue, apc); thr_ret = mono_mutex_unlock (&apc_mutex); g_assert (thr_ret == 0); pthread_cleanup_pop (0); } gboolean _wapi_thread_apc_pending (gpointer handle) { struct _WapiHandle_thread *thread; gboolean ok; ok = _wapi_lookup_handle (handle, WAPI_HANDLE_THREAD, (gpointer *)&thread); if (ok == FALSE) { #ifdef DEBUG /* This might happen at process shutdown, as all * thread handles are forcibly closed. If a thread * still has an alertable wait the final * _wapi_thread_apc_pending check will probably fail * to find the handle */ g_warning ("%s: error looking up thread handle %p", __func__, handle); #endif return (FALSE); } if (thread->owner_pid != _wapi_getpid ()) { return(FALSE); } return(thread->apc_queue != NULL); } gboolean _wapi_thread_dispatch_apc_queue (gpointer handle) { struct _WapiHandle_thread *thread; gboolean ok; ApcInfo *apc; GSList *list; int thr_ret; ok = _wapi_lookup_handle (handle, WAPI_HANDLE_THREAD, (gpointer *)&thread); if (ok == FALSE) { g_warning ("%s: error looking up thread handle %p", __func__, handle); return (FALSE); } if (thread->owner_pid != _wapi_getpid ()) { return(FALSE); } pthread_cleanup_push ((void(*)(void *))mono_mutex_unlock_in_cleanup, (void *)&apc_mutex); thr_ret = mono_mutex_lock (&apc_mutex); g_assert (thr_ret == 0); list = thread->apc_queue; thread->apc_queue = NULL; thr_ret = mono_mutex_unlock (&apc_mutex); g_assert (thr_ret == 0); pthread_cleanup_pop (0); while (list != NULL) { apc = (ApcInfo *)list->data; apc->callback (apc->param); g_free (apc); list = g_slist_next (list); } g_slist_free (list); return(TRUE); } guint32 QueueUserAPC (WapiApcProc apc_callback, gpointer handle, gpointer param) { struct _WapiHandle_thread *thread_handle; gboolean ok; ok = _wapi_lookup_handle (handle, WAPI_HANDLE_THREAD, (gpointer *)&thread_handle); if (ok == FALSE) { g_warning ("%s: error looking up thread handle %p", __func__, handle); return (0); } _wapi_thread_queue_apc (thread_handle, apc_callback, param); return(1); } void _wapi_thread_own_mutex (gpointer mutex) { struct _WapiHandle_thread *thread_handle; gboolean ok; gpointer thread; thread = _wapi_thread_handle_from_id (pthread_self ()); if (thread == NULL) { g_warning ("%s: error looking up thread by ID", __func__); return; } ok = _wapi_lookup_handle (thread, WAPI_HANDLE_THREAD, (gpointer *)&thread_handle); if (ok == FALSE) { g_warning ("%s: error looking up thread handle %p", __func__, thread); return; } _wapi_handle_ref (mutex); g_ptr_array_add (thread_handle->owned_mutexes, mutex); } void _wapi_thread_disown_mutex (gpointer mutex) { struct _WapiHandle_thread *thread_handle; gboolean ok; gpointer thread; thread = _wapi_thread_handle_from_id (pthread_self ()); if (thread == NULL) { g_warning ("%s: error looking up thread by ID", __func__); return; } ok = _wapi_lookup_handle (thread, WAPI_HANDLE_THREAD, (gpointer *)&thread_handle); if (ok == FALSE) { g_warning ("%s: error looking up thread handle %p", __func__, thread); return; } _wapi_handle_unref (mutex); g_ptr_array_remove (thread_handle->owned_mutexes, mutex); }