#!/bin/sh
#
# Automated tests for utimensat()
#
# Copyright (C) 2008, Linux Foundation
# Written by Michael Kerrisk <mtk.manpages@gmail.com>
# Licensed under GPLv2 or later
#
# Not (yet) included in this automated test set:
# * AT_SYMLINK_NOFOLLOW in flags: If pathname specifies a symbolic link,
# then update the timestamps of the link, rather than the file to which
# it refers.
# * Setting of nanosecond components of timestamps (support for
# nanosecond timestamps is file-system-dependent)
# * "Updated file timestamps are set to the greatest value supported
# by the file system that is not greater than the specified time."
# (i.e., if we set timestamp to {0, 999999999}, then the setting
# is rounded down, rather than up, to unit of timestamp resolution.
# * Privileged processes should be able to bypass permission checks.
# (except when file is marked with the "Immutable" EFA).
#=====================================================================
export TCID=utimensat01
export TST_TOTAL=99
export TST_COUNT=0
. test.sh
if tst_kvcmp -lt "2.6.22"; then
tst_brkm TCONF "System kernel version is less than 2.6.22,cannot execute test"
fi
# Starting with 4.8.0 operations on immutable files return EPERM instead of
# EACCES.
# This patch has also been merged to stable 4.4 with
# b3b4283 ("vfs: move permission checking into notify_change() for utimes(NULL)")
if tst_kvcmp -ge "4.4.27" -a -lt "4.5.0"; then
imaccess=EPERM
elif tst_kvcmp -lt "4.4.27"; then
imaccess=EACCES
else
imaccess=EPERM
fi
RESULT_FILE=$TMPDIR/utimensat.result
TEST_DIR=$TMPDIR/utimensat_tests
FILE=$TEST_DIR/utimensat.test_file
TEST_PROG=utimensat01
if [ ! -f $LTPROOT/testcases/bin/$TEST_PROG ]; then
tst_brkm TBROK "$LTPROOT/testcases/bin/$TEST_PROG is missing (please check install)"
fi
# Summary counters of all test results
test_num=0
failed_cnt=0
passed_cnt=0
failed_list=""
#=====================================================================
setup_file()
{
# $1 is test file pathname
# $2 is owner for test file (chown(1))
# $3 is permissions for test file (chmod(1))
# $4 is "ext2" extended file attributes for test file (chattr(1))
FILE=$1
# Make sure any old version of file is deleted
if test -e $FILE; then
sudo $s_arg chattr -ai $FILE || return $?
sudo $s_arg rm -f $FILE || return $?
fi
# Create file and make atime and mtime zero.
sudo $s_arg -u $test_user touch $FILE || return $?
if ! $TEST_PROG -q $FILE 0 0 0 0 > $RESULT_FILE; then
echo "Failed to set up test file $FILE" 1>&2
exit 1
fi
read res atime mtime < $RESULT_FILE
if test "X$res" != "XSUCCESS" ||
test $atime -ne 0 || test $mtime != 0; then
echo "Failed to set correct times on test file $FILE" 1>&2
exit 1
fi
# Set owner, permissions, and EFAs for file.
if test -n "$2"; then
sudo $s_arg chown $2 $FILE || return $?
fi
sudo $s_arg chmod $3 $FILE || return $?
if test -n "$4"; then
sudo $s_arg chattr $4 $FILE || return $?
fi
# Display file setup, for visual verification
ls -l $FILE | awk '{ printf "Owner=%s; perms=%s; ", $3, $1}'
if ! sudo $s_arg lsattr -l $FILE | sed 's/, /,/g' | awk '{print "EFAs=" $2}'
then
return $?
fi
}
test_failed()
{
tst_resm TFAIL "FAILED test $test_num"
failed_cnt=$(expr $failed_cnt + 1)
failed_list="$failed_list $test_num"
}
check_result()
{
STATUS=$1 # Exit status from test program
EXPECTED_RESULT=$2 # SUCCESS / EACCES / EPERM / EINVAL
EXPECT_ATIME_CHANGED=$3 # Should be 'y' or 'n' (only for SUCCESS)
EXPECT_MTIME_CHANGED=$4 # Should be 'y' or 'n' (only for SUCCESS)
test_num=$(expr $test_num + 1)
# If our test setup failed, stop immediately
if test $STATUS -gt 1; then
echo "FAILED (bad test setup)"
exit 1
fi
read res atime mtime < $RESULT_FILE
echo "EXPECTED: $EXPECTED_RESULT $EXPECT_ATIME_CHANGED "\
"$EXPECT_MTIME_CHANGED"
echo "RESULT: $res $atime $mtime"
if test "$res" != "$EXPECTED_RESULT"; then
test_failed
return
fi
passed=1
# If the test program exited successfully, then check that atime and
# and mtime were updated / not updated, as expected.
if test $EXPECTED_RESULT = "SUCCESS"; then
if test $EXPECT_ATIME_CHANGED = "y"; then
if test $atime -eq 0; then
echo "atime should have changed, but did not"
passed=0
fi
else
if test $atime -ne 0; then
echo "atime should not have changed, but did"
passed=0
fi
fi
if test $EXPECT_MTIME_CHANGED = "y"; then
if test $mtime -eq 0; then
echo "mtime should have changed, but did not"
passed=0
fi
else
if test $mtime -ne 0; then
echo "mtime should not have changed, but did"
passed=0
fi
fi
if test $passed -eq 0; then
test_failed
return
fi
fi
passed_cnt=$(expr $passed_cnt + 1)
tst_resm TPASS "PASSED test $test_num"
}
run_test()
{
# By default, we do three types of test:
# a) pathname (pathname != NULL)
# b) readable file descriptor (pathname == NULL, dirfd opened O_RDONLY)
# c) writable file descriptor (pathname == NULL, dirfd opened O_RDWR).
# For this case we also include O_APPEND in open flags, since that
# is needed if testing with a file that has the Append-only
# attribute enabled.
# -R says don't do tests with readable file descriptor
# -W says don't do tests with writable file descriptor
OPTIND=1
do_read_fd_test=1
do_write_fd_test=1
while getopts "RW" opt; do
case "$opt" in
R) do_read_fd_test=0
;;
W) do_write_fd_test=0
;;
*) echo "run_test: bad usage"
exit 1
;;
esac
done
shift `expr $OPTIND - 1`
echo "Pathname test"
setup_file $FILE "$1" "$2" "$3"
cp $LTPROOT/testcases/bin/$TEST_PROG ./
CMD="./$TEST_PROG -q $FILE $4"
echo "$CMD"
sudo $s_arg -u $test_user $CMD > $RESULT_FILE
check_result $? $5 $6 $7
echo
if test $do_read_fd_test -ne 0; then
echo "Readable file descriptor (futimens(3)) test"
setup_file $FILE "$1" "$2" "$3"
CMD="./$TEST_PROG -q -d $FILE NULL $4"
echo "$CMD"
sudo $s_arg -u $test_user $CMD > $RESULT_FILE
check_result $? $5 $6 $7
echo
fi
# Can't do the writable file descriptor test for immutable files
# (even root can't open an immutable file for writing)
if test $do_write_fd_test -ne 0; then
echo "Writable file descriptor (futimens(3)) test"
setup_file $FILE "$1" "$2" "$3"
CMD="./$TEST_PROG -q -w -d $FILE NULL $4"
echo "$CMD"
sudo $s_arg -u $test_user $CMD > $RESULT_FILE
check_result $? $5 $6 $7
echo
fi
sudo $s_arg chattr -ai $FILE
sudo $s_arg rm -f $FILE
}
#=====================================================================
# Since some automated testing systems have no tty while testing,
# comment this line in /etc/sudoers to avoid the error message:
# `sudo: sorry, you must have a tty to run sudo'
# Use trap to restore this line after program terminates.
sudoers=/etc/sudoers
if [ ! -r $sudoers ]; then
tst_brkm TBROK "can't read $sudoers"
fi
pattern="[[:space:]]*Defaults[[:space:]]*requiretty.*"
if grep -q "^${pattern}" $sudoers; then
tst_resm TINFO "Comment requiretty in $sudoers for automated testing systems"
if ! sed -r -i.$$ -e "s/^($pattern)/#\1/" $sudoers; then
tst_brkm TBROK "failed to mangle $sudoers properly"
fi
trap 'trap "" EXIT; restore_sudoers' EXIT
fi
restore_sudoers()
{
tst_resm TINFO "Restore requiretty in $sudoers"
mv /etc/sudoers.$$ /etc/sudoers
}
test_user=nobody
echo "test sudo for -n option, non-interactive"
if sudo -h | grep -q -- -n; then
s_arg="-n"
echo "sudo supports -n"
else
s_arg=
echo "sudo does not support -n"
fi
if ! sudo $s_arg true; then
tst_brkm TBROK "sudo cannot be run by user non-interactively"
fi
if test ! -f $sudoers
then
echo "root ALL=(ALL) ALL" > $sudoers || exit
chmod 440 $sudoers
trap 'trap "" EXIT; nuke_sudoers' EXIT
fi
nuke_sudoers()
{
sudo rm -f $sudoers
}
sudo $s_arg -u $test_user mkdir -p $TEST_DIR
# Make sure chattr command is supported
touch $TEST_DIR/tmp_file
chattr +a $TEST_DIR/tmp_file
if [ $? -ne 0 ] ; then
rm -rf $TEST_DIR
tst_brkm TCONF "chattr not supported"
fi
chattr -a $TEST_DIR/tmp_file
cd $TEST_DIR
chown root $LTPROOT/testcases/bin/$TEST_PROG
chmod ugo+x,u+s $LTPROOT/testcases/bin/$TEST_PROG
#=====================================================================
echo "============================================================"
echo
echo "Testing read-only file, owned by self"
echo
echo "***** Testing times==NULL case *****"
run_test -W "" 400 "" "" SUCCESS y y
echo "***** Testing times=={ UTIME_NOW, UTIME_NOW } case *****"
run_test -W "" 400 "" "0 n 0 n" SUCCESS y y
echo "***** Testing times=={ UTIME_OMIT, UTIME_OMIT } case *****"
run_test -W "" 400 "" "0 o 0 o" SUCCESS n n
echo "***** Testing times=={ UTIME_NOW, UTIME_OMIT } case *****"
run_test -W "" 400 "" "0 n 0 o" SUCCESS y n
echo "***** Testing times=={ UTIME_OMIT, UTIME_NOW } case *****"
run_test -W "" 400 "" "0 o 0 n" SUCCESS n y
echo "***** Testing times=={ x, y } case *****"
run_test -W "" 400 "" "1 1 1 1" SUCCESS y y
echo "============================================================"
echo
echo "Testing read-only file, not owned by self"
echo
echo "***** Testing times==NULL case *****"
run_test -RW root 400 "" "" EACCES
echo "***** Testing times=={ UTIME_NOW, UTIME_NOW } case *****"
run_test -RW root 400 "" "0 n 0 n" EACCES
echo "***** Testing times=={ UTIME_OMIT, UTIME_OMIT } case *****"
run_test -RW root 400 "" "0 o 0 o" SUCCESS n n
echo "***** Testing times=={ UTIME_NOW, UTIME_OMIT } case *****"
run_test -RW root 400 "" "0 n 0 o" EPERM
echo "***** Testing times=={ UTIME_OMIT, UTIME_NOW } case *****"
run_test -RW root 400 "" "0 o 0 n" EPERM
echo "***** Testing times=={ x, y } case *****"
run_test -RW root 400 "" "1 1 1 1" EPERM
echo "============================================================"
echo
echo "Testing writable file, not owned by self"
echo
echo "***** Testing times==NULL case *****"
run_test root 666 "" "" SUCCESS y y
echo "***** Testing times=={ UTIME_NOW, UTIME_NOW } case *****"
run_test root 666 "" "0 n 0 n" SUCCESS y y
echo "***** Testing times=={ UTIME_OMIT, UTIME_OMIT } case *****"
run_test root 666 "" "0 o 0 o" SUCCESS n n
echo "***** Testing times=={ UTIME_NOW, UTIME_OMIT } case *****"
run_test root 666 "" "0 n 0 o" EPERM
echo "***** Testing times=={ UTIME_OMIT, UTIME_NOW } case *****"
run_test root 666 "" "0 o 0 n" EPERM
echo "***** Testing times=={ x, y } case *****"
run_test root 666 "" "1 1 1 1" EPERM
echo "============================================================"
echo
echo "Testing append-only file, owned by self"
echo
echo "***** Testing times==NULL case *****"
run_test "" 600 "+a" "" SUCCESS y y
echo "***** Testing times=={ UTIME_NOW, UTIME_NOW } case *****"
run_test "" 600 "+a" "0 n 0 n" SUCCESS y y
echo "***** Testing times=={ UTIME_OMIT, UTIME_OMIT } case *****"
run_test "" 600 "+a" "0 o 0 o" SUCCESS n n
echo "***** Testing times=={ UTIME_NOW, UTIME_OMIT } case *****"
run_test "" 600 "+a" "0 n 0 o" EPERM
echo "***** Testing times=={ UTIME_OMIT, UTIME_NOW } case *****"
run_test "" 600 "+a" "0 o 0 n" EPERM
echo "***** Testing times=={ x, y } case *****"
run_test "" 600 "+a" "1 1 1 1" EPERM
echo "============================================================"
echo
echo "Testing immutable file, owned by self"
echo
echo "***** Testing times==NULL case *****"
run_test -W "" 600 "+i" "" $imaccess
echo "***** Testing times=={ UTIME_NOW, UTIME_NOW } case *****"
run_test -W "" 600 "+i" "0 n 0 n" $imaccess
echo "***** Testing times=={ UTIME_OMIT, UTIME_OMIT } case *****"
run_test -W "" 600 "+i" "0 o 0 o" SUCCESS n n
echo "***** Testing times=={ UTIME_NOW, UTIME_OMIT } case *****"
run_test -W "" 600 "+i" "0 n 0 o" EPERM
echo "***** Testing times=={ UTIME_OMIT, UTIME_NOW } case *****"
run_test -W "" 600 "+i" "0 o 0 n" EPERM
echo "***** Testing times=={ x, y } case *****"
run_test -W "" 600 "+i" "1 1 1 1" EPERM
echo "============================================================"
# Immutable+append-only should have same results as immutable
echo
echo "Testing immutable append-only file, owned by self"
echo
echo "***** Testing times==NULL case *****"
run_test -W "" 600 "+ai" "" $imaccess
echo "***** Testing times=={ UTIME_NOW, UTIME_NOW } case *****"
run_test -W "" 600 "+ai" "0 n 0 n" $imaccess
echo "***** Testing times=={ UTIME_OMIT, UTIME_OMIT } case *****"
run_test -W "" 600 "+ai" "0 o 0 o" SUCCESS n n
echo "***** Testing times=={ UTIME_NOW, UTIME_OMIT } case *****"
run_test -W "" 600 "+ai" "0 n 0 o" EPERM
echo "***** Testing times=={ UTIME_OMIT, UTIME_NOW } case *****"
run_test -W "" 600 "+ai" "0 o 0 n" EPERM
echo "***** Testing times=={ x, y } case *****"
run_test -W "" 600 "+ai" "1 1 1 1" EPERM
echo "============================================================"
echo
# EINVAL should result, if pathname is NULL, dirfd is not
# AT_FDCWD, and flags contains AT_SYMLINK_NOFOLLOW.
echo "***** Testing pathname==NULL, dirfd!=AT_FDCWD, flags has" \
"AT_SYMLINK_NOFOLLOW *****"
setup_file $FILE "" 600 ""
CMD="$TEST_PROG -q -n -d $FILE NULL $4"
echo "$CMD"
$CMD > $RESULT_FILE
check_result $? EINVAL
echo
echo "============================================================"
echo
# If UTIME_NOW / UTIME_OMIT in tv_nsec field, the tv_sec should
# be ignored.
echo "tv_sec should be ignored if tv_nsec is UTIME_OMIT or UTIME_NOW"
echo "***** Testing times=={ UTIME_NOW, UTIME_NOW } case *****"
run_test -RW "" 600 "" "1 n 1 n" SUCCESS y y
echo "***** Testing times=={ UTIME_OMIT, UTIME_OMIT } case *****"
run_test -RW "" 600 "" "1 o 1 o" SUCCESS n n
echo "============================================================"
echo
rm -rf "$TEST_DIR"
uname -a
date
echo "Total tests: $test_num; passed: $passed_cnt; failed: $failed_cnt"
if test $failed_cnt -gt 0; then
echo "Failed tests: $failed_list"
fi
tst_exit