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

git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/branches/v1_1r@15 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
183 lines
3.4 KiB
Ruby
183 lines
3.4 KiB
Ruby
#
|
||
# mutex_m.rb -
|
||
# $Release Version: 2.0$
|
||
# $Revision: 1.1.1.1 $
|
||
# $Date: 1998/01/16 04:05:49 $
|
||
# Original from mutex.rb
|
||
# by Keiju ISHITSUKA(SHL Japan Inc.)
|
||
#
|
||
# --
|
||
# Usage:
|
||
# require "mutex_m.rb"
|
||
# obj = Object.new
|
||
# obj.extend Mutex_m
|
||
# ...
|
||
# ¸å¤ÏMutex¤ÈƱ¤¸»È¤¤Êý
|
||
#
|
||
|
||
require "finalize"
|
||
|
||
module Mutex_m
|
||
def Mutex_m.extend_object(obj)
|
||
if Fixnum === obj or TRUE === obj or FALSE === obj or nil == obj
|
||
raise TypeError, "Mutex_m can't extend to this class(#{obj.type})"
|
||
else
|
||
begin
|
||
eval "class << obj
|
||
@mu_locked
|
||
end"
|
||
obj.extend(For_primitive_object)
|
||
rescue TypeError
|
||
obj.extend(For_general_object)
|
||
end
|
||
end
|
||
end
|
||
|
||
def mu_extended
|
||
unless (defined? locked? and
|
||
defined? lock and
|
||
defined? unlock and
|
||
defined? try_lock and
|
||
defined? synchronize)
|
||
eval "class << self
|
||
alias locked mu_locked?
|
||
alias lock mu_lock
|
||
alias unlock mu_unlock
|
||
alias try_lock mu_try_lock
|
||
alias synchronize mu_synchronize
|
||
end"
|
||
end
|
||
end
|
||
|
||
def mu_synchronize
|
||
begin
|
||
mu_lock
|
||
yield
|
||
ensure
|
||
mu_unlock
|
||
end
|
||
end
|
||
|
||
module For_general_object
|
||
include Mutex_m
|
||
|
||
def For_general_object.extend_object(obj)
|
||
super
|
||
obj.mu_extended
|
||
end
|
||
|
||
def mu_extended
|
||
super
|
||
@mu_waiting = []
|
||
@mu_locked = FALSE;
|
||
end
|
||
|
||
def mu_locked?
|
||
@mu_locked
|
||
end
|
||
|
||
def mu_try_lock
|
||
result = FALSE
|
||
Thread.critical = TRUE
|
||
unless @mu_locked
|
||
@mu_locked = TRUE
|
||
result = TRUE
|
||
end
|
||
Thread.critical = FALSE
|
||
result
|
||
end
|
||
|
||
def mu_lock
|
||
while (Thread.critical = TRUE; @mu_locked)
|
||
@mu_waiting.push Thread.current
|
||
Thread.stop
|
||
end
|
||
@mu_locked = TRUE
|
||
Thread.critical = FALSE
|
||
self
|
||
end
|
||
|
||
def mu_unlock
|
||
return unless @mu_locked
|
||
Thread.critical = TRUE
|
||
wait = @mu_waiting
|
||
@mu_waiting = []
|
||
@mu_locked = FALSE
|
||
Thread.critical = FALSE
|
||
for w in wait
|
||
w.run
|
||
end
|
||
self
|
||
end
|
||
|
||
end
|
||
|
||
module For_primitive_object
|
||
include Mutex_m
|
||
Mu_Locked = Hash.new
|
||
|
||
def For_primitive_object.extend_object(obj)
|
||
super
|
||
obj.mu_extended
|
||
Finalizer.add(obj, For_primitive_object, :mu_finalize)
|
||
end
|
||
|
||
def For_primitive_object.mu_finalize(id)
|
||
Thread.critical = TRUE
|
||
if wait = Mu_Locked.delete(id)
|
||
# wait == [] ¤È¤¤À¤± GC¤µ¤ì¤ë¤Î¤Ç, for w in wait ¤Ï°ÕÌ£¤Ê¤·.
|
||
Thread.critical = FALSE
|
||
for w in wait
|
||
w.run
|
||
end
|
||
else
|
||
Thread.critical = FALSE
|
||
end
|
||
self
|
||
end
|
||
|
||
def mu_locked?
|
||
Mu_Locked.key?(self.id)
|
||
end
|
||
|
||
def mu_try_lock
|
||
Thread.critical = TRUE
|
||
if Mu_Locked.key?(self.id)
|
||
ret = FALSE
|
||
else
|
||
Mu_Locked[self.id] = []
|
||
Finalizer.set(self, For_primitive_object, :mu_delete_Locked)
|
||
ret = TRUE
|
||
end
|
||
Thread.critical = FALSE
|
||
ret
|
||
end
|
||
|
||
def mu_lock
|
||
while (Thread.critical = TRUE; w = Mu_Locked[self.id])
|
||
w.push Thread.current
|
||
Thread.stop
|
||
end
|
||
Mu_Locked[self.id] = []
|
||
Finalizer.add(self, For_primitive_object, :mu_delete_Locked)
|
||
Thread.critical = FALSE
|
||
self
|
||
end
|
||
|
||
def mu_unlock
|
||
Thread.critical = TRUE
|
||
if wait = Mu_Locked.delete(self.id)
|
||
Finalizer.delete(self, For_primitive_object, :mu_finalize)
|
||
Thread.critical = FALSE
|
||
for w in wait
|
||
w.run
|
||
end
|
||
else
|
||
Thread.critical = FALSE
|
||
end
|
||
self
|
||
end
|
||
end
|
||
end
|
||
|
||
|