summaryrefslogtreecommitdiff
path: root/ext
diff options
context:
space:
mode:
authorSteve Hay <steve.m.hay@googlemail.com>2012-09-12 23:36:41 +0100
committerSteve Hay <steve.m.hay@googlemail.com>2012-09-13 21:09:17 +0100
commit8700fd387632099e570555f059e008cb2dff2759 (patch)
treed502688c5dd64eaa32e9b613ff30685a7747236e /ext
parent650f067c54a3baf4a72cb6953bd10c90abb28bd2 (diff)
downloadperl-8700fd387632099e570555f059e008cb2dff2759.tar.gz
Avoid POSIX::close when closing files by descriptor in IPC::Open3
Closing a file descriptor with POSIX::close bypasses PerlIO's ref-counting of file descriptors and leads to MSVC++'s invalid parameter handler being triggered when the PerlIO stream is closed later because that attempts to close the underlying file descriptor again, but it's already closed. So instead, we effectively fdopen() a new PerlIO stream and then close it again to effect the closure of the file descriptor.
Diffstat (limited to 'ext')
-rw-r--r--ext/IPC-Open3/lib/IPC/Open3.pm12
1 files changed, 7 insertions, 5 deletions
diff --git a/ext/IPC-Open3/lib/IPC/Open3.pm b/ext/IPC-Open3/lib/IPC/Open3.pm
index 31c68afc5a..29612af7c0 100644
--- a/ext/IPC-Open3/lib/IPC/Open3.pm
+++ b/ext/IPC-Open3/lib/IPC/Open3.pm
@@ -9,7 +9,7 @@ require Exporter;
use Carp;
use Symbol qw(gensym qualify);
-$VERSION = '1.12';
+$VERSION = '1.13';
@ISA = qw(Exporter);
@EXPORT = qw(open3);
@@ -163,7 +163,9 @@ sub xopen {
}
sub xclose {
- $_[0] =~ /\A=?(\d+)\z/ ? eval { require POSIX; POSIX::close($1); } : close $_[0]
+ $_[0] =~ /\A=?(\d+)\z/
+ ? do { my $fh; open($fh, $_[1] . '&=' . $1) and close($fh); }
+ : close $_[0]
or croak "$Me: close($_[0]) failed: $!";
}
@@ -264,7 +266,7 @@ sub _open3 {
xopen $_->{handle}, $_->{mode} . '&', $_->{parent}
if fileno $_->{handle} != xfileno($_->{parent});
} else {
- xclose $_->{parent};
+ xclose $_->{parent}, $_->{mode};
xopen $_->{handle}, $_->{mode} . '&=',
fileno $_->{open_as};
}
@@ -331,12 +333,12 @@ sub _open3 {
foreach (@handles) {
next if $_->{dup} or $_->{dup_of_out};
- xclose $_->{open_as};
+ xclose $_->{open_as}, $_->{mode};
}
# If the write handle is a dup give it away entirely, close my copy
# of it.
- xclose $handles[0]{parent} if $handles[0]{dup};
+ xclose $handles[0]{parent}, $handles[0]{mode} if $handles[0]{dup};
select((select($handles[0]{parent}), $| = 1)[0]); # unbuffer pipe
$kidpid;