mirror of
https://github.com/Perl/perl5.git
synced 2026-01-29 19:04:29 +00:00
Add Changes, Makefile.PL and examples directory. Fixes this module for https://github.com/Perl/perl5/issues/20874
124 lines
3.0 KiB
Perl
Executable File
124 lines
3.0 KiB
Perl
Executable File
#!/usr/bin/perl
|
|
|
|
# Simplified example illustrating event handling and callback threads
|
|
|
|
# Callback threads register their queues with the event handler thread.
|
|
# Events are passed to the event handler via a queue.
|
|
# The event handler then disseminates the event to the appropriately
|
|
# registered thread.
|
|
|
|
use strict;
|
|
use warnings;
|
|
|
|
use threads;
|
|
use Thread::Queue;
|
|
|
|
MAIN:
|
|
{
|
|
# Queue for registering callbacks
|
|
my $regis_q = Thread::Queue->new();
|
|
|
|
# Queue for disseminating events
|
|
my $event_q = Thread::Queue->new();
|
|
|
|
# Create callback threads
|
|
threads->create('CallBack', 'USR1', $regis_q)->detach();
|
|
threads->create('CallBack', 'USR2', $regis_q)->detach();
|
|
threads->create('CallBack', 'HUP', $regis_q)->detach();
|
|
threads->create('CallBack', 'ALRM', $regis_q)->detach();
|
|
|
|
# Create event handler thread
|
|
threads->create('EventHandler', $regis_q, $event_q)->detach();
|
|
|
|
# Capture SIGUSR1 events
|
|
$SIG{'USR1'} = sub {
|
|
$event_q->enqueue('USR1'); # Send to event handler
|
|
};
|
|
|
|
# Capture SIGUSR1 events
|
|
$SIG{'USR2'} = sub {
|
|
$event_q->enqueue('USR2'); # Send to event handler
|
|
};
|
|
|
|
# Capture SIGHUP events
|
|
$SIG{'HUP'} = sub {
|
|
$event_q->enqueue('HUP'); # Send to event handler
|
|
};
|
|
|
|
# Capture SIGHUP events
|
|
$SIG{'ALRM'} = sub {
|
|
$event_q->enqueue('ALRM'); # Send to event handler
|
|
alarm(5); # Reset alarm
|
|
};
|
|
|
|
# Ready
|
|
print(<<_MSG_);
|
|
Send signals to PID = $$
|
|
(e.g., 'kill -USR1 $$')
|
|
Use ^C (or 'kill -INT $$') to terminate
|
|
_MSG_
|
|
|
|
# Set initial alarm
|
|
alarm(5);
|
|
|
|
# Just hang around
|
|
while (1) {
|
|
sleep(10);
|
|
}
|
|
}
|
|
|
|
### Subroutines ###
|
|
|
|
sub EventHandler
|
|
{
|
|
my ($regis_q, $event_q) = @_;
|
|
|
|
my %callbacks; # Registered callback queues
|
|
|
|
while (1) {
|
|
# Check for any registrations
|
|
while (my ($event_type, $q) = $regis_q->dequeue_nb(2)) {
|
|
if ($q) {
|
|
$callbacks{$event_type} = $q;
|
|
} else {
|
|
warn("BUG: Bad callback registration for event type $event_type\n");
|
|
}
|
|
}
|
|
|
|
# Wait for event
|
|
if (my $event = $event_q->dequeue()) {
|
|
# Send event to appropriate queue
|
|
if (exists($callbacks{$event})) {
|
|
$callbacks{$event}->enqueue($event);
|
|
} else {
|
|
warn("WARNING: No callback for event type $event\n");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
sub CallBack
|
|
{
|
|
my $event_type = shift; # The type of event I'm handling
|
|
my $regis_q = shift;
|
|
|
|
# Announce registration
|
|
my $tid = threads->tid();
|
|
print("Callback thread $tid registering for $event_type events\n");
|
|
|
|
# Register my queue for my type of event
|
|
my $q = Thread::Queue->new();
|
|
$regis_q->enqueue($event_type, $q);
|
|
|
|
# Process loop
|
|
while (1) {
|
|
# Wait for event callback
|
|
my $item = $q->dequeue();
|
|
# Process event
|
|
print("Callback thread $tid notified of $item event\n") if $item;
|
|
}
|
|
}
|
|
|
|
# EOF
|