ruby-changes:63022
From: Benoit <ko1@a...>
Date: Sun, 20 Sep 2020 20:39:23 +0900 (JST)
Subject: [ruby-changes:63022] 6987c8997e (master): Remove from waiter in Mutex#lock with ensure when calling rb_scheduler_block()
https://git.ruby-lang.org/ruby.git/commit/?id=6987c8997e From 6987c8997e6cd8f45bbd7ece6582c0024be0cc0f Mon Sep 17 00:00:00 2001 From: Benoit Daloze <eregontp@g...> Date: Sun, 20 Sep 2020 13:29:24 +0200 Subject: Remove from waiter in Mutex#lock with ensure when calling rb_scheduler_block() * Previously this could lead to an invalid waiter entry and then trying to wake up that waiter would result in various issues in rb_mutex_unlock_th(). diff --git a/test/fiber/test_mutex.rb b/test/fiber/test_mutex.rb index a70c699..258f535 100644 --- a/test/fiber/test_mutex.rb +++ b/test/fiber/test_mutex.rb @@ -70,6 +70,38 @@ class TestFiberMutex < Test::Unit::TestCase https://github.com/ruby/ruby/blob/trunk/test/fiber/test_mutex.rb#L70 thread.join end + def test_mutex_fiber_raise + mutex = Mutex.new + ran = false + + main = Thread.new do + mutex.lock + + thread = Thread.new do + scheduler = Scheduler.new + Thread.current.scheduler = scheduler + + f = Fiber.schedule do + assert_raise_with_message(RuntimeError, "bye") do + assert_same scheduler, Thread.scheduler + mutex.lock + end + ran = true + end + + Fiber.schedule do + f.raise "bye" + end + end + + thread.join + end + + main.join # causes mutex to be released + assert_equal false, mutex.locked? + assert_equal true, ran + end + def test_condition_variable mutex = Mutex.new condition = ConditionVariable.new diff --git a/thread_sync.c b/thread_sync.c index 94e4d35..148e609 100644 --- a/thread_sync.c +++ b/thread_sync.c @@ -214,18 +214,17 @@ VALUE https://github.com/ruby/ruby/blob/trunk/thread_sync.c#L214 rb_mutex_trylock(VALUE self) { rb_mutex_t *mutex = mutex_ptr(self); - VALUE locked = Qfalse; if (mutex->fiber == 0) { rb_fiber_t *fiber = GET_EC()->fiber_ptr; rb_thread_t *th = GET_THREAD(); mutex->fiber = fiber; - locked = Qtrue; mutex_locked(th, self); + return Qtrue; } - return locked; + return Qfalse; } /* @@ -246,6 +245,16 @@ mutex_owned_p(rb_fiber_t *fiber, rb_mutex_t *mutex) https://github.com/ruby/ruby/blob/trunk/thread_sync.c#L245 } } +static VALUE call_rb_scheduler_block(VALUE mutex) { + return rb_scheduler_block(rb_thread_current_scheduler(), mutex); +} + +static VALUE remove_from_mutex_lock_waiters(VALUE arg) { + struct list_node *node = (struct list_node*)arg; + list_del(node); + return Qnil; +} + static VALUE do_mutex_lock(VALUE self, int interruptible_p) { @@ -276,9 +285,7 @@ do_mutex_lock(VALUE self, int interruptible_p) https://github.com/ruby/ruby/blob/trunk/thread_sync.c#L285 if (scheduler != Qnil) { list_add_tail(&mutex->waitq, &w.node); - rb_scheduler_block(scheduler, self); - - list_del(&w.node); + rb_ensure(call_rb_scheduler_block, self, remove_from_mutex_lock_waiters, (VALUE)&w.node); if (!mutex->fiber) { mutex->fiber = fiber; -- cgit v0.10.2 -- ML: ruby-changes@q... Info: http://www.atdot.net/~ko1/quickml/