Class: ActiveSupport::Concurrency::ShareLock
- Includes:
- MonitorMixin
- Defined in:
- activesupport/lib/active_support/concurrency/share_lock.rb
Overview
A share/exclusive lock, otherwise known as a read/write lock.
Instance Method Summary collapse
-
#exclusive(purpose: nil, compatible: [], after_compatible: [], no_wait: false) ⇒ Object
Execute the supplied block while holding the Exclusive lock.
-
#initialize ⇒ ShareLock
constructor
A new instance of ShareLock.
-
#raw_state ⇒ Object
We track Thread objects, instead of just using counters, because we need exclusive locks to be reentrant, and we need to be able to upgrade share locks to exclusive.
-
#sharing ⇒ Object
Execute the supplied block while holding the Share lock.
-
#start_exclusive(purpose: nil, compatible: [], no_wait: false) ⇒ Object
Returns false if
no_wait
is set and the lock is not immediately available. - #start_sharing ⇒ Object
-
#stop_exclusive(compatible: []) ⇒ Object
Relinquish the exclusive lock.
- #stop_sharing ⇒ Object
-
#yield_shares(purpose: nil, compatible: [], block_share: false) ⇒ Object
Temporarily give up all held Share locks while executing the supplied block, allowing any
compatible
exclusive lock request to proceed.
Constructor Details
#initialize ⇒ ShareLock
Returns a new instance of ShareLock.
49 50 51 52 53 54 55 56 57 58 59 |
# File 'activesupport/lib/active_support/concurrency/share_lock.rb', line 49 def initialize super() @cv = new_cond @sharing = Hash.new(0) @waiting = {} @sleeping = {} @exclusive_thread = nil @exclusive_depth = 0 end |
Instance Method Details
#exclusive(purpose: nil, compatible: [], after_compatible: [], no_wait: false) ⇒ Object
Execute the supplied block while holding the Exclusive lock. If no_wait
is set and the lock is not immediately available, returns nil
without yielding. Otherwise, returns the result of the block.
See start_exclusive
for other options.
147 148 149 150 151 152 153 154 155 |
# File 'activesupport/lib/active_support/concurrency/share_lock.rb', line 147 def exclusive(purpose: nil, compatible: [], after_compatible: [], no_wait: false) if start_exclusive(purpose: purpose, compatible: compatible, no_wait: no_wait) begin yield ensure stop_exclusive(compatible: after_compatible) end end end |
#raw_state ⇒ Object
We track Thread objects, instead of just using counters, because we need exclusive locks to be reentrant, and we need to be able to upgrade share locks to exclusive.
17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 |
# File 'activesupport/lib/active_support/concurrency/share_lock.rb', line 17 def raw_state # :nodoc: synchronize do threads = @sleeping.keys | @sharing.keys | @waiting.keys threads |= [@exclusive_thread] if @exclusive_thread data = {} threads.each do |thread| purpose, compatible = @waiting[thread] data[thread] = { thread: thread, sharing: @sharing[thread], exclusive: @exclusive_thread == thread, purpose: purpose, compatible: compatible, waiting: !!@waiting[thread], sleeper: @sleeping[thread], } end # NB: Yields while holding our *internal* synchronize lock, # which is supposed to be used only for a few instructions at # a time. This allows the caller to inspect additional state # without things changing out from underneath, but would have # disastrous effects upon normal operation. Fortunately, this # method is only intended to be called when things have # already gone wrong. yield data end end |
#sharing ⇒ Object
Execute the supplied block while holding the Share lock.
158 159 160 161 162 163 164 165 |
# File 'activesupport/lib/active_support/concurrency/share_lock.rb', line 158 def sharing start_sharing begin yield ensure stop_sharing end end |
#start_exclusive(purpose: nil, compatible: [], no_wait: false) ⇒ Object
Returns false if no_wait
is set and the lock is not immediately available. Otherwise, returns true after the lock has been acquired.
purpose
and compatible
work together; while this thread is waiting for the exclusive lock, it will yield its share (if any) to any other attempt whose purpose
appears in this attempt’s compatible
list. This allows a “loose” upgrade, which, being less strict, prevents some classes of deadlocks.
For many resources, loose upgrades are sufficient: if a thread is awaiting a lock, it is not running any other code. With purpose
matching, it is possible to yield only to other threads whose activity will not interfere.
75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 |
# File 'activesupport/lib/active_support/concurrency/share_lock.rb', line 75 def start_exclusive(purpose: nil, compatible: [], no_wait: false) synchronize do unless @exclusive_thread == Thread.current if busy_for_exclusive?(purpose) return false if no_wait yield_shares(purpose: purpose, compatible: compatible, block_share: true) do wait_for(:start_exclusive) { busy_for_exclusive?(purpose) } end end @exclusive_thread = Thread.current end @exclusive_depth += 1 true end end |
#start_sharing ⇒ Object
113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 |
# File 'activesupport/lib/active_support/concurrency/share_lock.rb', line 113 def start_sharing synchronize do if @sharing[Thread.current] > 0 || @exclusive_thread == Thread.current # We already hold a lock; nothing to wait for elsif @waiting[Thread.current] # We're nested inside a +yield_shares+ call: we'll resume as # soon as there isn't an exclusive lock in our way wait_for(:start_sharing) { @exclusive_thread } else # This is an initial / outermost share call: any outstanding # requests for an exclusive lock get to go first wait_for(:start_sharing) { busy_for_sharing?(false) } end @sharing[Thread.current] += 1 end end |
#stop_exclusive(compatible: []) ⇒ Object
Relinquish the exclusive lock. Must only be called by the thread that called start_exclusive (and currently holds the lock).
95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 |
# File 'activesupport/lib/active_support/concurrency/share_lock.rb', line 95 def stop_exclusive(compatible: []) synchronize do raise "invalid unlock" if @exclusive_thread != Thread.current @exclusive_depth -= 1 if @exclusive_depth == 0 @exclusive_thread = nil if eligible_waiters?(compatible) yield_shares(compatible: compatible, block_share: true) do wait_for(:stop_exclusive) { @exclusive_thread || eligible_waiters?(compatible) } end end @cv.broadcast end end end |
#stop_sharing ⇒ Object
130 131 132 133 134 135 136 137 138 139 |
# File 'activesupport/lib/active_support/concurrency/share_lock.rb', line 130 def stop_sharing synchronize do if @sharing[Thread.current] > 1 @sharing[Thread.current] -= 1 else @sharing.delete Thread.current @cv.broadcast end end end |
#yield_shares(purpose: nil, compatible: [], block_share: false) ⇒ Object
Temporarily give up all held Share locks while executing the supplied block, allowing any compatible
exclusive lock request to proceed.
170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 |
# File 'activesupport/lib/active_support/concurrency/share_lock.rb', line 170 def yield_shares(purpose: nil, compatible: [], block_share: false) loose_shares = previous_wait = nil synchronize do if loose_shares = @sharing.delete(Thread.current) if previous_wait = @waiting[Thread.current] purpose = nil unless purpose == previous_wait[0] compatible &= previous_wait[1] end compatible |= [false] unless block_share @waiting[Thread.current] = [purpose, compatible] end @cv.broadcast end begin yield ensure synchronize do wait_for(:yield_shares) { @exclusive_thread && @exclusive_thread != Thread.current } if previous_wait @waiting[Thread.current] = previous_wait else @waiting.delete Thread.current end @sharing[Thread.current] = loose_shares if loose_shares end end end |