From 4cab29ccd439baf90f6e9754a08e65a890333e68 Mon Sep 17 00:00:00 2001 From: eregon Date: Sat, 18 Aug 2018 13:52:53 +0000 Subject: Revert r64441 * This reverts commit 647fc1227a4146ecbfeb0d59358abc8d99cd8ae6: "thread_sync.c (rb_mutex_synchronize): only unlock if we own the mutex" * Let's try to preserve the semantics of always being locked inside Mutex#synchronize, even if an exception interrupts ConditionVariable#wait. * As discussed on [Bug #14999]. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@64448 b2dd03c8-39d4-4d8f-98ff-823fe69b080e --- thread_sync.c | 74 +++++++++++++++++++++-------------------------------------- 1 file changed, 26 insertions(+), 48 deletions(-) (limited to 'thread_sync.c') diff --git a/thread_sync.c b/thread_sync.c index f3d1ccb120..5e511af0db 100644 --- a/thread_sync.c +++ b/thread_sync.c @@ -321,38 +321,6 @@ rb_mutex_owned_p(VALUE self) return owned; } -static void -mutex_do_unlock(rb_mutex_t *mutex, rb_thread_t *th) -{ - struct sync_waiter *cur = 0, *next = 0; - rb_mutex_t **th_mutex = &th->keeping_mutexes; - - VM_ASSERT(mutex->th == th); - - mutex->th = 0; - list_for_each_safe(&mutex->waitq, cur, next, node) { - list_del_init(&cur->node); - switch (cur->th->status) { - case THREAD_RUNNABLE: /* from someone else calling Thread#run */ - case THREAD_STOPPED_FOREVER: /* likely (rb_mutex_lock) */ - rb_threadptr_interrupt(cur->th); - goto found; - case THREAD_STOPPED: /* probably impossible */ - rb_bug("unexpected THREAD_STOPPED"); - case THREAD_KILLED: - /* not sure about this, possible in exit GC? */ - rb_bug("unexpected THREAD_KILLED"); - continue; - } - } - found: - while (*th_mutex != mutex) { - th_mutex = &(*th_mutex)->next_mutex; - } - *th_mutex = mutex->next_mutex; - mutex->next_mutex = NULL; -} - static const char * rb_mutex_unlock_th(rb_mutex_t *mutex, rb_thread_t *th) { @@ -365,7 +333,31 @@ rb_mutex_unlock_th(rb_mutex_t *mutex, rb_thread_t *th) err = "Attempt to unlock a mutex which is locked by another thread"; } else { - mutex_do_unlock(mutex, th); + struct sync_waiter *cur = 0, *next = 0; + rb_mutex_t **th_mutex = &th->keeping_mutexes; + + mutex->th = 0; + list_for_each_safe(&mutex->waitq, cur, next, node) { + list_del_init(&cur->node); + switch (cur->th->status) { + case THREAD_RUNNABLE: /* from someone else calling Thread#run */ + case THREAD_STOPPED_FOREVER: /* likely (rb_mutex_lock) */ + rb_threadptr_interrupt(cur->th); + goto found; + case THREAD_STOPPED: /* probably impossible */ + rb_bug("unexpected THREAD_STOPPED"); + case THREAD_KILLED: + /* not sure about this, possible in exit GC? */ + rb_bug("unexpected THREAD_KILLED"); + continue; + } + } + found: + while (*th_mutex != mutex) { + th_mutex = &(*th_mutex)->next_mutex; + } + *th_mutex = mutex->next_mutex; + mutex->next_mutex = NULL; } return err; @@ -505,20 +497,6 @@ mutex_sleep(int argc, VALUE *argv, VALUE self) return rb_mutex_sleep(self, timeout); } -static VALUE -mutex_unlock_if_owned(VALUE self) -{ - rb_thread_t *th = GET_THREAD(); - rb_mutex_t *mutex; - GetMutexPtr(self, mutex); - - /* we may not own the mutex if an exception occured */ - if (mutex->th == th) { - mutex_do_unlock(mutex, th); - } - return Qfalse; -} - /* * call-seq: * mutex.synchronize { ... } -> result of the block @@ -531,7 +509,7 @@ VALUE rb_mutex_synchronize(VALUE mutex, VALUE (*func)(VALUE arg), VALUE arg) { rb_mutex_lock(mutex); - return rb_ensure(func, arg, mutex_unlock_if_owned, mutex); + return rb_ensure(func, arg, rb_mutex_unlock, mutex); } /* -- cgit v1.2.3