mirror of
git://git.sv.gnu.org/coreutils.git
synced 2026-02-20 14:22:26 +02:00
Running the new fiemap-empty test uses 600MB of disk space via fallocate, and in so doing caused failure in unrelated tests that were running in parallel on a small file system. Rather than simply running fallocate (which allocates the space, inducing disk full when it fails), skip the test if there is less than 800MB of free space, as computed via stat and awk. * tests/init.cfg (require_file_system_bytes_free_): New function. * tests/cp/fiemap-empty: Use it.
442 lines
12 KiB
INI
442 lines
12 KiB
INI
# This file is sourced by init.sh, *before* its initialization.
|
|
|
|
# This goes hand in hand with the "exec 9>&2;" in tests/Makefile.am's
|
|
# TESTS_ENVIRONMENT definition.
|
|
stderr_fileno_=9
|
|
|
|
# FIXME: eventually s/error_/fail_/ and remove the definition of error_ below.
|
|
# FIXME: s/(framework_failure)\>/${1}_/ and remove def. of framework_failure
|
|
|
|
# Having an unsearchable directory in PATH causes execve to fail with EACCES
|
|
# when applied to an unresolvable program name, contrary to the desired ENOENT.
|
|
# Avoid the problem by rewriting PATH to exclude unsearchable directories.
|
|
sanitize_path_()
|
|
{
|
|
# FIXME: remove double quotes around $IFS when all tests use init.sh.
|
|
# They constitute a work-around for a bug in FreeBSD 8.1's /bin/sh.
|
|
local saved_IFS="$IFS"
|
|
IFS=:
|
|
set -- $PATH
|
|
IFS=$saved_IFS
|
|
|
|
local d d1
|
|
local colon=
|
|
local new_path=
|
|
for d in "$@"; do
|
|
test -z "$d" && d1=. || d1=$d
|
|
if ls -d "$d1/." > /dev/null 2>&1; then
|
|
new_path="$new_path$colon$d"
|
|
colon=':'
|
|
fi
|
|
done
|
|
|
|
PATH=$new_path
|
|
export PATH
|
|
}
|
|
|
|
skip_test_()
|
|
{
|
|
echo "$0: skipping test: $@" | head -1 1>&9
|
|
echo "$0: skipping test: $@" 1>&2
|
|
Exit 77
|
|
}
|
|
|
|
getlimits_()
|
|
{
|
|
eval $(getlimits)
|
|
test "$INT_MAX" ||
|
|
error_ "Error running getlimits"
|
|
}
|
|
|
|
require_acl_()
|
|
{
|
|
getfacl --version < /dev/null > /dev/null 2>&1 \
|
|
&& setfacl --version < /dev/null > /dev/null 2>&1 \
|
|
|| skip_test_ "This test requires getfacl and setfacl."
|
|
|
|
id -u bin > /dev/null 2>&1 \
|
|
|| skip_test_ "This test requires a local user named bin."
|
|
}
|
|
|
|
is_local_dir_()
|
|
{
|
|
test $# = 1 || framework_failure
|
|
df --local "$1" >/dev/null 2>&1
|
|
}
|
|
|
|
require_local_dir_()
|
|
{
|
|
is_local_dir_ . ||
|
|
skip_test_ "This test must be run on a local file system."
|
|
}
|
|
|
|
# Skip this test if we're not in SELinux "enforcing" mode.
|
|
require_selinux_enforcing_()
|
|
{
|
|
test "$(getenforce)" = Enforcing \
|
|
|| skip_test_ "This test is useful only with SELinux in Enforcing mode."
|
|
}
|
|
|
|
require_openat_support_()
|
|
{
|
|
# Skip this test if your system has neither the openat-style functions
|
|
# nor /proc/self/fd support with which to emulate them.
|
|
test -z "$CONFIG_HEADER" \
|
|
&& skip_test_ 'internal error: CONFIG_HEADER not defined'
|
|
|
|
_skip=yes
|
|
grep '^#define HAVE_OPENAT' "$CONFIG_HEADER" > /dev/null && _skip=no
|
|
test -d /proc/self/fd && _skip=no
|
|
if test $_skip = yes; then
|
|
skip_test_ 'this system lacks openat support'
|
|
fi
|
|
}
|
|
|
|
require_ulimit_()
|
|
{
|
|
ulimit_works=yes
|
|
# Expect to be able to exec a program in 10MB of virtual memory,
|
|
# but not in 20KB. I chose "date". It must not be a shell built-in
|
|
# function, so you can't use echo, printf, true, etc.
|
|
# Of course, in coreutils, I could use $top_builddir/src/true,
|
|
# but this should be able to work for other projects, too.
|
|
( ulimit -v 10000; date ) > /dev/null 2>&1 || ulimit_works=no
|
|
( ulimit -v 20; date ) > /dev/null 2>&1 && ulimit_works=no
|
|
|
|
test $ulimit_works = no \
|
|
&& skip_test_ "this shell lacks ulimit support"
|
|
}
|
|
|
|
require_readable_root_()
|
|
{
|
|
test -r / || skip_test_ "/ is not readable"
|
|
}
|
|
|
|
# Skip the current test if strace is not available or doesn't work
|
|
# with the named syscall. Usage: require_strace_ unlink
|
|
require_strace_()
|
|
{
|
|
test $# = 1 || framework_failure
|
|
|
|
strace -V < /dev/null > /dev/null 2>&1 ||
|
|
skip_test_ 'no strace program'
|
|
|
|
strace -qe "$1" echo > /dev/null 2>&1 ||
|
|
skip_test_ 'strace -qe "'"$1"'" does not work'
|
|
}
|
|
|
|
# Require a controlling input `terminal'.
|
|
require_controlling_input_terminal_()
|
|
{
|
|
tty -s || have_input_tty=no
|
|
test -t 0 || have_input_tty=no
|
|
if test "$have_input_tty" = no; then
|
|
skip_test_ 'requires controlling input terminal
|
|
This test must have a controlling input "terminal", so it may not be
|
|
run via "batch", "at", or "ssh". On some systems, it may not even be
|
|
run in the background.'
|
|
fi
|
|
}
|
|
|
|
require_built_()
|
|
{
|
|
skip_=no
|
|
for i in "$@"; do
|
|
case " $built_programs " in
|
|
*" $i "*) ;;
|
|
*) echo "$i: not built" 1>&2; skip_=yes ;;
|
|
esac
|
|
done
|
|
|
|
test $skip_ = yes && skip_test_ "required program(s) not built"
|
|
}
|
|
|
|
require_file_system_bytes_free_()
|
|
{
|
|
local req=$1
|
|
local expr=$(stat -f --printf "$req / %S <= %a" .)
|
|
awk "BEGIN{ exit !($expr) }" \
|
|
|| skip_test_ "this test needs at least $req bytes of free space"
|
|
}
|
|
|
|
uid_is_privileged_()
|
|
{
|
|
# Make sure id -u succeeds.
|
|
my_uid=$(id -u) \
|
|
|| { echo "$0: cannot run \`id -u'" 1>&2; return 1; }
|
|
|
|
# Make sure it gives valid output.
|
|
case $my_uid in
|
|
0) ;;
|
|
*[!0-9]*)
|
|
echo "$0: invalid output (\`$my_uid') from \`id -u'" 1>&2
|
|
return 1 ;;
|
|
*) return 1 ;;
|
|
esac
|
|
}
|
|
|
|
get_process_status_()
|
|
{
|
|
sed -n '/^State:[ ]*\([[:alpha:]]\).*/s//\1/p' /proc/$1/status
|
|
}
|
|
|
|
# Convert an ls-style permission string, like drwxr----x and -rw-r-x-wx
|
|
# to the equivalent chmod --mode (-m) argument, (=,u=rwx,g=r,o=x and
|
|
# =,u=rw,g=rx,o=wx). Ignore ACLs.
|
|
rwx_to_mode_()
|
|
{
|
|
case $# in
|
|
1) rwx=$1;;
|
|
*) echo "$0: wrong number of arguments" 1>&2
|
|
echo "Usage: $0 ls-style-mode-string" 1>&2
|
|
return;;
|
|
esac
|
|
|
|
case $rwx in
|
|
[ld-][rwx-][rwx-][rwxsS-][rwx-][rwx-][rwxsS-][rwx-][rwx-][rwxtT-]) ;;
|
|
[ld-][rwx-][rwx-][rwxsS-][rwx-][rwx-][rwxsS-][rwx-][rwx-][rwxtT-][+.]) ;;
|
|
*) echo "$0: invalid mode string: $rwx" 1>&2; return;;
|
|
esac
|
|
|
|
# Perform these conversions:
|
|
# S s
|
|
# s xs
|
|
# T t
|
|
# t xt
|
|
# The `T' and `t' ones are only valid for `other'.
|
|
s='s/S/@/;s/s/x@/;s/@/s/'
|
|
t='s/T/@/;s/t/x@/;s/@/t/'
|
|
|
|
u=`echo $rwx|sed 's/^.\(...\).*/,u=\1/;s/-//g;s/^,u=$//;'$s`
|
|
g=`echo $rwx|sed 's/^....\(...\).*/,g=\1/;s/-//g;s/^,g=$//;'$s`
|
|
o=`echo $rwx|sed 's/^.......\(...\).*/,o=\1/;s/-//g;s/^,o=$//;'$s';'$t`
|
|
echo "=$u$g$o"
|
|
}
|
|
|
|
skip_if_()
|
|
{
|
|
case $1 in
|
|
root) skip_test_ must be run as root ;;
|
|
non-root) skip_test_ must be run as non-root ;;
|
|
*) ;; # FIXME?
|
|
esac
|
|
}
|
|
|
|
require_selinux_()
|
|
{
|
|
# When in a chroot of an SELinux-enabled system, but with a mock-simulated
|
|
# SELinux-*disabled* system, recognize that SELinux is disabled system wide:
|
|
grep 'selinuxfs$' /proc/filesystems > /dev/null \
|
|
|| skip_test_ "this system lacks SELinux support"
|
|
|
|
# Independent of whether SELinux is enabled system-wide,
|
|
# the current file system may lack SELinux support.
|
|
case `ls -Zd .` in
|
|
'? .'|'unlabeled .')
|
|
skip_test_ "this system (or maybe just" \
|
|
"the current file system) lacks SELinux support"
|
|
;;
|
|
esac
|
|
}
|
|
|
|
very_expensive_()
|
|
{
|
|
if test "$RUN_VERY_EXPENSIVE_TESTS" != yes; then
|
|
skip_test_ 'very expensive: disabled by default
|
|
This test is very expensive, so it is disabled by default.
|
|
To run it anyway, rerun make check with the RUN_VERY_EXPENSIVE_TESTS
|
|
environment variable set to yes. E.g.,
|
|
|
|
env RUN_VERY_EXPENSIVE_TESTS=yes make check
|
|
'
|
|
fi
|
|
}
|
|
|
|
expensive_()
|
|
{
|
|
if test "$RUN_EXPENSIVE_TESTS" != yes; then
|
|
skip_test_ 'expensive: disabled by default
|
|
This test is relatively expensive, so it is disabled by default.
|
|
To run it anyway, rerun make check with the RUN_EXPENSIVE_TESTS
|
|
environment variable set to yes. E.g.,
|
|
|
|
env RUN_EXPENSIVE_TESTS=yes make check
|
|
'
|
|
fi
|
|
}
|
|
|
|
require_root_()
|
|
{
|
|
uid_is_privileged_ || skip_test_ "must be run as root"
|
|
NON_ROOT_USERNAME=${NON_ROOT_USERNAME=nobody}
|
|
NON_ROOT_GROUP=${NON_ROOT_GROUP=$(id -g $NON_ROOT_USERNAME)}
|
|
}
|
|
|
|
skip_if_root_() { uid_is_privileged_ && skip_test_ "must be run as non-root"; }
|
|
error_() { echo "$0: $@" 1>&2; Exit 1; }
|
|
framework_failure() { error_ 'failure in testing framework'; }
|
|
|
|
# Set `groups' to a space-separated list of at least two groups
|
|
# of which the user is a member.
|
|
require_membership_in_two_groups_()
|
|
{
|
|
test $# = 0 || framework_failure
|
|
|
|
groups=${COREUTILS_GROUPS-`(id -G || /usr/xpg4/bin/id -G) 2>/dev/null`}
|
|
case "$groups" in
|
|
*' '*) ;;
|
|
*) skip_test_ 'requires membership in two groups
|
|
this test requires that you be a member of more than one group,
|
|
but running `id -G'\'' either failed or found just one. If you really
|
|
are a member of at least two groups, then rerun this test with
|
|
COREUTILS_GROUPS set in your environment to the space-separated list
|
|
of group names or numbers. E.g.,
|
|
|
|
env COREUTILS_GROUPS='users cdrom' make check
|
|
|
|
'
|
|
;;
|
|
esac
|
|
}
|
|
|
|
# Is /proc/$PID/status supported?
|
|
require_proc_pid_status_()
|
|
{
|
|
sleep 2 &
|
|
local pid=$!
|
|
sleep .5
|
|
grep '^State:[ ]*[S]' /proc/$pid/status > /dev/null 2>&1 ||
|
|
skip_test_ "/proc/$pid/status: missing or 'different'"
|
|
kill $pid
|
|
}
|
|
|
|
# Return nonzero if the specified path is on a file system for
|
|
# which FIEMAP support exists. Note some file systems (like ext3 and btrfs)
|
|
# only support FIEMAP for files, not directories.
|
|
fiemap_capable_()
|
|
{
|
|
if ! python < /dev/null; then
|
|
warn_ 'fiemap_capable_: python missing: assuming not fiemap capable'
|
|
return 1
|
|
fi
|
|
python $abs_srcdir/fiemap-capable "$@"
|
|
}
|
|
|
|
# Does the current (working-dir) file system support sparse files?
|
|
require_sparse_support_()
|
|
{
|
|
test $# = 0 || framework_failure
|
|
# Test whether we can create a sparse file.
|
|
# For example, on Darwin6.5 with a file system of type hfs, it's not possible.
|
|
# NTFS requires 128K before a hole appears in a sparse file.
|
|
t=sparse.$$
|
|
dd bs=1 seek=128K of=$t < /dev/null 2> /dev/null
|
|
set x `du -sk $t`
|
|
kb_size=$2
|
|
rm -f $t
|
|
if test $kb_size -ge 128; then
|
|
skip_test_ 'this file system does not support sparse files'
|
|
fi
|
|
}
|
|
|
|
mkfifo_or_skip_()
|
|
{
|
|
test $# = 1 || framework_failure
|
|
if ! mkfifo "$1"; then
|
|
# Make an exception of this case -- usually we interpret framework-creation
|
|
# failure as a test failure. However, in this case, when running on a SunOS
|
|
# system using a disk NFS mounted from OpenBSD, the above fails like this:
|
|
# mkfifo: cannot make fifo `fifo-10558': Not owner
|
|
skip_test_ 'NOTICE: unable to create test prerequisites'
|
|
fi
|
|
}
|
|
|
|
# Disable the current test if the working directory seems to have
|
|
# the setgid bit set.
|
|
skip_if_setgid_()
|
|
{
|
|
setgid_tmpdir=setgid-$$
|
|
(umask 77; mkdir $setgid_tmpdir)
|
|
perms=$(stat --printf %A $setgid_tmpdir)
|
|
rmdir $setgid_tmpdir
|
|
case $perms in
|
|
drwx------);;
|
|
drwxr-xr-x);; # Windows98 + DJGPP 2.03
|
|
*) skip_test_ 'this directory has the setgid bit set';;
|
|
esac
|
|
}
|
|
|
|
skip_if_mcstransd_is_running_()
|
|
{
|
|
test $# = 0 || framework_failure
|
|
|
|
# When mcstransd is running, you'll see only the 3-component
|
|
# version of file-system context strings. Detect that,
|
|
# and if it's running, skip this test.
|
|
__ctx=$(stat --printf='%C\n' .) || framework_failure
|
|
case $__ctx in
|
|
*:*:*:*) ;; # four components is ok
|
|
*) # anything else probably means mcstransd is running
|
|
skip_test_ "unexpected context '$__ctx'; turn off mcstransd" ;;
|
|
esac
|
|
}
|
|
|
|
# Skip the current test if umask doesn't work as usual.
|
|
# This test should be run in the temporary directory that ends
|
|
# up being removed via the trap commands.
|
|
working_umask_or_skip_()
|
|
{
|
|
umask 022
|
|
touch file1 file2
|
|
chmod 644 file2
|
|
perms=`ls -l file1 file2 | sed 's/ .*//' | uniq`
|
|
rm -f file1 file2
|
|
|
|
case $perms in
|
|
*'
|
|
'*) skip_test_ 'your build directory has unusual umask semantics'
|
|
esac
|
|
}
|
|
|
|
# Retry a function requiring a sufficient delay to _pass_
|
|
# using a truncated exponential backoff method.
|
|
# Example: retry_delay_ dd_reblock_1 .1 6
|
|
# This example will call the dd_reblock_1 function with
|
|
# an initial delay of .1 second and call it at most 6 times
|
|
# with a max delay of 3.2s (doubled each time), or a total of 6.3s
|
|
# Note ensure you do _not_ quote the parameter to GNU sleep in
|
|
# your function, as it may contain separate values that `sleep`
|
|
# needs to accumulate.
|
|
retry_delay_()
|
|
{
|
|
local test_func=$1
|
|
local init_delay=$2
|
|
local max_n_tries=$3
|
|
|
|
local attempt=1
|
|
local num_sleeps=$attempt
|
|
local time_fail
|
|
while test $attempt -le $max_n_tries; do
|
|
local delay=$($AWK -v n=$num_sleeps -v s="$init_delay" \
|
|
'BEGIN { print s * n }')
|
|
"$test_func" "$delay" && { time_fail=0; break; } || time_fail=1
|
|
attempt=$(expr $attempt + 1)
|
|
num_sleeps=$(expr $num_sleeps '*' 2)
|
|
done
|
|
test "$time_fail" = 0
|
|
}
|
|
|
|
# Call this with a list of programs under test immediately after
|
|
# sourcing init.sh.
|
|
print_ver_()
|
|
{
|
|
if test "$VERBOSE" = yes; then
|
|
local i
|
|
for i in $*; do
|
|
env $i --version
|
|
done
|
|
fi
|
|
}
|
|
|
|
sanitize_path_
|