| libalarm documentation | Contained in the libalarm distribution. |
Alarm::Concurrent - Allow multiple, concurrent alarms.
This module is an attempt to enhance Perl's built-in
alarm/$SIG{ALRM} functionality.
This function, and its associated signal handler, allow you to arrange for your program to receive a SIGALRM signal, which you can then catch and deal with appropriately.
Unfortunately, due to the nature of the design of these signals (at the OS level), you can only have one alarm and handler active at any given time. That's where this module comes in.
This module allows you to define multiple alarms, each with an associated handler. These alarms are sequenced (in a queue) but concurrent, which means that their order is preserved but they always go off as their set time expires, regardless of the state of the other alarms. (If you'd like to have the alarms only go off in the order you set them, see Alarm::Queued.)
To set an alarm, call the setalarm() function with the
set time of the alarm and a reference to the subroutine
to be called when the alarm goes off. You can then go on
with your program and the alarm will be called after the
set time has passed.
It is also possible to set an alarm that does
not have a handler associated with it using
Alarm::Concurrent::alarm(). (This function can also
be imported into your namespace, in which case it will
replace Perl's built-in alarm for your package only.)
If an alarm that does not have a handler associated
with it goes off, the default handler, pointed to by
$Alarm::Concurrent::DEFAULT_HANLDER, is called. You can
change the default handler by assigning to this variable.
The default $Alarm::Concurrent::DEFAULT_HANDLER simply
dies with the message "Alarm clock!\n".
No methods are exported by default but you can import any of the functions in the FUNCTIONS|"FUNCTIONS" section.
You can also import the special tag :ALL which will import
all the functions in the FUNCTIONS|"FUNCTIONS" section
(except Alarm::Concurrent::restore()).
If you import the special tag :OVERRIDE, this module
will override Perl's built-in alarm function for
every namespace and it will take over Perl's magic
%SIG variable, changing any attempts to read or
write $SIG{ALRM} into calls to gethandler() and
sethandler(), respectively (reading and writing to
other keys in %SIG is unaffected).
This can be useful when you are calling code that tries to
set its own alarm "the old fashioned way." It can also,
however, be dangerous. Overriding alarm is documented
and should be stable but taking over %SIG is more risky
(see CAVEATS|"CAVEATS").
Note that if you do not override alarm and
%SIG, any code you use that sets "legacy alarms"
will disable all of your concurrent alarms. You can
call Alarm::Concurrent::restore() to reinstall the
Alarm::Concurrent handler. This function can not be
imported.
The following functions are available for use.
Sets a new alarm and associates a handler with it. The handler is called when the specified number of seconds have elapsed. See DESCRIPTION|"DESCRIPTION" for more information.
Clears one or more previously set alarms. The index is an array index, with 0 being the currently active alarm and -1 being the last (most recent) alarm that was set.
INDEX defaults to 0 and LENGTH defaults to 1.
Creates a new alarm with no handler. A handler can
later be set for it via sethandler() or $SIG{ALRM},
if overridden.
For the most part, this function behaves exactly like Perl's built-in alarm function, except that it sets up a concurrent alarm instead. Thus, each call to alarm does not disable previous alarms unless called with a set time of 0.
Calling alarm() with a set time of 0 will disable the
last alarm set.
If SECONDS is not specified, the value stored in $_
is used.
Sets a handler for the alarm found at INDEX in the queue. This is an array index, so negative values may be used to indicate position relative to the end of the queue.
If INDEX is not specified, the handler is set for the
last alarm in the queue that doesn't have one associated
with it. This means that if you set multiple alarms
using alarm(), you should arrange their respective
sethandler()'s in the opposite order.
Returns the handler for the alarm found at INDEX in the queue. This is an array index, so negative values may be used.
If INDEX is not specified, returns the handler for the currently active alarm.
This function reinstalls the Alarm::Concurrent alarm handler if it has been replaced by a "legacy alarm handler."
If FLAG is present and true, restore() will save the
current handler by setting it as a new concurrent alarm (as
if you had called setalarm() for it).
This function may not be imported.
Note: Do not call this function if you have imported
the :OVERLOAD symbol. It can have unpredictable results.
%SIG is Perl magic and should probably not be messed
with, though I have not witnessed any problems in the
(admittedly limited) testing I've done. I would be
interested to hear from anyone who performs extensive
testing, with different versions of Perl, of the
reliability of doing this.
$SIG{ALRM}, the entire magic hash is usurped and any
other %SIG} accesses are simply passed through to the
original magic hash. This means that if there are any
problems, they will most likely affect all other signal
handlers you have defined, including $SIG{__WARN__}
and $SIG{__DIE__} and others.
:OVERRIDE
option, you do so at your own risk (and you'd better be
pretty damn sure of yourself, too). $DEFAULT_HANDLER simply dies with the
message "Alarm clock!\n". Written by Cory Johns (c) 2001.
| libalarm documentation | Contained in the libalarm distribution. |
package Alarm::Concurrent; $VERSION = 1.0; use strict;
# In case they want to take over $SIG{ALRM}. use Alarm::_TieSIG; use Carp; use Exporter; use vars qw( @ISA @EXPORT_OK %EXPORT_TAGS ); @ISA = qw(Exporter); @EXPORT_OK = qw( setalarm clearalarm alarm sethandler gethandler ); %EXPORT_TAGS = ( ALL => [@EXPORT_OK], ); # # Exporter doesn't allow hooks for handling # special tags. So, we have to do it ourselves. # sub import { my $thispkg = shift; # Look for and remove special :OVERRIDE tag. my $override = 0; @_ = grep { ($_ eq ':OVERLOAD') ? ($override = 1, 0) : 1 } @_; if($override) { $thispkg->export('CORE::GLOBAL', 'alarm'); Alarm::_TieSIG::tiesig(); # ALL YOUR %SIG ARE BELONG TO US!!! }; $thispkg->export_to_level(1, $thispkg, @_); # export the rest } # Called for an alarm with no defined handler. sub _default_handler { die "Alarm clock!\n"; } use vars '$DEFAULT_HANDLER'; $DEFAULT_HANDLER = \&_default_handler; # Overeridable. # # Each element of @ALARM_QUEUE should be a pointer # to an array containing exactly four elements: # # 0) The duration of the alarm in seconds # 1) The time at which the alarm was set # 2) A pointer to a subroutine that should be called # when the alarm goes off. # use vars qw( @ALARM_QUEUE $ACTIVE_ALARM ); @ALARM_QUEUE = (); $ACTIVE_ALARM = -1; # Install our alarm handler. (& is to avoid prototype warning) &restore(1); # Custom alarm handler. sub _alrm { return unless(@ALARM_QUEUE); # Call handler for this alarm and remove it from the queue. my $handler = splice(@ALARM_QUEUE, $ACTIVE_ALARM, 1)->[2]; $handler ||= $DEFAULT_HANDLER; $handler->(); $ACTIVE_ALARM = -1; # Have to use a C-style loop because we modify # the index variable inside the loop. :-( for(my $i = 0; $i < @ALARM_QUEUE; ++$i) { my $time_remaining = $ALARM_QUEUE[$i][1]+$ALARM_QUEUE[$i][0]-time; my $active_time; $active_time = $ALARM_QUEUE[$ACTIVE_ALARM][1]; $active_time += $ALARM_QUEUE[$ACTIVE_ALARM][0]; $active_time -= time; if($time_remaining <= 0) { # Note the -- on $i. When we splice out an alarm, all the ones # after it shift down by one. We have to account for this. $handler = splice(@ALARM_QUEUE, $i--, 1)->[2] || $DEFAULT_HANDLER; $handler->(); # Call handler for this alarm. } elsif($ACTIVE_ALARM == -1 or $time_remaining < $active_time) { $ACTIVE_ALARM = $i; CORE::alarm($time_remaining); } } } #********************************************************************#
sub setalarm($$) { my ($alarm, $code) = @_; unless(not defined($code) or UNIVERSAL::isa($code, 'CODE')) { croak("Alarm handler must be CODEREF"); } push( @ALARM_QUEUE, [ $alarm, time(), $code ] ); my $time_remaining; $time_remaining = $ALARM_QUEUE[$ACTIVE_ALARM][1]; $time_remaining += $ALARM_QUEUE[$ACTIVE_ALARM][0]; $time_remaining -= time; if($ACTIVE_ALARM == -1 or $alarm < $time_remaining) { $ACTIVE_ALARM = $#ALARM_QUEUE; CORE::alarm($alarm); } }
sub clearalarm(;$$) { my $index = shift || 0; my $length = shift || 1; splice @ALARM_QUEUE, $index, $length; if(($index < 0 ? $#ALARM_QUEUE+$index : $index) == $ACTIVE_ALARM) { $ACTIVE_ALARM = -1; CORE::alarm(0); } # Have to use a C-style loop because we modify the index # variable inside the loop. :-( for(my $i = 0; $i < @ALARM_QUEUE; ++$i) { my $time_remaining = $ALARM_QUEUE[$i][1]+$ALARM_QUEUE[$i][0]-time; my $active_time; $active_time = $ALARM_QUEUE[$ACTIVE_ALARM][1]; $active_time += $ALARM_QUEUE[$ACTIVE_ALARM][0]; $active_time -= time; if($time_remaining <= 0) { # Note the -- on $i. When we splice out an alarm, all the ones # after it shift down by one. We have to account for this. my $handler = splice(@ALARM_QUEUE,$i--,1)->[2]; $handler ||= $DEFAULT_HANDLER; $handler->(); # Call handler for this alarm. } elsif($ACTIVE_ALARM == -1 or $time_remaining < $active_time) { $ACTIVE_ALARM = $i; CORE::alarm($time_remaining); } } }
sub alarm(;$) { my $alarm = @_ ? shift : $_; if($alarm == 0) { clearalarm(-1); } else { push( @ALARM_QUEUE, [ $alarm, time(), undef ] ); my $time_remaining; $time_remaining = $ALARM_QUEUE[$ACTIVE_ALARM][1]; $time_remaining += $ALARM_QUEUE[$ACTIVE_ALARM][0]; $time_remaining -= time; if($ACTIVE_ALARM == -1 or $alarm < $time_remaining) { $ACTIVE_ALARM = $#ALARM_QUEUE; CORE::alarm($alarm); } } }
sub sethandler($;$) { unless(not defined($_[-1]) or UNIVERSAL::isa($_[-1], 'CODE')) { croak("Alarm handler must be CODEREF"); } if(@_ == 2) { $ALARM_QUEUE[$_[0]]->[2] = $_[1]; } else { foreach my $alarm (reverse @ALARM_QUEUE) { if(not defined $alarm->[2]) { $alarm->[2] = shift(); last; } } } }
sub gethandler(;$) { my $index = shift || $ACTIVE_ALARM; return( ($index < @ALARM_QUEUE and $index > -1) ? $ALARM_QUEUE[$index][2] : undef ); }
sub restore(;$) { return if(defined($SIG{ALRM}) and $SIG{ALRM} == \&_alrm); my $oldalrm = CORE::alarm(0); if($oldalrm and shift) { # Save legacy alarm. setalarm($oldalrm, $SIG{ALRM}); } # Install our alarm handler. $SIG{ALRM} = \&_alrm; }
1;