# NAME IO::SigGuard - SA\_RESTART in pure Perl # SYNOPSIS IO::SigGuard::sysread( $fh, $buf, $size ); IO::SigGuard::sysread( $fh, $buf, $size, $offset ); IO::SigGuard::syswrite( $fh, $buf ); IO::SigGuard::syswrite( $fh, $buf, $len ); IO::SigGuard::syswrite( $fh, $buf, $len, $offset ); IO::SigGuard::send( $fh, $msg, $flags ); IO::SigGuard::send( $fh, $msg, $flags, $to ); IO::SigGuard::select( $read, $write, $exc, $timeout ); # DESCRIPTION `perldoc perlipc` describes how Perl versions from 5.8.0 onward disable the OS’s SA\_RESTART flag when installing Perl signal handlers. This module imitates that pattern in pure Perl: it does an automatic restart when a signal interrupts an operation so you can avoid the generally-useless EINTR error when using `sysread()`, `syswrite()`, and `select()`. For this to work, whatever signal handler you implement will need to break out of this module, probably via either `die()` or `exit()`. # ABOUT `sysread()` and `syswrite()` Other than that you’ll never see EINTR and that there are no function prototypes used (i.e., you need parentheses on all invocations), `sysread()` and `syswrite()` work exactly the same as Perl’s equivalent built-ins. # LAZY-LOADING As of version 0.13 this module’s functions lazy-load by default. To have functionality loaded at compile time give the function name to the import logic, e.g.: use IO::SigGuard qw(send recv); # ABOUT `select()` To handle EINTR, `IO::SigGuard::select()` has to subtract the elapsed time from the given timeout then repeat the internal `select()`. Because the `select()` built-in’s `$timeleft` return is not reliable across all platforms, we have to compute the elapsed time ourselves. By default the only means of doing this is the `time()` built-in, which can only measure individual seconds. This works, but there are two ways to make it more accurate: - Have [Time::HiRes](https://metacpan.org/pod/Time::HiRes) loaded, and `IO::SigGuard::select()` will use that module rather than the `time()` built-in. - Set `$IO::SigGuard::TIME_CR` to a compatible code reference. This is useful, e.g., if you have your own logic to do the equivalent of [Time::HiRes](https://metacpan.org/pod/Time::HiRes)—for example, in Linux you may prefer to call the `gettimeofday` system call directly from Perl to avoid [Time::HiRes](https://metacpan.org/pod/Time::HiRes)’s XS overhead. In scalar contact, `IO::SigGuard::select()` is a drop-in replacement for Perl’s 4-argument built-in. In list context, there may be discrepancies re the `$timeleft` value that Perl returns from a call to `select`. As per Perl’s documentation this value is generally not reliable anyway, though, so that shouldn’t be a big deal. In fact, on systems like MacOS where the built-in’s `$timeleft` is completely useless, IO::SigGuard’s return is actually **better** since it does provide at least a rough estimate of how much of the given timeout value is left. See `perlport` for portability notes for `select`. # TODO This pattern could probably be extended to other system calls that can receive EINTR. I’ll consider adding new calls as requested. # REPOSITORY [https://github.com/FGasper/p5-IO-SigGuard](https://github.com/FGasper/p5-IO-SigGuard) # AUTHOR Felipe Gasper (FELIPE) … with special thanks to Mario Roy (MARIOROY) for extra testing and a few fixes/improvements. # COPYRIGHT Copyright 2017 by [Gasper Software Consulting](http://gaspersoftware.com) # LICENSE This distribution is released under the same license as Perl.