]> git.ipfire.org Git - thirdparty/coreutils.git/commitdiff
tail: flush initial output before possibly blocking
authorJim Meyering <meyering@redhat.com>
Sun, 6 Sep 2009 07:39:31 +0000 (09:39 +0200)
committerJim Meyering <meyering@redhat.com>
Sun, 6 Sep 2009 07:40:43 +0000 (09:40 +0200)
* src/tail.c (main): Flush any output from tail_file,
before calling tail_forever_inotify, which can block.
* tests/tail-2/flush-initial: New file.  Test for the bug.
* tests/Makefile.am (TESTS): Add tail-2/flush-initial.
* NEWS (Bug fixes): Mention it.
This bug was introduced in coreutils-7.5 via commit ae494d4b,
2009-06-02, "tail: use inotify if it is available".

NEWS
src/tail.c
tests/Makefile.am
tests/tail-2/flush-initial [new file with mode: 0755]

diff --git a/NEWS b/NEWS
index cb01227860a22bc891b9e0d3be6a15d680f15aa7..b02d2daea2edba6fc527a791d9d481db066fbe0f 100644 (file)
--- a/NEWS
+++ b/NEWS
@@ -27,6 +27,11 @@ GNU coreutils NEWS                                    -*- outline -*-
   because ls must stat every file in order to obtain a guaranteed-valid
   inode number.  [bug introduced in coreutils-6.0]
 
+  tail -f (inotify-enabled) now flushes any initial output before blocking.
+  Before, this would print nothing and wait: stdbuf -o 4K tail -f /etc/passwd
+  Note that this bug affects tail -f only when its standard output is buffered,
+  which is relatively unusual.
+
 ** New features
 
   cp --reflink accepts a new "auto" parameter which falls back to
index fee3f1f2d610b5e5bb5afb0feedd7508b6b3504d..e3b9529f843e6b4c70b1792d84fa6637ad0f9cad 100644 (file)
@@ -1399,7 +1399,6 @@ tail_forever_inotify (int wd, struct File_spec *f, size_t n_files,
       if (fflush (stdout) != 0)
         error (EXIT_FAILURE, errno, _("write error"));
     }
-
 }
 #endif
 
@@ -1990,6 +1989,12 @@ main (int argc, char **argv)
             error (0, errno, _("inotify cannot be used, reverting to polling"));
           else
             {
+              /* Flush any output from tail_file, now, since
+                 tail_forever_inotify flushes only after writing,
+                 not before reading.  */
+              if (fflush (stdout) != 0)
+                error (EXIT_FAILURE, errno, _("write error"));
+
               tail_forever_inotify (wd, F, n_files, sleep_interval);
 
               /* The only way the above returns is upon failure.  */
index d9ff95be47dfcbad0c66f4f37d815e75bd43cc37..d83d41b9bebd633a2a62d31a89346a5756d58af2 100644 (file)
@@ -427,6 +427,7 @@ TESTS =                                             \
   rmdir/t-slash                                        \
   tail-2/assert-2                              \
   tail-2/big-4gb                               \
+  tail-2/flush-initial                         \
   tail-2/proc-ksyms                            \
   tail-2/start-middle                          \
   touch/dangling-symlink                       \
diff --git a/tests/tail-2/flush-initial b/tests/tail-2/flush-initial
new file mode 100755 (executable)
index 0000000..2deff84
--- /dev/null
@@ -0,0 +1,41 @@
+#!/bin/sh
+# inotify-based tail -f didn't flush its initial output before blocking
+
+# Copyright (C) 2009 Free Software Foundation, Inc.
+
+# This program is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+# GNU General Public License for more details.
+
+# You should have received a copy of the GNU General Public License
+# along with this program.  If not, see <http://www.gnu.org/licenses/>.
+
+if test "$VERBOSE" = yes; then
+  set -x
+  tail --version
+fi
+
+. $srcdir/test-lib.sh
+
+fail=0
+echo line > in || fail=1
+stdbuf --output=1K tail -f in > out &
+tail_pid=$!
+
+# Wait for the backgrounded `tail' to start.
+while :; do
+  env kill -0 $tail_pid && break
+  sleep .1
+done
+
+test -s out || fail=1
+
+kill $tail_pid
+
+Exit $fail