From 804720d2eb2b04ebc9295fc1181185941a889cb5 Mon Sep 17 00:00:00 2001 From: ngoto Date: Tue, 15 Dec 2015 15:26:47 +0000 Subject: * lib/webrick/utils.rb (WEBrick::Utils::TimeoutHandler#initialize): TimeoutMutex should be acquired when accessing @timeout_info. To avoid deadlock, interrupt() calls are delayed. Due to the mutex, it is safe to treat ary without ary.dup. [Bug #11742] [ruby-dev:49387] git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53130 b2dd03c8-39d4-4d8f-98ff-823fe69b080e --- ChangeLog | 8 ++++++++ lib/webrick/utils.rb | 23 ++++++++++++++--------- 2 files changed, 22 insertions(+), 9 deletions(-) diff --git a/ChangeLog b/ChangeLog index 72fc47146f..884c2bb4d4 100644 --- a/ChangeLog +++ b/ChangeLog @@ -1,3 +1,11 @@ +Wed Dec 16 00:25:41 2015 Naohisa Goto + + * lib/webrick/utils.rb (WEBrick::Utils::TimeoutHandler#initialize): + TimeoutMutex should be acquired when accessing @timeout_info. + To avoid deadlock, interrupt() calls are delayed. + Due to the mutex, it is safe to treat ary without ary.dup. + [Bug #11742] [ruby-dev:49387] + Tue Dec 15 23:13:10 2015 Naohisa Goto * gc.c: Delete excess semicolon after RUBY_ALIAS_FUNCTION(). diff --git a/lib/webrick/utils.rb b/lib/webrick/utils.rb index da6386c96c..dc7ce508ea 100644 --- a/lib/webrick/utils.rb +++ b/lib/webrick/utils.rb @@ -154,20 +154,25 @@ module WEBrick def initialize @timeout_info = Hash.new @watcher = Thread.start{ + to_interrupt = [] while true now = Time.now wakeup = nil - @timeout_info.each {|thread, ary| - next unless ary - ary.dup.each{|info| - time, exception = *info - if time < now - interrupt(thread, info.object_id, exception) - elsif !wakeup || time < wakeup - wakeup = time - end + to_interrupt.clear + TimeoutMutex.synchronize{ + @timeout_info.each {|thread, ary| + next unless ary + ary.each{|info| + time, exception = *info + if time < now + to_interrupt.push [thread, info.object_id, exception] + elsif !wakeup || time < wakeup + wakeup = time + end + } } } + to_interrupt.each {|arg| interrupt(*arg)} if !wakeup sleep elsif (wakeup -= now) > 0 -- cgit v1.2.3