X-Git-Url: http://www.git.stargrave.org/?a=blobdiff_plain;f=lib%2FPublicInbox%2FLeiPmdir.pm;fp=lib%2FPublicInbox%2FLeiPmdir.pm;h=5efb012e4e8c69ad146f52d59a862999e29da83e;hb=10b523eb017162240b1ac3647f8dcbbf2be348a7;hp=0000000000000000000000000000000000000000;hpb=ba34a69490dce6ea3ba85ee5416b6590fa0c0a39;p=public-inbox.git diff --git a/lib/PublicInbox/LeiPmdir.pm b/lib/PublicInbox/LeiPmdir.pm new file mode 100644 index 00000000..5efb012e --- /dev/null +++ b/lib/PublicInbox/LeiPmdir.pm @@ -0,0 +1,67 @@ +# Copyright (C) 2021 all contributors +# License: AGPL-3.0+ + +# WQ worker for dealing with parallel Maildir reads; +# this does NOT use the {shard_info} field of LeiToMail +# (and we may remove {shard_info}) +# WQ key: {pmd} +package PublicInbox::LeiPmdir; +use strict; +use v5.10.1; +use parent qw(PublicInbox::IPC); + +sub new { + my ($cls, $lei, $ipt) = @_; + my $self = bless { -wq_ident => 'lei Maildir worker' }, $cls; + my $jobs = $lei->{opt}->{jobs}; + $jobs =~ /\A[0-9]+,([0-9]+)\z/ and $jobs = $1; + my $nproc = $jobs // do { + # untested with >=4 CPUs, though I suspect I/O latency + # of SATA SSD storage will make >=4 processes unnecessary, + # here. NVMe users may wish to use '-j' + my $n = $self->detect_nproc; + $n = 4 if $n > 4; + }; + my ($op_c, $ops) = $lei->workers_start($self, $nproc, + undef, { ipt => $ipt }); # LeiInput subclass + $op_c->{ops} = $ops; # for PktOp->event_step + $lei->{pmd} = $self; +} + +sub ipc_atfork_child { + my ($self) = @_; + my $lei = $self->{lei}; + $lei->_lei_atfork_child; + my $ipt = $self->{ipt} // die 'BUG: no self->{ipt}'; + $ipt->{lei} = $lei; + $ipt->{sto} = $lei->{sto} // die 'BUG: no lei->{sto}'; + $ipt->{lse} = $ipt->{sto}->search; + $ipt->{over} = $ipt->{lse}->over; + $ipt->{-lms_ro} //= $ipt->{lse}->lms; # may be undef or '0' + $self->SUPER::ipc_atfork_child; +} + +sub each_mdir_fn { # maildir_each_file callback + my ($f, $self, @args) = @_; + $self->wq_io_do('mdir_iter', [], $f, @args); +} + +sub mdir_iter { # via wq_io_do + my ($self, $f, @args) = @_; + $self->{ipt}->pmdir_cb($f, @args); +} + +sub pmd_done_wait { + my ($arg, $pid) = @_; + my ($self, $lei) = @$arg; + my $wait = $lei->{sto}->ipc_do('done'); + $lei->can('wq_done_wait')->($arg, $pid); +} + +sub _lei_wq_eof { # EOF callback for main lei daemon + my ($lei) = @_; + my $pmd = delete $lei->{pmd} or return $lei->fail; + $pmd->wq_wait_old(\&pmd_done_wait, $lei); +} + +1;