125 lines
4.1 KiB
Bash
Executable file
125 lines
4.1 KiB
Bash
Executable file
#!/bin/sh
|
|
# Exercise stdbuf functionality
|
|
|
|
# Copyright (C) 2009-2025 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 <https://www.gnu.org/licenses/>.
|
|
|
|
. "${srcdir=.}/tests/init.sh"; path_prepend_ ./src
|
|
print_ver_ stdbuf env
|
|
|
|
getlimits_
|
|
|
|
# stdbuf fails when the absolute top build dir name contains e.g.,
|
|
# space, TAB, NL
|
|
lf='
|
|
'
|
|
case $abs_top_builddir in
|
|
*[\\\"\#\$\&\'\`$lf\ \ ]*)
|
|
skip_ "unsafe absolute build directory name: $abs_top_builddir";;
|
|
esac
|
|
|
|
# Use a fifo rather than a pipe in the tests below
|
|
# so that the producer (uniq) will wait until the
|
|
# consumer (dd) opens the fifo therefore increasing
|
|
# the chance that dd will read the data from each
|
|
# write separately.
|
|
mkfifo_or_skip_ fifo
|
|
|
|
|
|
# Verify input parameter checking
|
|
stdbuf -o1 true || fail=1 # verify size syntax
|
|
stdbuf -oK true || fail=1 # verify size syntax
|
|
stdbuf -o0 true || fail=1 # verify unbuffered syntax
|
|
stdbuf -oL true || fail=1 # verify line buffered syntax
|
|
|
|
# Capital 'L' required
|
|
# Internal error is a particular status
|
|
returns_ 125 stdbuf -ol true || fail=1
|
|
|
|
returns_ 125 stdbuf -o$SIZE_OFLOW true || fail=1 # size too large
|
|
returns_ 125 stdbuf -iL true || fail=1 # line buffering stdin disallowed
|
|
returns_ 125 stdbuf true || fail=1 # a buffering mode must be specified
|
|
stdbuf -i0 -o0 -e0 true || fail=1 #check all files
|
|
returns_ 126 env . && { returns_ 126 stdbuf -o1 . || fail=1; } # invalid command
|
|
returns_ 127 stdbuf -o1 no_such || fail=1 # no such command
|
|
|
|
# Terminate any background processes
|
|
cleanup_() { kill $pid 2>/dev/null && wait $pid; }
|
|
|
|
# Ensure line buffering stdout takes effect
|
|
stdbuf_linebuffer()
|
|
{
|
|
local delay="$1"
|
|
|
|
printf '1\n' > exp
|
|
> out || framework_failure_
|
|
dd count=1 if=fifo > out 2> err & pid=$!
|
|
(printf '1\n'; sleep $delay; printf '2\n') | stdbuf -oL uniq > fifo
|
|
wait $pid
|
|
compare exp out
|
|
}
|
|
|
|
retry_delay_ stdbuf_linebuffer .1 6 || fail=1
|
|
|
|
stdbuf_unbuffer()
|
|
{
|
|
local delay="$1"
|
|
|
|
# Ensure un buffering stdout takes effect
|
|
printf '1\n' > exp
|
|
> out || framework_failure_
|
|
dd count=1 if=fifo > out 2> err & pid=$!
|
|
(printf '1\n'; sleep $delay; printf '2\n') | stdbuf -o0 uniq > fifo
|
|
wait $pid
|
|
compare exp out
|
|
}
|
|
|
|
retry_delay_ stdbuf_unbuffer .1 6 || fail=1
|
|
|
|
# Ensure un buffering stdin takes effect
|
|
# The following works for me, but is racy. I.e., we're depending
|
|
# on dd to run and close the fifo before the second write by uniq.
|
|
# If we add a sleep, then we're just testing -oL
|
|
# printf '3\n' > exp
|
|
# dd count=1 if=fifo > /dev/null 2> err &
|
|
# printf '1\n\2\n3\n' | (stdbuf -i0 -oL uniq > fifo; cat) > out
|
|
# wait # for dd to complete
|
|
# compare exp out || fail=1
|
|
# One could remove the need for dd (used to close the fifo to get uniq to quit
|
|
# early), if head -n1 read stdin char by char. Note uniq | head -c2 doesn't
|
|
# suffice due to the buffering implicit in the pipe. sed currently does read
|
|
# stdin char by char, so we can test with 'sed 1q'. However I'm wary about
|
|
# adding this dependency on a program outside of coreutils.
|
|
# printf '2\n' > exp
|
|
# printf '1\n2\n' | (stdbuf -i0 sed 1q >/dev/null; cat) > out
|
|
# compare exp out || fail=1
|
|
|
|
# Ensure block buffering stdout takes effect
|
|
# We don't currently test block buffering failures as
|
|
# this doesn't work on GLIBC-2.7 or GLIBC-2.9 at least.
|
|
# stdbuf_blockbuffer()
|
|
# {
|
|
# local delay="$1"
|
|
#
|
|
# printf '1\n2\n' > exp
|
|
# dd count=1 if=fifo > out 2> err &
|
|
# (printf '1\n'; sleep $delay; printf '2\n') | stdbuf -o4 uniq > fifo
|
|
# wait # for dd to complete
|
|
# compare exp out
|
|
# }
|
|
#
|
|
# retry_delay_ stdbuf_blockbuffer .1 6 || fail=1
|
|
|
|
Exit $fail
|