mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
68 lines
1.6 KiB
Ruby
68 lines
1.6 KiB
Ruby
|
require File.expand_path('../../../spec_helper', __FILE__)
|
||
|
require 'thread'
|
||
|
|
||
|
describe "ConditionVariable#broadcast" do
|
||
|
it "returns self if nothing to broadcast to" do
|
||
|
cv = ConditionVariable.new
|
||
|
cv.broadcast.should == cv
|
||
|
end
|
||
|
|
||
|
it "returns self if something is waiting for a broadcast" do
|
||
|
m = Mutex.new
|
||
|
cv = ConditionVariable.new
|
||
|
in_synchronize = false
|
||
|
|
||
|
th = Thread.new do
|
||
|
m.synchronize do
|
||
|
in_synchronize = true
|
||
|
cv.wait(m)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
# wait for m to acquire the mutex
|
||
|
Thread.pass until in_synchronize
|
||
|
# wait until th is sleeping (ie waiting)
|
||
|
Thread.pass while th.status and th.status != "sleep"
|
||
|
|
||
|
m.synchronize { cv.broadcast }.should == cv
|
||
|
|
||
|
th.join
|
||
|
end
|
||
|
|
||
|
it "releases all threads waiting in line for this resource" do
|
||
|
m = Mutex.new
|
||
|
cv = ConditionVariable.new
|
||
|
threads = []
|
||
|
r1 = []
|
||
|
r2 = []
|
||
|
|
||
|
# large number to attempt to cause race conditions
|
||
|
100.times do |i|
|
||
|
threads << Thread.new(i) do |tid|
|
||
|
m.synchronize do
|
||
|
r1 << tid
|
||
|
cv.wait(m)
|
||
|
r2 << tid
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
# wait for all threads to acquire the mutex the first time
|
||
|
Thread.pass until m.synchronize { r1.size == threads.size }
|
||
|
# wait until all threads are sleeping (ie waiting)
|
||
|
Thread.pass until threads.all? {|th| th.status == "sleep" }
|
||
|
|
||
|
r2.should be_empty
|
||
|
m.synchronize do
|
||
|
cv.broadcast
|
||
|
end
|
||
|
|
||
|
threads.each {|t| t.join }
|
||
|
|
||
|
# ensure that all threads that enter cv.wait are released
|
||
|
r2.sort.should == r1.sort
|
||
|
# note that order is not specified as broadcast results in a race
|
||
|
# condition on regaining the lock m
|
||
|
end
|
||
|
end
|