# frozen_string_literal: truerequire"monitor"moduleActiveSupportmoduleConcurrency# A share/exclusive lock, otherwise known as a read/write lock.## https://en.wikipedia.org/wiki/Readers%E2%80%93writer_lockclassShareLockincludeMonitorMixin# 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.defraw_state# :nodoc:synchronizedothreads=@sleeping.keys|@sharing.keys|@waiting.keysthreads|=[@exclusive_thread]if@exclusive_threaddata={}threads.eachdo|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.yielddataendenddefinitializesuper()@cv=new_cond@sharing=Hash.new(0)@waiting={}@sleeping={}@exclusive_thread=nil@exclusive_depth=0end# 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.defstart_exclusive(purpose: nil,compatible: [],no_wait: false)synchronizedounless@exclusive_thread==Thread.currentifbusy_for_exclusive?(purpose)returnfalseifno_waityield_shares(purpose: purpose,compatible: compatible,block_share: true)dowait_for(:start_exclusive){busy_for_exclusive?(purpose)}endend@exclusive_thread=Thread.currentend@exclusive_depth+=1trueendend# Relinquish the exclusive lock. Must only be called by the thread# that called start_exclusive (and currently holds the lock).defstop_exclusive(compatible: [])synchronizedoraise"invalid unlock"if@exclusive_thread!=Thread.current@exclusive_depth-=1if@exclusive_depth==0@exclusive_thread=nilifeligible_waiters?(compatible)yield_shares(compatible: compatible,block_share: true)dowait_for(:stop_exclusive){@exclusive_thread||eligible_waiters?(compatible)}endend@cv.broadcastendendenddefstart_sharingsynchronizedoif@sharing[Thread.current]>0||@exclusive_thread==Thread.current# We already hold a lock; nothing to wait forelsif@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 waywait_for(:start_sharing){@exclusive_thread}else# This is an initial / outermost share call: any outstanding# requests for an exclusive lock get to go firstwait_for(:start_sharing){busy_for_sharing?(false)}end@sharing[Thread.current]+=1endenddefstop_sharingsynchronizedoif@sharing[Thread.current]>1@sharing[Thread.current]-=1else@sharing.deleteThread.current@cv.broadcastendendend# 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.defexclusive(purpose: nil,compatible: [],after_compatible: [],no_wait: false)ifstart_exclusive(purpose: purpose,compatible: compatible,no_wait: no_wait)beginyieldensurestop_exclusive(compatible: after_compatible)endendend# Execute the supplied block while holding the Share lock.defsharingstart_sharingbeginyieldensurestop_sharingendend# Temporarily give up all held Share locks while executing the# supplied block, allowing any +compatible+ exclusive lock request# to proceed.defyield_shares(purpose: nil,compatible: [],block_share: false)loose_shares=previous_wait=nilsynchronizedoifloose_shares=@sharing.delete(Thread.current)ifprevious_wait=@waiting[Thread.current]purpose=nilunlesspurpose==previous_wait[0]compatible&=previous_wait[1]endcompatible|=[false]unlessblock_share@waiting[Thread.current]=[purpose,compatible]end@cv.broadcastendbeginyieldensuresynchronizedowait_for(:yield_shares){@exclusive_thread&&@exclusive_thread!=Thread.current}ifprevious_wait@waiting[Thread.current]=previous_waitelse@waiting.deleteThread.currentend@sharing[Thread.current]=loose_sharesifloose_sharesendendendprivate# Must be called within synchronizedefbusy_for_exclusive?(purpose)busy_for_sharing?(purpose)||@sharing.size>(@sharing[Thread.current]>0?1:0)enddefbusy_for_sharing?(purpose)(@exclusive_thread&&@exclusive_thread!=Thread.current)||@waiting.any?{|t,(_,c)|t!=Thread.current&&!c.include?(purpose)}enddefeligible_waiters?(compatible)@waiting.any?{|t,(p,_)|compatible.include?(p)&&@waiting.all?{|t2,(_,c2)|t==t2||c2.include?(p)}}enddefwait_for(method,&block)@sleeping[Thread.current]=method@cv.wait_while(&block)ensure@sleeping.deleteThread.currentendendendend