Reset the owner ivar in LRU#synchronize

Without this the following could happen:

1. Thread A acquires the lock and sets the ownership to A.
2. Thread A yields and returns
3. Thread B tries to acquire the lock
4. At this exact moment Thread A calls the "synchronize" method again
   and sees that the "owner" variable is still set to Thread A
5. Both thread A and B can now access the underlying data in parallel,
   possibly leading to corrupted objects

This can be demonstrated using the following script:

    require 'oga'

    lru = Oga::LRU.new(64)

    threads = 50.times.map do
      Thread.new do
        loop do
          number = rand(100)

          lru[number] = number
        end
      end
    end

    threads.each(&:join)

Run this for a while on either JRuby or Rubinius and you'll end up with
errors such as "ConcurrencyError: Detected invalid array contents due to
unsynchronized modifications with concurrent users" on JRuby or
"ArgumentError: negative array size" on Rubinius.

Resetting the owner variable ensures the above can never happen. Thanks
to @chrisseaton for bringing this up earlier today.
This commit is contained in:
Yorick Peterse 2015-08-03 21:40:51 +02:00
parent ed3cbe7975
commit 32b75bf62c
1 changed files with 3 additions and 1 deletions

View File

@ -137,8 +137,10 @@ module Oga
if @owner != Thread.current if @owner != Thread.current
@mutex.synchronize do @mutex.synchronize do
@owner = Thread.current @owner = Thread.current
retval = yield
@owner = nil
yield retval
end end
else else
yield yield