about summary refs log tree commit homepage
diff options
context:
space:
mode:
authorEric Wong <e@80x24.org>2021-07-29 10:01:31 +0000
committerEric Wong <e@80x24.org>2021-08-04 10:04:23 +0000
commit7fc6e30aeab9925bece4bb00f88bb91af5646aa2 (patch)
treea97e16eb1d89fdfb1ba6552f7a5cf36318c10d0f
parentdd80dcaa1e46543893de533938a1651639f91f10 (diff)
downloadpublic-inbox-7fc6e30aeab9925bece4bb00f88bb91af5646aa2.tar.gz
Linux::Inotify2 2.3+ includes an ->fh method to give us the
ability to safely close an FD without hitting EBADF (and
automatically use FD_CLOEXEC).

We'll still need a new wrapper class (LI2Wrap) to handle it for
users of old versions, though.

Link: http://lists.schmorp.de/pipermail/perl/2021q3/thread.html
-rw-r--r--MANIFEST1
-rw-r--r--lib/PublicInbox/DirIdle.pm11
-rw-r--r--lib/PublicInbox/LEI.pm2
-rw-r--r--lib/PublicInbox/LI2Wrap.pm19
4 files changed, 32 insertions, 1 deletions
diff --git a/MANIFEST b/MANIFEST
index a3913501..fb9f16bf 100644
--- a/MANIFEST
+++ b/MANIFEST
@@ -197,6 +197,7 @@ lib/PublicInbox/InputPipe.pm
 lib/PublicInbox/Isearch.pm
 lib/PublicInbox/KQNotify.pm
 lib/PublicInbox/LEI.pm
+lib/PublicInbox/LI2Wrap.pm
 lib/PublicInbox/LeiALE.pm
 lib/PublicInbox/LeiAddWatch.pm
 lib/PublicInbox/LeiAuth.pm
diff --git a/lib/PublicInbox/DirIdle.pm b/lib/PublicInbox/DirIdle.pm
index 65896f95..d572c274 100644
--- a/lib/PublicInbox/DirIdle.pm
+++ b/lib/PublicInbox/DirIdle.pm
@@ -84,4 +84,15 @@ sub event_step {
         warn "$self->{inot}->read err: $@\n" if $@;
 }
 
+sub force_close {
+        my ($self) = @_;
+        my $inot = delete $self->{inot} // return;
+        if ($inot->can('fh')) { # Linux::Inotify2 2.3+
+                close($inot->fh) or warn "CLOSE ERROR: $!";
+        } elsif ($inot->isa('Linux::Inotify2')) {
+                require PublicInbox::LI2Wrap;
+                PublicInbox::LI2Wrap::wrapclose($inot);
+        }
+}
+
 1;
diff --git a/lib/PublicInbox/LEI.pm b/lib/PublicInbox/LEI.pm
index d9fd40fd..e6f763e1 100644
--- a/lib/PublicInbox/LEI.pm
+++ b/lib/PublicInbox/LEI.pm
@@ -556,7 +556,7 @@ sub _lei_atfork_child {
         }
         close $listener if $listener;
         undef $listener;
-        undef $dir_idle;
+        $dir_idle->force_close if $dir_idle;
         %PATH2CFG = ();
         $MDIR2CFGPATH = {};
         eval 'no warnings; undef $PublicInbox::LeiNoteEvent::to_flush';
diff --git a/lib/PublicInbox/LI2Wrap.pm b/lib/PublicInbox/LI2Wrap.pm
new file mode 100644
index 00000000..61cf4bee
--- /dev/null
+++ b/lib/PublicInbox/LI2Wrap.pm
@@ -0,0 +1,19 @@
+# Copyright (C) all contributors <meta@public-inbox.org>
+# License: AGPL-3.0+ <https://www.gnu.org/licenses/agpl-3.0.txt>
+
+# Wrapper for Linux::Inotify2 < 2.3 which lacked ->fh and auto-close
+# Remove this when supported LTS/enterprise distros are all
+# Linux::Inotify2 >= 2.3
+package PublicInbox::LI2Wrap;
+use v5.10.1;
+our @ISA = qw(Linux::Inotify2);
+
+sub wrapclose {
+        my ($inot) = @_;
+        my $fd = $inot->fileno;
+        open my $fh, '<&=', $fd or die "open <&= $fd $!";
+}
+
+sub DESTROY {} # no-op
+
+1