]> Sergey Matveev's repositories - public-inbox.git/blobdiff - lib/PublicInbox/ProcessPipe.pm
No ext_urls
[public-inbox.git] / lib / PublicInbox / ProcessPipe.pm
index afbb048df86ea55f42ec61ede5c0e076913707c0..1bc792c4bada77676916b10f330e87d1c0d9b98f 100644 (file)
@@ -1,17 +1,31 @@
-# Copyright (C) 2016-2020 all contributors <meta@public-inbox.org>
+# Copyright (C) all contributors <meta@public-inbox.org>
 # License: AGPL-3.0+ <https://www.gnu.org/licenses/agpl-3.0.txt>
 
-# a tied handle for auto reaping of children tied to a pipe, see perltie(1)
+# a tied handle for auto reaping of children tied to a read-only pipe, see perltie(1)
+# DO NOT use this as-is for bidirectional pipes/sockets (e.g. in PublicInbox::Git),
+# both ends of the pipe must be at the same level of the Perl object hierarchy
+# to ensure orderly destruction.
 package PublicInbox::ProcessPipe;
-use strict;
-use v5.10.1;
-use PublicInbox::DS qw(dwaitpid);
+use v5.12;
+use PublicInbox::DS qw(awaitpid);
+
+sub waitcb { # awaitpid callback
+       my ($pid, $err_ref, $cb, @args) = @_;
+       $$err_ref = $?; # sets >{pp_chld_err} for _close
+       $cb->($pid, @args) if $cb;
+}
 
 sub TIEHANDLE {
-       my ($class, $pid, $fh, $cb, $arg) = @_;
-       bless { pid => $pid, fh => $fh, cb => $cb, arg => $arg }, $class;
+       my ($cls, $pid, $fh, @cb_arg) = @_;
+       my $self = bless { pid => $pid, fh => $fh, ppid => $$ }, $cls;
+       # we share $err (and not $self) with awaitpid to avoid a ref cycle
+       $self->{pp_chld_err} = \(my $err);
+       awaitpid($pid, \&waitcb, \$err, @cb_arg);
+       $self;
 }
 
+sub BINMODE { binmode(shift->{fh}) } # for IO::Uncompress::Gunzip
+
 sub READ { read($_[0]->{fh}, $_[1], $_[2], $_[3] || 0) }
 
 sub READLINE { readline($_[0]->{fh}) }
@@ -26,32 +40,33 @@ sub PRINT {
        print { $self->{fh} } @_;
 }
 
-sub adjust_ret { # dwaitpid callback
-       my ($retref, $pid) = @_;
-       $$retref = '' if $?
-}
+sub FILENO { fileno($_[0]->{fh}) }
 
-sub CLOSE {
-       my $fh = delete($_[0]->{fh});
-       my $ret = defined $fh ? close($fh) : '';
-       my ($pid, $cb, $arg) = delete @{$_[0]}{qw(pid cb arg)};
-       if (defined $pid) {
-               unless ($cb) {
-                       $cb = \&adjust_ret;
-                       $arg = \$ret;
-               }
-               dwaitpid $pid, $cb, $arg;
+sub _close ($;$) {
+       my ($self, $wait) = @_;
+       my ($fh, $pid) = delete(@$self{qw(fh pid)});
+       my $ret = defined($fh) ? close($fh) : '';
+       return $ret unless defined($pid) && $self->{ppid} == $$;
+       if ($wait) { # caller cares about the exit status:
+               # synchronous wait via defined(wantarray) on awaitpid:
+               defined(${$self->{pp_chld_err}}) or $wait = awaitpid($pid);
+               ($? = ${$self->{pp_chld_err}}) and $ret = '';
+       } else {
+               awaitpid($pid); # depends on $in_loop or not
        }
        $ret;
 }
 
-sub FILENO { fileno($_[0]->{fh}) }
+# if caller uses close(), assume they want to check $? immediately so
+# we'll waitpid() synchronously.  n.b. wantarray doesn't seem to
+# propagate `undef' down to tied methods, otherwise I'd rely on that.
+sub CLOSE { _close($_[0], 1) }
 
+# if relying on DESTROY, assume the caller doesn't care about $? and
+# we can let the event loop call waitpid() whenever it gets SIGCHLD
 sub DESTROY {
-       CLOSE(@_);
+       _close($_[0]);
        undef;
 }
 
-sub pid { $_[0]->{pid} }
-
 1;