You want to read from or write to a filehandle without the system blocking your process until the program, file, socket, or device at the other end is ready. This is desired less often of regular files than of special files.
Open the file with sysopen,
and specify the O_NONBLOCK
option:
use Fcntl; sysopen(MODEM, "/dev/cua0", O_NONBLOCK|O_RDWR) or die "Can't open modem: $!\n";
If you already have a filehandle, use fcntl
to change the flags:
use Fcntl; $flags = ''; fcntl(HANDLE, F_GETFL, $flags) or die "Couldn't get flags for HANDLE : $!\n"; $flags |= O_NONBLOCK; fcntl(HANDLE, F_SETFL, $flags) or die "Couldn't set flags for HANDLE: $!\n";
Once a filehandle is set for non-blocking I/O, the sysread
or syswrite
calls that would block will instead return undef
and set $!
to EAGAIN:
use POSIX qw(:errno_h); $rv = syswrite(HANDLE, $buffer, length $buffer); if (!defined($rv) && $! == EAGAIN) { # would block } elsif ($rv != length $buffer) { # incomplete write } else { # successfully wrote } $rv = sysread(HANDLE, $buffer, $BUFSIZ); if (!defined($rv) && $! == EAGAIN) { # would block } else { # successfully read $rv bytes from HANDLE }
The O_NONBLOCK
constant is part of the POSIX standard, so most machines should support it. We use the POSIX module to get the numeric value for the error EAGAIN.
The sysopen
and fcntl
functions in perlfunc (1) and in Chapter 3 of Programming Perl; the documentation for the standard POSIX module; your system's open (2) and fcntl (2) manpages; Recipe 7.13; Recipe 7.15
Copyright © 2002 O'Reilly & Associates. All rights reserved.