PublicInbox::CmdIPC4->can('send_cmd4');
};
-sub wq_set_recv_modes {
- my ($self, @modes) = @_;
- $self->{-wq_recv_modes} = \@modes;
-}
-
sub _get_rec ($) {
my ($r) = @_;
defined(my $len = <$r>) or return;
my ($self, $s2, $len, $full_stream) = @_;
my @fds = $recv_cmd->($s2, my $buf, $len);
my $n = length($buf // '') or return;
- my @m = @{$self->{-wq_recv_modes} // [qw( +<&= >&= >&= )]};
my $nfd = 0;
for my $fd (@fds) {
- my $mode = shift(@m);
- if (open(my $cmdfh, $mode, $fd)) {
+ if (open(my $cmdfh, '+<&=', $fd)) {
$self->{$nfd++} = $cmdfh;
$cmdfh->autoflush(1);
} else {
- die "$$ open($mode$fd) (FD:$nfd): $!";
+ die "$$ open(+<&=$fd) (FD:$nfd): $!";
}
}
while ($full_stream && $n < $len) {
return send($sock, 'timed out waiting to recv FDs', MSG_EOR);
my @fds = $recv_cmd->($sock, my $buf, 4096 * 33); # >MAX_ARG_STRLEN
if (scalar(@fds) == 4) {
- my $i = 0;
- for my $rdr (qw(<&= >&= >&= <&=)) {
+ for my $i (0..3) {
my $fd = shift(@fds);
- open($self->{$i++}, $rdr, $fd) and next;
- send($sock, "open($rdr$fd) (FD=$i): $!", MSG_EOR);
+ open($self->{$i}, '+<&=', $fd) and next;
+ send($sock, "open(+<&=$fd) (FD=$i): $!", MSG_EOR);
}
} else {
return send($sock, "recv_cmd failed: $!", MSG_EOR);
sub dup_src ($) {
my ($in) = @_;
- # fileno needed because wq_set_recv_modes only used ">&=" for {1}
- # and Perl blindly trusts that to reject the '+' (readability flag)
- open my $dup, '+>>&=', fileno($in) or die "dup: $!";
+ open my $dup, '+>>&', $in or die "dup: $!";
$dup;
}
sub ipc_atfork_prepare {
my ($self) = @_;
- # (done_wr, stdout|mbox, stderr, 3: sock, 4: each_smsg_done_wr)
- $self->wq_set_recv_modes(qw[+<&= >&= >&= +<&= >&=]);
+ # FDs: (done_wr, stdout|mbox, stderr, 3: sock, 4: each_smsg_done_wr)
$self->SUPER::ipc_atfork_prepare; # PublicInbox::IPC
}
require PublicInbox::MboxReader;
require IO::Uncompress::Gunzip;
}
- # (0: done_wr, 1: stdout|mbox, 2: stderr,
- # 3: sock, 4: $l2m->{-wq_s1}, 5: $startq)
- $self->wq_set_recv_modes(qw[+<&= >&= >&= +<&= +<&= <&=]);
+ # FDS: (0: done_wr, 1: stdout|mbox, 2: stderr,
+ # 3: sock, 4: $l2m->{-wq_s1}, 5: $startq)
$self->SUPER::ipc_atfork_prepare; # PublicInbox::IPC
}
sub exec_cmd {
my ($fds, $argc, @argv) = @_;
- my @m = (*STDIN{IO}, '<&=', *STDOUT{IO}, '>&=', *STDERR{IO}, '>&=');
+ my @old = (*STDIN{IO}, *STDOUT{IO}, *STDERR{IO});
my @rdr;
for my $fd (@$fds) {
- my ($old_io, $mode) = splice(@m, 0, 2);
- open(my $tmpfh, $mode, $fd) or die "open $mode$fd: $!";
- push @rdr, $old_io, $mode, $tmpfh;
+ open(my $tmpfh, '+<&=', $fd) or die "open +<&=$fd: $!";
+ push @rdr, shift(@old), $tmpfh;
}
require POSIX; # WNOHANG
$SIG{CHLD} = \&sigchld;
my $pid = fork // die "fork: $!";
if ($pid == 0) {
my %env = map { split(/=/, $_, 2) } splice(@argv, $argc);
- while (my ($old_io, $mode, $tmpfh) = splice(@rdr, 0, 3)) {
- open $old_io, $mode, $tmpfh or die "open $mode: $!";
+ while (my ($old_io, $tmpfh) = splice(@rdr, 0, 2)) {
+ open $old_io, '+<&', $tmpfh or die "open +<&=: $!";
}
%ENV = (%ENV, %env);
exec(@argv);
$ipc->ipc_worker_stop; # idempotent
# work queues
-$ipc->wq_set_recv_modes(qw( +>&= >&= >&= ));
pipe(my ($ra, $wa)) or BAIL_OUT $!;
pipe(my ($rb, $wb)) or BAIL_OUT $!;
pipe(my ($rc, $wc)) or BAIL_OUT $!;