… | |
… | |
1289 | } |
1289 | } |
1290 | |
1290 | |
1291 | # default implementation for ->signal |
1291 | # default implementation for ->signal |
1292 | |
1292 | |
1293 | our $HAVE_ASYNC_INTERRUPT; |
1293 | our $HAVE_ASYNC_INTERRUPT; |
|
|
1294 | |
|
|
1295 | sub _have_async_interrupt() { |
|
|
1296 | $HAVE_ASYNC_INTERRUPT = 1*(!$ENV{PERL_ANYEVENT_AVOID_ASYNC_INTERRUPT} |
|
|
1297 | && eval "use Async::Interrupt 1.0 (); 1") |
|
|
1298 | unless defined $HAVE_ASYNC_INTERRUPT; |
|
|
1299 | |
|
|
1300 | $HAVE_ASYNC_INTERRUPT |
|
|
1301 | } |
|
|
1302 | |
1294 | our ($SIGPIPE_R, $SIGPIPE_W, %SIG_CB, %SIG_EV, $SIG_IO); |
1303 | our ($SIGPIPE_R, $SIGPIPE_W, %SIG_CB, %SIG_EV, $SIG_IO); |
1295 | our (%SIG_ASY, %SIG_ASY_W); |
1304 | our (%SIG_ASY, %SIG_ASY_W); |
1296 | our ($SIG_COUNT, $SIG_TW); |
1305 | our ($SIG_COUNT, $SIG_TW); |
1297 | |
1306 | |
1298 | sub _signal_exec { |
1307 | sub _signal_exec { |
… | |
… | |
1325 | sub _sig_del { |
1334 | sub _sig_del { |
1326 | undef $SIG_TW |
1335 | undef $SIG_TW |
1327 | unless --$SIG_COUNT; |
1336 | unless --$SIG_COUNT; |
1328 | } |
1337 | } |
1329 | |
1338 | |
|
|
1339 | our %SIGNAME2NUM; |
|
|
1340 | our @SIGNUM2NAME; |
|
|
1341 | our $_sig_name_init; $_sig_name_init = sub { |
|
|
1342 | undef $_sig_name_init; |
|
|
1343 | |
|
|
1344 | if (_have_async_interrupt) { |
|
|
1345 | *sig2num = \&Async::Interrupt::sig2num; |
|
|
1346 | *sig2name = \&Async::Interrupt::sig2name; |
|
|
1347 | } else { |
|
|
1348 | require Config; |
|
|
1349 | |
|
|
1350 | @SIGNAME2NUM{ split ' ', $Config::Config{sig_name} } |
|
|
1351 | = split ' ', $Config::Config{sig_num}; |
|
|
1352 | @SIGNUM2NAME[values %SIGNAME2NUM] = keys %SIGNAME2NUM; |
|
|
1353 | |
|
|
1354 | *sig2num = sub($) { |
|
|
1355 | $_[0] > 0 ? shift : $SIGNAME2NUM{+shift} |
|
|
1356 | }; |
|
|
1357 | *sig2name = sub ($) { |
|
|
1358 | $_[0] > 0 ? $SIGNUM2NAME[+shift] : shift |
|
|
1359 | }; |
|
|
1360 | } |
|
|
1361 | }; |
|
|
1362 | |
|
|
1363 | sub sig2num ($) { &$_sig_name_init; &sig2num } |
|
|
1364 | sub sig2name($) { &$_sig_name_init; &sig2name } |
|
|
1365 | |
1330 | sub _signal { |
1366 | sub _signal { |
1331 | my (undef, %arg) = @_; |
1367 | my (undef, %arg) = @_; |
1332 | |
1368 | |
1333 | my $signal = uc $arg{signal} |
1369 | my $signal = uc $arg{signal} |
1334 | or Carp::croak "required option 'signal' is missing"; |
1370 | or Carp::croak "required option 'signal' is missing"; |
1335 | |
1371 | |
1336 | if ($HAVE_ASYNC_INTERRUPT) { |
1372 | if ($HAVE_ASYNC_INTERRUPT) { |
1337 | # async::interrupt |
1373 | # async::interrupt |
1338 | |
1374 | |
1339 | $signal = Async::Interrupt::sig2num ($signal); |
1375 | $signal = sig2num $signal; |
1340 | $SIG_CB{$signal}{$arg{cb}} = $arg{cb}; |
1376 | $SIG_CB{$signal}{$arg{cb}} = $arg{cb}; |
1341 | |
1377 | |
1342 | $SIG_ASY{$signal} ||= new Async::Interrupt |
1378 | $SIG_ASY{$signal} ||= new Async::Interrupt |
1343 | cb => sub { undef $SIG_EV{$signal} }, |
1379 | cb => sub { undef $SIG_EV{$signal} }, |
1344 | signal => $signal, |
1380 | signal => $signal, |
… | |
… | |
1348 | |
1384 | |
1349 | } else { |
1385 | } else { |
1350 | # pure perl |
1386 | # pure perl |
1351 | |
1387 | |
1352 | # AE::Util has been loaded in signal |
1388 | # AE::Util has been loaded in signal |
1353 | $signal = AnyEvent::Util::sig2name ($signal); |
1389 | $signal = sig2name $signal; |
1354 | $SIG_CB{$signal}{$arg{cb}} = $arg{cb}; |
1390 | $SIG_CB{$signal}{$arg{cb}} = $arg{cb}; |
1355 | |
1391 | |
1356 | $SIG{$signal} ||= sub { |
1392 | $SIG{$signal} ||= sub { |
1357 | local $!; |
1393 | local $!; |
1358 | syswrite $SIGPIPE_W, "\x00", 1 unless %SIG_EV; |
1394 | syswrite $SIGPIPE_W, "\x00", 1 unless %SIG_EV; |
… | |
… | |
1367 | bless [$signal, $arg{cb}], "AnyEvent::Base::signal" |
1403 | bless [$signal, $arg{cb}], "AnyEvent::Base::signal" |
1368 | } |
1404 | } |
1369 | |
1405 | |
1370 | sub signal { |
1406 | sub signal { |
1371 | # probe for availability of Async::Interrupt |
1407 | # probe for availability of Async::Interrupt |
1372 | if (!$ENV{PERL_ANYEVENT_AVOID_ASYNC_INTERRUPT} && eval "use Async::Interrupt 1.0 (); 1") { |
1408 | if (_have_async_interrupt) { |
1373 | warn "AnyEvent: using Async::Interrupt for race-free signal handling.\n" if $VERBOSE >= 8; |
1409 | warn "AnyEvent: using Async::Interrupt for race-free signal handling.\n" if $VERBOSE >= 8; |
1374 | |
1410 | |
1375 | $HAVE_ASYNC_INTERRUPT = 1; |
|
|
1376 | $SIGPIPE_R = new Async::Interrupt::EventPipe; |
1411 | $SIGPIPE_R = new Async::Interrupt::EventPipe; |
1377 | $SIG_IO = AnyEvent->io (fh => $SIGPIPE_R->fileno, poll => "r", cb => \&_signal_exec); |
1412 | $SIG_IO = AnyEvent->io (fh => $SIGPIPE_R->fileno, poll => "r", cb => \&_signal_exec); |
1378 | |
1413 | |
1379 | } else { |
1414 | } else { |
1380 | warn "AnyEvent: using emulated perl signal handling with latency timer.\n" if $VERBOSE >= 8; |
1415 | warn "AnyEvent: using emulated perl signal handling with latency timer.\n" if $VERBOSE >= 8; |