mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-06 05:02:31 +00:00
dbd61921a6
Support testcases that are themselves not executable, but need an interpreter to run them. If a test file is not executable, but an executable file ksft_runner.sh exists in the TARGET dir, kselftest will run ./ksft_runner.sh ./$BASENAME_TEST Packetdrill may add hundreds of packetdrill scripts for testing. These scripts must be passed to the packetdrill process. Have kselftest run each test directly, as it already solves common runner requirements like parallel execution and isolation (netns). A previous RFC added a wrapper in between, which would have to reimplement such functionality. Link: https://lore.kernel.org/netdev/66d4d97a4cac_3df182941a@willemb.c.googlers.com.notmuch/T/ Signed-off-by: Willem de Bruijn <willemb@google.com> Link: https://patch.msgid.link/20240905231653.2427327-2-willemdebruijn.kernel@gmail.com Signed-off-by: Jakub Kicinski <kuba@kernel.org>
198 lines
4.9 KiB
Bash
198 lines
4.9 KiB
Bash
#!/bin/sh
|
|
# SPDX-License-Identifier: GPL-2.0
|
|
#
|
|
# Runs a set of tests in a given subdirectory.
|
|
export skip_rc=4
|
|
export timeout_rc=124
|
|
export logfile=/dev/stdout
|
|
export per_test_logging=
|
|
export RUN_IN_NETNS=
|
|
|
|
# Defaults for "settings" file fields:
|
|
# "timeout" how many seconds to let each test run before running
|
|
# over our soft timeout limit.
|
|
export kselftest_default_timeout=45
|
|
|
|
# There isn't a shell-agnostic way to find the path of a sourced file,
|
|
# so we must rely on BASE_DIR being set to find other tools.
|
|
if [ -z "$BASE_DIR" ]; then
|
|
echo "Error: BASE_DIR must be set before sourcing." >&2
|
|
exit 1
|
|
fi
|
|
|
|
TR_CMD=$(command -v tr)
|
|
|
|
# If Perl is unavailable, we must fall back to line-at-a-time prefixing
|
|
# with sed instead of unbuffered output.
|
|
tap_prefix()
|
|
{
|
|
if [ ! -x /usr/bin/perl ]; then
|
|
sed -e 's/^/# /'
|
|
else
|
|
"$BASE_DIR"/kselftest/prefix.pl
|
|
fi
|
|
}
|
|
|
|
tap_timeout()
|
|
{
|
|
# Make sure tests will time out if utility is available.
|
|
if [ -x /usr/bin/timeout ] ; then
|
|
/usr/bin/timeout --foreground "$kselftest_timeout" \
|
|
/usr/bin/timeout "$kselftest_timeout" $1
|
|
else
|
|
$1
|
|
fi
|
|
}
|
|
|
|
run_one()
|
|
{
|
|
DIR="$1"
|
|
TEST="$2"
|
|
local test_num="$3"
|
|
|
|
BASENAME_TEST=$(basename $TEST)
|
|
|
|
# Reset any "settings"-file variables.
|
|
export kselftest_timeout="$kselftest_default_timeout"
|
|
|
|
# Safe default if tr not available
|
|
kselftest_cmd_args_ref="KSELFTEST_ARGS"
|
|
|
|
# Optional arguments for this command, possibly defined as an
|
|
# environment variable built using the test executable in all
|
|
# uppercase and sanitized substituting non acceptable shell
|
|
# variable name characters with "_" as in:
|
|
#
|
|
# KSELFTEST_<UPPERCASE_SANITIZED_TESTNAME>_ARGS="<options>"
|
|
#
|
|
# e.g.
|
|
#
|
|
# rtctest --> KSELFTEST_RTCTEST_ARGS="/dev/rtc1"
|
|
#
|
|
# cpu-on-off-test.sh --> KSELFTEST_CPU_ON_OFF_TEST_SH_ARGS="-a -p 10"
|
|
#
|
|
if [ -n "$TR_CMD" ]; then
|
|
BASENAME_SANITIZED=$(echo "$BASENAME_TEST" | \
|
|
$TR_CMD -d "[:blank:][:cntrl:]" | \
|
|
$TR_CMD -c "[:alnum:]_" "_" | \
|
|
$TR_CMD [:lower:] [:upper:])
|
|
kselftest_cmd_args_ref="KSELFTEST_${BASENAME_SANITIZED}_ARGS"
|
|
fi
|
|
|
|
# Load per-test-directory kselftest "settings" file.
|
|
settings="$BASE_DIR/$DIR/settings"
|
|
if [ -r "$settings" ] ; then
|
|
while read line ; do
|
|
# Skip comments.
|
|
if echo "$line" | grep -q '^#'; then
|
|
continue
|
|
fi
|
|
field=$(echo "$line" | cut -d= -f1)
|
|
value=$(echo "$line" | cut -d= -f2-)
|
|
eval "kselftest_$field"="$value"
|
|
done < "$settings"
|
|
fi
|
|
|
|
# Command line timeout overrides the settings file
|
|
if [ -n "$kselftest_override_timeout" ]; then
|
|
kselftest_timeout="$kselftest_override_timeout"
|
|
echo "# overriding timeout to $kselftest_timeout" >> "$logfile"
|
|
else
|
|
echo "# timeout set to $kselftest_timeout" >> "$logfile"
|
|
fi
|
|
|
|
TEST_HDR_MSG="selftests: $DIR: $BASENAME_TEST"
|
|
echo "# $TEST_HDR_MSG"
|
|
if [ ! -e "$TEST" ]; then
|
|
echo "# Warning: file $TEST is missing!"
|
|
echo "not ok $test_num $TEST_HDR_MSG"
|
|
else
|
|
if [ -x /usr/bin/stdbuf ]; then
|
|
stdbuf="/usr/bin/stdbuf --output=L "
|
|
fi
|
|
eval kselftest_cmd_args="\$${kselftest_cmd_args_ref:-}"
|
|
if [ -x "$TEST" ]; then
|
|
cmd="$stdbuf ./$BASENAME_TEST $kselftest_cmd_args"
|
|
elif [ -x "./ksft_runner.sh" ]; then
|
|
cmd="$stdbuf ./ksft_runner.sh ./$BASENAME_TEST"
|
|
else
|
|
echo "# Warning: file $TEST is not executable"
|
|
|
|
if [ $(head -n 1 "$TEST" | cut -c -2) = "#!" ]
|
|
then
|
|
interpreter=$(head -n 1 "$TEST" | cut -c 3-)
|
|
cmd="$stdbuf $interpreter ./$BASENAME_TEST"
|
|
else
|
|
echo "not ok $test_num $TEST_HDR_MSG"
|
|
return
|
|
fi
|
|
fi
|
|
cd `dirname $TEST` > /dev/null
|
|
((((( tap_timeout "$cmd" 2>&1; echo $? >&3) |
|
|
tap_prefix >&4) 3>&1) |
|
|
(read xs; exit $xs)) 4>>"$logfile" &&
|
|
echo "ok $test_num $TEST_HDR_MSG") ||
|
|
(rc=$?; \
|
|
if [ $rc -eq $skip_rc ]; then \
|
|
echo "ok $test_num $TEST_HDR_MSG # SKIP"
|
|
elif [ $rc -eq $timeout_rc ]; then \
|
|
echo "#"
|
|
echo "not ok $test_num $TEST_HDR_MSG # TIMEOUT $kselftest_timeout seconds"
|
|
else
|
|
echo "not ok $test_num $TEST_HDR_MSG # exit=$rc"
|
|
fi)
|
|
cd - >/dev/null
|
|
fi
|
|
}
|
|
|
|
in_netns()
|
|
{
|
|
local name=$1
|
|
ip netns exec $name bash <<-EOF
|
|
BASE_DIR=$BASE_DIR
|
|
source $BASE_DIR/kselftest/runner.sh
|
|
logfile=$logfile
|
|
run_one $DIR $TEST $test_num
|
|
EOF
|
|
}
|
|
|
|
run_in_netns()
|
|
{
|
|
local netns=$(mktemp -u ${BASENAME_TEST}-XXXXXX)
|
|
local tmplog="/tmp/$(mktemp -u ${BASENAME_TEST}-XXXXXX)"
|
|
ip netns add $netns
|
|
if [ $? -ne 0 ]; then
|
|
echo "# Warning: Create namespace failed for $BASENAME_TEST"
|
|
echo "not ok $test_num selftests: $DIR: $BASENAME_TEST # Create NS failed"
|
|
fi
|
|
ip -n $netns link set lo up
|
|
in_netns $netns &> $tmplog
|
|
ip netns del $netns &> /dev/null
|
|
cat $tmplog
|
|
rm -f $tmplog
|
|
}
|
|
|
|
run_many()
|
|
{
|
|
echo "TAP version 13"
|
|
DIR="${PWD#${BASE_DIR}/}"
|
|
test_num=0
|
|
total=$(echo "$@" | wc -w)
|
|
echo "1..$total"
|
|
for TEST in "$@"; do
|
|
BASENAME_TEST=$(basename $TEST)
|
|
test_num=$(( test_num + 1 ))
|
|
if [ -n "$per_test_logging" ]; then
|
|
logfile="/tmp/$BASENAME_TEST"
|
|
cat /dev/null > "$logfile"
|
|
fi
|
|
if [ -n "$RUN_IN_NETNS" ]; then
|
|
run_in_netns &
|
|
else
|
|
run_one "$DIR" "$TEST" "$test_num"
|
|
fi
|
|
done
|
|
|
|
wait
|
|
}
|