1
0
Fork 0
mirror of https://github.com/ruby/ruby.git synced 2022-11-09 12:17:21 -05:00

* lib/thread.rb (Queue#push, #pop, SizedQueue#push, #pop): remove

code that kicks waiting thread twice, which caused race and
  deadlock.  [ruby-core:25537]

* test/thread/test_queue.rb: added.

git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@27356 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
mame 2010-04-16 11:10:08 +00:00
parent 7d26c313a1
commit 93c5002a7c
3 changed files with 50 additions and 20 deletions

View file

@ -1,3 +1,11 @@
Fri Apr 16 20:05:24 2010 Yusuke Endoh <mame@tsg.ne.jp>
* lib/thread.rb (Queue#push, #pop, SizedQueue#push, #pop): remove
code that kicks waiting thread twice, which caused race and
deadlock. [ruby-core:25537]
* test/thread/test_queue.rb: added.
Fri Apr 16 20:01:47 2010 Yusuke Endoh <mame@tsg.ne.jp>
* .gitignore: updated.

View file

@ -150,7 +150,6 @@ class Queue
# Pushes +obj+ to the queue.
#
def push(obj)
t = nil
@mutex.synchronize{
@que.push obj
begin
@ -160,10 +159,6 @@ class Queue
retry
end
}
begin
t.run if t
rescue ThreadError
end
end
#
@ -182,8 +177,8 @@ class Queue
# thread isn't suspended, and an exception is raised.
#
def pop(non_block=false)
while true
@mutex.synchronize{
@mutex.synchronize{
while true
if @que.empty?
raise ThreadError, "queue empty" if non_block
@waiting.push Thread.current
@ -191,8 +186,8 @@ class Queue
else
return @que.shift
end
}
end
end
}
end
#
@ -295,7 +290,6 @@ class SizedQueue < Queue
# until space becomes available.
#
def push(obj)
t = nil
@mutex.synchronize{
while true
break if @que.length < @max
@ -311,11 +305,6 @@ class SizedQueue < Queue
retry
end
}
begin
t.run if t
rescue ThreadError
end
end
#
@ -333,7 +322,6 @@ class SizedQueue < Queue
#
def pop(*args)
retval = super
t = nil
@mutex.synchronize {
if @que.length < @max
begin
@ -344,10 +332,6 @@ class SizedQueue < Queue
end
end
}
begin
t.run if t
rescue ThreadError
end
retval
end

38
test/thread/test_queue.rb Normal file
View file

@ -0,0 +1,38 @@
require 'test/unit'
require 'thread'
class TestQueue < Test::Unit::TestCase
def test_queue
grind(5, 1000, 15, Queue)
end
def test_sized_queue
grind(5, 1000, 15, SizedQueue, 1000)
end
def grind(num_threads, num_objects, num_iterations, klass, *args)
from_workers = klass.new(*args)
to_workers = klass.new(*args)
workers = (1..num_threads).map {
Thread.new {
while object = to_workers.pop
from_workers.push object
end
}
}
Thread.new {
num_iterations.times {
num_objects.times { to_workers.push 99 }
num_objects.times { from_workers.pop }
}
}.join
num_threads.times { to_workers.push nil }
workers.each { |t| t.join }
assert 0, from_workers.size
assert 0, to_workers.size
end
end