ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/Coro/Coro/Semaphore.pm
(Generate patch)

Comparing Coro/Coro/Semaphore.pm (file contents):
Revision 1.4 by root, Tue Jul 10 21:19:47 2001 UTC vs.
Revision 1.25 by root, Tue Nov 27 03:11:48 2001 UTC

14 14
15 $sig->up; 15 $sig->up;
16 16
17=head1 DESCRIPTION 17=head1 DESCRIPTION
18 18
19This module implements counting semaphores. You can initialize a mutex
20with any level of parallel users, that is, you can intialize a sempahore
21that can be C<down>ed more than once until it blocks. There is no owner
22associated with semaphores, so one coroutine can C<down> it while another
23can C<up> it.
24
25Counting semaphores are typically used to coordinate access to
26resources, with the semaphore count initialized to the number of free
27resources. Coroutines then increment the count when resources are added
28and decrement the count when resources are removed.
29
19=over 4 30=over 4
20 31
21=cut 32=cut
22 33
23package Coro::Semaphore; 34package Coro::Semaphore;
24 35
25use Coro::Process (); 36no warnings qw(uninitialized);
26 37
27$VERSION = 0.01; 38use Coro ();
28 39
40$VERSION = 0.52;
41
29=item new [inital count, default zero] 42=item new [inital count]
30 43
31Creates a new sempahore object with the given initial lock count. The 44Creates a new sempahore object with the given initial lock count. The
32default lock count is 1, which means it is unlocked by default. 45default lock count is 1, which means it is unlocked by default. Zero (or
46negative values) are also allowed, in which case the semaphore is locked
47by default.
33 48
34=cut 49=cut
35 50
36sub new { 51sub new {
37 bless [defined $_[1] ? $_[1] : 1], $_[0]; 52 bless [defined $_[1] ? $_[1] : 1], $_[0];
40=item $sem->down 55=item $sem->down
41 56
42Decrement the counter, therefore "locking" the semaphore. This method 57Decrement the counter, therefore "locking" the semaphore. This method
43waits until the semaphore is available if the counter is zero. 58waits until the semaphore is available if the counter is zero.
44 59
60=item $status = $sem->timed_down($timeout)
61
62Like C<down>, but returns false if semaphore couldn't be acquired within
63$timeout seconds, otherwise true.
64
45=cut 65=cut
46 66
47sub down { 67sub down {
48 my $self = shift;
49 while ($self->[0] <= 0) { 68 while ($_[0][0] <= 0) {
50 push @{$self->[1]}, $Coro::Process::current; 69 push @{$_[0][1]}, $Coro::current;
51 Coro::Process::schedule; 70 Coro::schedule;
52 } 71 }
53 --$self->[0]; 72 --$_[0][0];
73}
74
75sub timed_down {
76 require Coro::Timer;
77 my $timeout = Coro::Timer::timeout($_[1]);
78
79 while ($_[0][0] <= 0) {
80 push @{$_[0][1]}, $Coro::current;
81 Coro::schedule;
82 $timeout and return;
83 }
84
85 --$_[0][0];
86 return 1;
54} 87}
55 88
56=item $sem->up 89=item $sem->up
57 90
58Unlock the semaphore again. 91Unlock the semaphore again.
59 92
60=cut 93=cut
61 94
62sub up { 95sub up {
63 my $self = shift;
64 if (++$self->[0] > 0) { 96 if (++$_[0][0] > 0) {
65 (shift @{$self->[1]})->ready if @{$self->[1]}; 97 (shift @{$_[0][1]})->ready if @{$_[0][1]};
66 } 98 }
67} 99}
68 100
69=item $sem->try 101=item $sem->try
70 102
72otherwise return false and leave the semaphore unchanged. 104otherwise return false and leave the semaphore unchanged.
73 105
74=cut 106=cut
75 107
76sub try { 108sub try {
77 my $self = shift;
78 if ($self->[0] > 0) { 109 if ($_[0][0] > 0) {
79 --$self->[0]; 110 --$_[0][0];
80 return 1; 111 return 1;
81 } else { 112 } else {
82 return 0; 113 return 0;
83 } 114 }
115}
116
117=item $sem->waiters
118
119In scalar context, returns the number of coroutines waiting for this
120semaphore.
121
122=cut
123
124sub waiters {
125 @{$_[0][1]};
126}
127
128=item $guard = $sem->guard
129
130This method calls C<down> and then creates a guard object. When the guard
131object is destroyed it automatically calls C<up>.
132
133=item $guard = $sem->timed_guard($timeout)
134
135Like C<guard>, but returns undef if semaphore couldn't be acquired within
136$timeout seconds, otherwise the guard object.
137
138=cut
139
140sub guard {
141 &down;
142 # double indirection because bless works on the referenced
143 # object, not (only) on the reference itself.
144 bless \\$_[0], Coro::Semaphore::guard;
145}
146
147sub timed_guard {
148 &timed_down
149 ? bless \\$_[0], Coro::Semaphore::guard
150 : ();
151}
152
153sub Coro::Semaphore::guard::DESTROY {
154 &up(${${$_[0]}});
84} 155}
85 156
861; 1571;
87 158
88=back 159=back

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines