mirror of
https://github.com/ruby/ruby.git
synced 2025-08-15 13:39:04 +02:00
171 lines
3.3 KiB
Ruby
171 lines
3.3 KiB
Ruby
# frozen_string_literal: true
|
|
require "test/unit"
|
|
require_relative 'scheduler'
|
|
|
|
class TestFiberThread < Test::Unit::TestCase
|
|
def test_thread_join
|
|
thread = Thread.new do
|
|
scheduler = Scheduler.new
|
|
Fiber.set_scheduler scheduler
|
|
|
|
result = nil
|
|
Fiber.schedule do
|
|
result = Thread.new{:done}.value
|
|
end
|
|
|
|
scheduler.run
|
|
result
|
|
end
|
|
|
|
assert_equal :done, thread.value
|
|
end
|
|
|
|
def test_thread_join_timeout
|
|
sleeper = nil
|
|
|
|
thread = Thread.new do
|
|
scheduler = Scheduler.new
|
|
Fiber.set_scheduler scheduler
|
|
|
|
Fiber.schedule do
|
|
sleeper = Thread.new{sleep}
|
|
sleeper.join(0.1)
|
|
end
|
|
|
|
scheduler.run
|
|
end
|
|
|
|
thread.join
|
|
|
|
assert_predicate sleeper, :alive?
|
|
ensure
|
|
sleeper&.kill&.join
|
|
end
|
|
|
|
def test_thread_join_implicit
|
|
sleeping = false
|
|
finished = false
|
|
|
|
thread = Thread.new do
|
|
scheduler = Scheduler.new
|
|
Fiber.set_scheduler scheduler
|
|
|
|
Fiber.schedule do
|
|
sleeping = true
|
|
sleep(0.1)
|
|
finished = true
|
|
end
|
|
|
|
:done
|
|
end
|
|
|
|
Thread.pass until sleeping
|
|
|
|
thread.join
|
|
|
|
assert_equal :done, thread.value
|
|
assert finished, "Scheduler thread's task should be finished!"
|
|
end
|
|
|
|
def test_thread_join_blocking
|
|
thread = Thread.new do
|
|
scheduler = Scheduler.new
|
|
Fiber.set_scheduler scheduler
|
|
|
|
result = nil
|
|
Fiber.schedule do
|
|
Fiber.new(blocking: true) do
|
|
# This can deadlock if the blocking state is not taken into account:
|
|
Thread.new do
|
|
sleep(0)
|
|
result = :done
|
|
end.join
|
|
end.resume
|
|
end
|
|
|
|
scheduler.run
|
|
result
|
|
end
|
|
|
|
assert_equal :done, thread.value
|
|
end
|
|
|
|
def test_spurious_unblock_during_thread_join
|
|
ready = Thread::Queue.new
|
|
|
|
target_thread = Thread.new do
|
|
ready.pop
|
|
:success
|
|
end
|
|
|
|
Thread.pass until target_thread.status == "sleep"
|
|
|
|
result = nil
|
|
|
|
thread = Thread.new do
|
|
scheduler = Scheduler.new
|
|
Fiber.set_scheduler scheduler
|
|
|
|
# Create a fiber that will join a long-running thread:
|
|
joining_fiber = Fiber.schedule do
|
|
result = target_thread.value
|
|
end
|
|
|
|
# Create another fiber that spuriously unblocks the joining fiber:
|
|
Fiber.schedule do
|
|
# This interrupts the join in joining_fiber:
|
|
scheduler.unblock(:spurious_wakeup, joining_fiber)
|
|
|
|
# This allows the unblock to be processed:
|
|
sleep(0)
|
|
|
|
# This allows the target thread to finish:
|
|
ready.push(:done)
|
|
end
|
|
|
|
scheduler.run
|
|
end
|
|
|
|
thread.join
|
|
|
|
assert_equal :success, result
|
|
end
|
|
|
|
def test_broken_unblock
|
|
thread = Thread.new do
|
|
Thread.current.report_on_exception = false
|
|
|
|
scheduler = BrokenUnblockScheduler.new
|
|
|
|
Fiber.set_scheduler scheduler
|
|
|
|
Fiber.schedule do
|
|
Thread.new{
|
|
Thread.current.report_on_exception = false
|
|
}.join
|
|
end
|
|
|
|
scheduler.run
|
|
ensure
|
|
scheduler.close
|
|
end
|
|
|
|
assert_raise(RuntimeError) do
|
|
thread.join
|
|
end
|
|
end
|
|
|
|
def test_thread_join_hang
|
|
thread = Thread.new do
|
|
scheduler = SleepingUnblockScheduler.new
|
|
|
|
Fiber.set_scheduler scheduler
|
|
|
|
Fiber.schedule do
|
|
Thread.new{sleep(0.01)}.value
|
|
end
|
|
end
|
|
|
|
thread.join
|
|
end
|
|
end
|