blob: 9832e3125e25dedc865d458c579cdddaf48e8e29 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
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
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
|
require_relative '../../spec_helper'
describe "Mutex#sleep" do
describe "when not locked by the current thread" do
it "raises a ThreadError" do
m = Mutex.new
-> { m.sleep }.should raise_error(ThreadError)
end
it "raises an ArgumentError if passed a negative duration" do
m = Mutex.new
-> { m.sleep(-0.1) }.should raise_error(ArgumentError)
-> { m.sleep(-1) }.should raise_error(ArgumentError)
end
end
it "raises an ArgumentError if passed a negative duration" do
m = Mutex.new
m.lock
-> { m.sleep(-0.1) }.should raise_error(ArgumentError)
-> { m.sleep(-1) }.should raise_error(ArgumentError)
end
it "pauses execution for approximately the duration requested" do
m = Mutex.new
m.lock
duration = 0.001
start = Process.clock_gettime(Process::CLOCK_MONOTONIC)
m.sleep duration
now = Process.clock_gettime(Process::CLOCK_MONOTONIC)
(now - start).should >= 0
(now - start).should < (duration + TIME_TOLERANCE)
end
it "unlocks the mutex while sleeping" do
m = Mutex.new
locked = false
th = Thread.new { m.lock; locked = true; m.sleep }
Thread.pass until locked
Thread.pass until th.stop?
m.locked?.should be_false
th.run
th.join
end
it "relocks the mutex when woken" do
m = Mutex.new
m.lock
m.sleep(0.001)
m.locked?.should be_true
end
it "relocks the mutex when woken by an exception being raised" do
m = Mutex.new
locked = false
th = Thread.new do
m.lock
locked = true
begin
m.sleep
rescue Exception
m.locked?
end
end
Thread.pass until locked
Thread.pass until th.stop?
th.raise(Exception)
th.value.should be_true
end
it "returns the rounded number of seconds asleep" do
m = Mutex.new
locked = false
th = Thread.start do
m.lock
locked = true
m.sleep
end
Thread.pass until locked
Thread.pass until th.stop?
th.wakeup
th.value.should be_kind_of(Integer)
end
it "wakes up when requesting sleep times near or equal to zero" do
times = []
val = 1
# power of two divisor so we eventually get near zero
loop do
val = val / 16.0
times << val
break if val == 0.0
end
m = Mutex.new
m.lock
times.each do |time|
# just testing that sleep completes
-> {m.sleep(time)}.should_not raise_error
end
end
end
|