mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	 fc2fb38c85
			
		
	
	
		fc2fb38c85
		
	
	
	
	
		
			
			This Kselftest update for Linux 5.8-rc1 consists of:
 
 - Several fixes from Masami Hiramatsu to improve coverage for
   lib and sysctl tests.
 - Clean up to vdso test and a new test for getcpu() from Mark Brown.
 - Add new gen_tar selftests Makefile target generate selftest package
   running "make gen_tar" in selftests directory from Veronika Kabatova.
 - Other miscellaneous fixes to timens, exec, tpm2 tests.
 -----BEGIN PGP SIGNATURE-----
 
 iQIzBAABCgAdFiEEPZKym/RZuOCGeA/kCwJExA0NQxwFAl7ernMACgkQCwJExA0N
 QxzgRg/+PYwHxTtKYX2VyQ533oTUXMdTAW8el8M72/ngRBRJMtkjYu9Jc5PfAaP9
 vsyDrC8Kz6D3Nw0Gf+nH2LFJ2hJs0OooBb4HqouAyyNitXGkSiyGb6+1ICmW0Ro8
 NIlwCYrIuoPUKS9ivRoKs0Vsnt0xh12asTlEwUPEEjmvE7Jf+6eoWIExu7i3nSnc
 symOpc3ElCXex4AlRIMI07naQqlT7rtOSQMDpjEXzMMW4rKMlgO8dFCvLkZQh/lj
 gUt45W9vF/Sem6a9jRMPrb8vEGkC2WURNsUVbl4JUbKp8vAXr5bYUD9CMXNdK6r2
 o241nNScjr9RkZoN94DtXfcQbpwnvnhomJzqe/BgKVhfROhv4xf6fdvkxnAMYeXD
 J/szKvLkKkvZ5O19rBqEUCXxYdiSgFOAgHOvnUKfjKKvhMrTQ//3r8L4K6qv489A
 PAyoLuu8GjEVxX2dISB01ASwHeEUTZ6uF2wsu5i2dVWfTAIIpeL2Xnjsa+WmIcH7
 iwBI/2RCuIeC3hViwGKuiSKaN70E/rgXcKO8jxnsF/2+hZnCuMg1D7PJLdIomJjI
 6EiPukLGkZxJcHK2kdJSXIR0elEZ4pH7kweanIDPTxDiUqGwK7ZAFrA2feDJ11Iz
 dye9uC7sjGh27FRbVvoDvZ9659SZ1cqYS+wRcTkPx3K1wS1kuZI=
 =IB5c
 -----END PGP SIGNATURE-----
Merge tag 'linux-kselftest-5.8-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/shuah/linux-kselftest
Pull kselftest updates from Shuah Khan:
 "This consists of:
   - Several fixes from Masami Hiramatsu to improve coverage for lib and
     sysctl tests.
   - Clean up to vdso test and a new test for getcpu() from Mark Brown.
   - Add new gen_tar selftests Makefile target generate selftest package
     running "make gen_tar" in selftests directory from Veronika
     Kabatova.
   - Other miscellaneous fixes to timens, exec, tpm2 tests"
* tag 'linux-kselftest-5.8-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/shuah/linux-kselftest:
  selftests/sysctl: Make sysctl test driver as a module
  selftests/sysctl: Fix to load test_sysctl module
  lib: Make test_sysctl initialized as module
  lib: Make prime number generator independently selectable
  selftests/ftrace: Return unsupported if no error_log file
  selftests/ftrace: Use printf for backslash included command
  selftests/timens: handle a case when alarm clocks are not supported
  Kernel selftests: Add check if TPM devices are supported
  selftests: vdso: Add a selftest for vDSO getcpu()
  selftests: vdso: Use a header file to prototype parse_vdso API
  selftests: vdso: Rename vdso_test to vdso_test_gettimeofday
  selftests/exec: Verify execve of non-regular files fail
  selftests: introduce gen_tar Makefile target
		
	
			
		
			
				
	
	
		
			972 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			Bash
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			972 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			Bash
		
	
	
		
			Executable File
		
	
	
	
	
| #!/bin/bash
 | |
| # Copyright (C) 2017 Luis R. Rodriguez <mcgrof@kernel.org>
 | |
| #
 | |
| # 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 2 of the License, or at your option any
 | |
| # later version; or, when distributed separately from the Linux kernel or
 | |
| # when incorporated into other software packages, subject to the following
 | |
| # license:
 | |
| #
 | |
| # This program is free software; you can redistribute it and/or modify it
 | |
| # under the terms of copyleft-next (version 0.3.1 or later) as published
 | |
| # at http://copyleft-next.org/.
 | |
| 
 | |
| # This performs a series tests against the proc sysctl interface.
 | |
| 
 | |
| # Kselftest framework requirement - SKIP code is 4.
 | |
| ksft_skip=4
 | |
| 
 | |
| TEST_NAME="sysctl"
 | |
| TEST_DRIVER="test_${TEST_NAME}"
 | |
| TEST_DIR=$(dirname $0)
 | |
| TEST_FILE=$(mktemp)
 | |
| 
 | |
| # This represents
 | |
| #
 | |
| # TEST_ID:TEST_COUNT:ENABLED:TARGET
 | |
| #
 | |
| # TEST_ID: is the test id number
 | |
| # TEST_COUNT: number of times we should run the test
 | |
| # ENABLED: 1 if enabled, 0 otherwise
 | |
| # TARGET: test target file required on the test_sysctl module
 | |
| #
 | |
| # Once these are enabled please leave them as-is. Write your own test,
 | |
| # we have tons of space.
 | |
| ALL_TESTS="0001:1:1:int_0001"
 | |
| ALL_TESTS="$ALL_TESTS 0002:1:1:string_0001"
 | |
| ALL_TESTS="$ALL_TESTS 0003:1:1:int_0002"
 | |
| ALL_TESTS="$ALL_TESTS 0004:1:1:uint_0001"
 | |
| ALL_TESTS="$ALL_TESTS 0005:3:1:int_0003"
 | |
| ALL_TESTS="$ALL_TESTS 0006:50:1:bitmap_0001"
 | |
| ALL_TESTS="$ALL_TESTS 0007:1:1:boot_int"
 | |
| 
 | |
| function allow_user_defaults()
 | |
| {
 | |
| 	if [ -z $DIR ]; then
 | |
| 		DIR="/sys/module/test_sysctl/"
 | |
| 	fi
 | |
| 	if [ -z $DEFAULT_NUM_TESTS ]; then
 | |
| 		DEFAULT_NUM_TESTS=50
 | |
| 	fi
 | |
| 	if [ -z $SYSCTL ]; then
 | |
| 		SYSCTL="/proc/sys/debug/test_sysctl"
 | |
| 	fi
 | |
| 	if [ -z $PROD_SYSCTL ]; then
 | |
| 		PROD_SYSCTL="/proc/sys"
 | |
| 	fi
 | |
| 	if [ -z $WRITES_STRICT ]; then
 | |
| 		WRITES_STRICT="${PROD_SYSCTL}/kernel/sysctl_writes_strict"
 | |
| 	fi
 | |
| }
 | |
| 
 | |
| function check_production_sysctl_writes_strict()
 | |
| {
 | |
| 	echo -n "Checking production write strict setting ... "
 | |
| 	if [ ! -e ${WRITES_STRICT} ]; then
 | |
| 		echo "FAIL, but skip in case of old kernel" >&2
 | |
| 	else
 | |
| 		old_strict=$(cat ${WRITES_STRICT})
 | |
| 		if [ "$old_strict" = "1" ]; then
 | |
| 			echo "ok"
 | |
| 		else
 | |
| 			echo "FAIL, strict value is 0 but force to 1 to continue" >&2
 | |
| 			echo "1" > ${WRITES_STRICT}
 | |
| 		fi
 | |
| 	fi
 | |
| 
 | |
| 	if [ -z $PAGE_SIZE ]; then
 | |
| 		PAGE_SIZE=$(getconf PAGESIZE)
 | |
| 	fi
 | |
| 	if [ -z $MAX_DIGITS ]; then
 | |
| 		MAX_DIGITS=$(($PAGE_SIZE/8))
 | |
| 	fi
 | |
| 	if [ -z $INT_MAX ]; then
 | |
| 		INT_MAX=$(getconf INT_MAX)
 | |
| 	fi
 | |
| 	if [ -z $UINT_MAX ]; then
 | |
| 		UINT_MAX=$(getconf UINT_MAX)
 | |
| 	fi
 | |
| }
 | |
| 
 | |
| test_reqs()
 | |
| {
 | |
| 	uid=$(id -u)
 | |
| 	if [ $uid -ne 0 ]; then
 | |
| 		echo $msg must be run as root >&2
 | |
| 		exit $ksft_skip
 | |
| 	fi
 | |
| 
 | |
| 	if ! which perl 2> /dev/null > /dev/null; then
 | |
| 		echo "$0: You need perl installed"
 | |
| 		exit $ksft_skip
 | |
| 	fi
 | |
| 	if ! which getconf 2> /dev/null > /dev/null; then
 | |
| 		echo "$0: You need getconf installed"
 | |
| 		exit $ksft_skip
 | |
| 	fi
 | |
| 	if ! which diff 2> /dev/null > /dev/null; then
 | |
| 		echo "$0: You need diff installed"
 | |
| 		exit $ksft_skip
 | |
| 	fi
 | |
| }
 | |
| 
 | |
| function load_req_mod()
 | |
| {
 | |
| 	if [ ! -d $SYSCTL ]; then
 | |
| 		if ! modprobe -q -n $TEST_DRIVER; then
 | |
| 			echo "$0: module $TEST_DRIVER not found [SKIP]"
 | |
| 			echo "You must set CONFIG_TEST_SYSCTL=m in your kernel" >&2
 | |
| 			exit $ksft_skip
 | |
| 		fi
 | |
| 		modprobe $TEST_DRIVER
 | |
| 		if [ $? -ne 0 ]; then
 | |
| 			echo "$0: modprobe $TEST_DRIVER failed."
 | |
| 			exit
 | |
| 		fi
 | |
| 	fi
 | |
| }
 | |
| 
 | |
| reset_vals()
 | |
| {
 | |
| 	VAL=""
 | |
| 	TRIGGER=$(basename ${TARGET})
 | |
| 	case "$TRIGGER" in
 | |
| 		int_0001)
 | |
| 			VAL="60"
 | |
| 			;;
 | |
| 		int_0002)
 | |
| 			VAL="1"
 | |
| 			;;
 | |
| 		uint_0001)
 | |
| 			VAL="314"
 | |
| 			;;
 | |
| 		string_0001)
 | |
| 			VAL="(none)"
 | |
| 			;;
 | |
| 		bitmap_0001)
 | |
| 			VAL=""
 | |
| 			;;
 | |
| 		*)
 | |
| 			;;
 | |
| 	esac
 | |
| 	echo -n $VAL > $TARGET
 | |
| }
 | |
| 
 | |
| set_orig()
 | |
| {
 | |
| 	if [ ! -z $TARGET ] && [ ! -z $ORIG ]; then
 | |
| 		if [ -f ${TARGET} ]; then
 | |
| 			echo "${ORIG}" > "${TARGET}"
 | |
| 		fi
 | |
| 	fi
 | |
| }
 | |
| 
 | |
| set_test()
 | |
| {
 | |
| 	echo "${TEST_STR}" > "${TARGET}"
 | |
| }
 | |
| 
 | |
| verify()
 | |
| {
 | |
| 	local seen
 | |
| 	seen=$(cat "$1")
 | |
| 	if [ "${seen}" != "${TEST_STR}" ]; then
 | |
| 		return 1
 | |
| 	fi
 | |
| 	return 0
 | |
| }
 | |
| 
 | |
| # proc files get read a page at a time, which can confuse diff,
 | |
| # and get you incorrect results on proc files with long data. To use
 | |
| # diff against them you must first extract the output to a file, and
 | |
| # then compare against that file.
 | |
| verify_diff_proc_file()
 | |
| {
 | |
| 	TMP_DUMP_FILE=$(mktemp)
 | |
| 	cat $1 > $TMP_DUMP_FILE
 | |
| 
 | |
| 	if ! diff -w -q $TMP_DUMP_FILE $2; then
 | |
| 		return 1
 | |
| 	else
 | |
| 		return 0
 | |
| 	fi
 | |
| }
 | |
| 
 | |
| verify_diff_w()
 | |
| {
 | |
| 	echo "$TEST_STR" | diff -q -w -u - $1 > /dev/null
 | |
| 	return $?
 | |
| }
 | |
| 
 | |
| test_rc()
 | |
| {
 | |
| 	if [[ $rc != 0 ]]; then
 | |
| 		echo "Failed test, return value: $rc" >&2
 | |
| 		exit $rc
 | |
| 	fi
 | |
| }
 | |
| 
 | |
| test_finish()
 | |
| {
 | |
| 	set_orig
 | |
| 	rm -f "${TEST_FILE}"
 | |
| 
 | |
| 	if [ ! -z ${old_strict} ]; then
 | |
| 		echo ${old_strict} > ${WRITES_STRICT}
 | |
| 	fi
 | |
| 	exit $rc
 | |
| }
 | |
| 
 | |
| run_numerictests()
 | |
| {
 | |
| 	echo "== Testing sysctl behavior against ${TARGET} =="
 | |
| 
 | |
| 	rc=0
 | |
| 
 | |
| 	echo -n "Writing test file ... "
 | |
| 	echo "${TEST_STR}" > "${TEST_FILE}"
 | |
| 	if ! verify "${TEST_FILE}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		exit 1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 
 | |
| 	echo -n "Checking sysctl is not set to test value ... "
 | |
| 	if verify "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		exit 1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 
 | |
| 	echo -n "Writing sysctl from shell ... "
 | |
| 	set_test
 | |
| 	if ! verify "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		exit 1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 
 | |
| 	echo -n "Resetting sysctl to original value ... "
 | |
| 	set_orig
 | |
| 	if verify "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		exit 1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 
 | |
| 	# Now that we've validated the sanity of "set_test" and "set_orig",
 | |
| 	# we can use those functions to set starting states before running
 | |
| 	# specific behavioral tests.
 | |
| 
 | |
| 	echo -n "Writing entire sysctl in single write ... "
 | |
| 	set_orig
 | |
| 	dd if="${TEST_FILE}" of="${TARGET}" bs=4096 2>/dev/null
 | |
| 	if ! verify "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 
 | |
| 	echo -n "Writing middle of sysctl after synchronized seek ... "
 | |
| 	set_test
 | |
| 	dd if="${TEST_FILE}" of="${TARGET}" bs=1 seek=1 skip=1 2>/dev/null
 | |
| 	if ! verify "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 
 | |
| 	echo -n "Writing beyond end of sysctl ... "
 | |
| 	set_orig
 | |
| 	dd if="${TEST_FILE}" of="${TARGET}" bs=20 seek=2 2>/dev/null
 | |
| 	if verify "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 
 | |
| 	echo -n "Writing sysctl with multiple long writes ... "
 | |
| 	set_orig
 | |
| 	(perl -e 'print "A" x 50;'; echo "${TEST_STR}") | \
 | |
| 		dd of="${TARGET}" bs=50 2>/dev/null
 | |
| 	if verify "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 	test_rc
 | |
| }
 | |
| 
 | |
| check_failure()
 | |
| {
 | |
| 	echo -n "Testing that $1 fails as expected..."
 | |
| 	reset_vals
 | |
| 	TEST_STR="$1"
 | |
| 	orig="$(cat $TARGET)"
 | |
| 	echo -n "$TEST_STR" > $TARGET 2> /dev/null
 | |
| 
 | |
| 	# write should fail and $TARGET should retain its original value
 | |
| 	if [ $? = 0 ] || [ "$(cat $TARGET)" != "$orig" ]; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 	test_rc
 | |
| }
 | |
| 
 | |
| run_wideint_tests()
 | |
| {
 | |
| 	# sysctl conversion functions receive a boolean sign and ulong
 | |
| 	# magnitude; here we list the magnitudes we want to test (each of
 | |
| 	# which will be tested in both positive and negative forms).  Since
 | |
| 	# none of these values fit in 32 bits, writing them to an int- or
 | |
| 	# uint-typed sysctl should fail.
 | |
| 	local magnitudes=(
 | |
| 		# common boundary-condition values (zero, +1, -1, INT_MIN,
 | |
| 		# and INT_MAX respectively) if truncated to lower 32 bits
 | |
| 		# (potential for being falsely deemed in range)
 | |
| 		0x0000000100000000
 | |
| 		0x0000000100000001
 | |
| 		0x00000001ffffffff
 | |
| 		0x0000000180000000
 | |
| 		0x000000017fffffff
 | |
| 
 | |
| 		# these look like negatives, but without a leading '-' are
 | |
| 		# actually large positives (should be rejected as above
 | |
| 		# despite being zero/+1/-1/INT_MIN/INT_MAX in the lower 32)
 | |
| 		0xffffffff00000000
 | |
| 		0xffffffff00000001
 | |
| 		0xffffffffffffffff
 | |
| 		0xffffffff80000000
 | |
| 		0xffffffff7fffffff
 | |
| 	)
 | |
| 
 | |
| 	for sign in '' '-'; do
 | |
| 		for mag in "${magnitudes[@]}"; do
 | |
| 			check_failure "${sign}${mag}"
 | |
| 		done
 | |
| 	done
 | |
| }
 | |
| 
 | |
| # Your test must accept digits 3 and 4 to use this
 | |
| run_limit_digit()
 | |
| {
 | |
| 	echo -n "Checking ignoring spaces up to PAGE_SIZE works on write ..."
 | |
| 	reset_vals
 | |
| 
 | |
| 	LIMIT=$((MAX_DIGITS -1))
 | |
| 	TEST_STR="3"
 | |
| 	(perl -e 'print " " x '$LIMIT';'; echo "${TEST_STR}") | \
 | |
| 		dd of="${TARGET}" 2>/dev/null
 | |
| 
 | |
| 	if ! verify "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 	test_rc
 | |
| 
 | |
| 	echo -n "Checking passing PAGE_SIZE of spaces fails on write ..."
 | |
| 	reset_vals
 | |
| 
 | |
| 	LIMIT=$((MAX_DIGITS))
 | |
| 	TEST_STR="4"
 | |
| 	(perl -e 'print " " x '$LIMIT';'; echo "${TEST_STR}") | \
 | |
| 		dd of="${TARGET}" 2>/dev/null
 | |
| 
 | |
| 	if verify "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 	test_rc
 | |
| }
 | |
| 
 | |
| # You are using an int
 | |
| run_limit_digit_int()
 | |
| {
 | |
| 	echo -n "Testing INT_MAX works ..."
 | |
| 	reset_vals
 | |
| 	TEST_STR="$INT_MAX"
 | |
| 	echo -n $TEST_STR > $TARGET
 | |
| 
 | |
| 	if ! verify "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 	test_rc
 | |
| 
 | |
| 	echo -n "Testing INT_MAX + 1 will fail as expected..."
 | |
| 	reset_vals
 | |
| 	let TEST_STR=$INT_MAX+1
 | |
| 	echo -n $TEST_STR > $TARGET 2> /dev/null
 | |
| 
 | |
| 	if verify "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 	test_rc
 | |
| 
 | |
| 	echo -n "Testing negative values will work as expected..."
 | |
| 	reset_vals
 | |
| 	TEST_STR="-3"
 | |
| 	echo -n $TEST_STR > $TARGET 2> /dev/null
 | |
| 	if ! verify "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 	test_rc
 | |
| }
 | |
| 
 | |
| # You used an int array
 | |
| run_limit_digit_int_array()
 | |
| {
 | |
| 	echo -n "Testing array works as expected ... "
 | |
| 	TEST_STR="4 3 2 1"
 | |
| 	echo -n $TEST_STR > $TARGET
 | |
| 
 | |
| 	if ! verify_diff_w "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 	test_rc
 | |
| 
 | |
| 	echo -n "Testing skipping trailing array elements works ... "
 | |
| 	# Do not reset_vals, carry on the values from the last test.
 | |
| 	# If we only echo in two digits the last two are left intact
 | |
| 	TEST_STR="100 101"
 | |
| 	echo -n $TEST_STR > $TARGET
 | |
| 	# After we echo in, to help diff we need to set on TEST_STR what
 | |
| 	# we expect the result to be.
 | |
| 	TEST_STR="100 101 2 1"
 | |
| 
 | |
| 	if ! verify_diff_w "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 	test_rc
 | |
| 
 | |
| 	echo -n "Testing PAGE_SIZE limit on array works ... "
 | |
| 	# Do not reset_vals, carry on the values from the last test.
 | |
| 	# Even if you use an int array, you are still restricted to
 | |
| 	# MAX_DIGITS, this is a known limitation. Test limit works.
 | |
| 	LIMIT=$((MAX_DIGITS -1))
 | |
| 	TEST_STR="9"
 | |
| 	(perl -e 'print " " x '$LIMIT';'; echo "${TEST_STR}") | \
 | |
| 		dd of="${TARGET}" 2>/dev/null
 | |
| 
 | |
| 	TEST_STR="9 101 2 1"
 | |
| 	if ! verify_diff_w "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 	test_rc
 | |
| 
 | |
| 	echo -n "Testing exceeding PAGE_SIZE limit fails as expected ... "
 | |
| 	# Do not reset_vals, carry on the values from the last test.
 | |
| 	# Now go over limit.
 | |
| 	LIMIT=$((MAX_DIGITS))
 | |
| 	TEST_STR="7"
 | |
| 	(perl -e 'print " " x '$LIMIT';'; echo "${TEST_STR}") | \
 | |
| 		dd of="${TARGET}" 2>/dev/null
 | |
| 
 | |
| 	TEST_STR="7 101 2 1"
 | |
| 	if verify_diff_w "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 	test_rc
 | |
| }
 | |
| 
 | |
| # You are using an unsigned int
 | |
| run_limit_digit_uint()
 | |
| {
 | |
| 	echo -n "Testing UINT_MAX works ..."
 | |
| 	reset_vals
 | |
| 	TEST_STR="$UINT_MAX"
 | |
| 	echo -n $TEST_STR > $TARGET
 | |
| 
 | |
| 	if ! verify "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 	test_rc
 | |
| 
 | |
| 	echo -n "Testing UINT_MAX + 1 will fail as expected..."
 | |
| 	reset_vals
 | |
| 	TEST_STR=$(($UINT_MAX+1))
 | |
| 	echo -n $TEST_STR > $TARGET 2> /dev/null
 | |
| 
 | |
| 	if verify "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 	test_rc
 | |
| 
 | |
| 	echo -n "Testing negative values will not work as expected ..."
 | |
| 	reset_vals
 | |
| 	TEST_STR="-3"
 | |
| 	echo -n $TEST_STR > $TARGET 2> /dev/null
 | |
| 
 | |
| 	if verify "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 	test_rc
 | |
| }
 | |
| 
 | |
| run_stringtests()
 | |
| {
 | |
| 	echo -n "Writing entire sysctl in short writes ... "
 | |
| 	set_orig
 | |
| 	dd if="${TEST_FILE}" of="${TARGET}" bs=1 2>/dev/null
 | |
| 	if ! verify "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 
 | |
| 	echo -n "Writing middle of sysctl after unsynchronized seek ... "
 | |
| 	set_test
 | |
| 	dd if="${TEST_FILE}" of="${TARGET}" bs=1 seek=1 2>/dev/null
 | |
| 	if verify "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 
 | |
| 	echo -n "Checking sysctl maxlen is at least $MAXLEN ... "
 | |
| 	set_orig
 | |
| 	perl -e 'print "A" x ('"${MAXLEN}"'-2), "B";' | \
 | |
| 		dd of="${TARGET}" bs="${MAXLEN}" 2>/dev/null
 | |
| 	if ! grep -q B "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 
 | |
| 	echo -n "Checking sysctl keeps original string on overflow append ... "
 | |
| 	set_orig
 | |
| 	perl -e 'print "A" x ('"${MAXLEN}"'-1), "B";' | \
 | |
| 		dd of="${TARGET}" bs=$(( MAXLEN - 1 )) 2>/dev/null
 | |
| 	if grep -q B "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 
 | |
| 	echo -n "Checking sysctl stays NULL terminated on write ... "
 | |
| 	set_orig
 | |
| 	perl -e 'print "A" x ('"${MAXLEN}"'-1), "B";' | \
 | |
| 		dd of="${TARGET}" bs="${MAXLEN}" 2>/dev/null
 | |
| 	if grep -q B "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 
 | |
| 	echo -n "Checking sysctl stays NULL terminated on overwrite ... "
 | |
| 	set_orig
 | |
| 	perl -e 'print "A" x ('"${MAXLEN}"'-1), "BB";' | \
 | |
| 		dd of="${TARGET}" bs=$(( $MAXLEN + 1 )) 2>/dev/null
 | |
| 	if grep -q B "${TARGET}"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 	fi
 | |
| 
 | |
| 	test_rc
 | |
| }
 | |
| 
 | |
| target_exists()
 | |
| {
 | |
| 	TARGET="${SYSCTL}/$1"
 | |
| 	TEST_ID="$2"
 | |
| 
 | |
| 	if [ ! -f ${TARGET} ] ; then
 | |
| 		echo "Target for test $TEST_ID: $TARGET not exist, skipping test ..."
 | |
| 		return 0
 | |
| 	fi
 | |
| 	return 1
 | |
| }
 | |
| 
 | |
| run_bitmaptest() {
 | |
| 	# Total length of bitmaps string to use, a bit under
 | |
| 	# the maximum input size of the test node
 | |
| 	LENGTH=$((RANDOM % 65000))
 | |
| 
 | |
| 	# First bit to set
 | |
| 	BIT=$((RANDOM % 1024))
 | |
| 
 | |
| 	# String containing our list of bits to set
 | |
| 	TEST_STR=$BIT
 | |
| 
 | |
| 	# build up the string
 | |
| 	while [ "${#TEST_STR}" -le "$LENGTH" ]; do
 | |
| 		# Make sure next entry is discontiguous,
 | |
| 		# skip ahead at least 2
 | |
| 		BIT=$((BIT + $((2 + RANDOM % 10))))
 | |
| 
 | |
| 		# Add new bit to the list
 | |
| 		TEST_STR="${TEST_STR},${BIT}"
 | |
| 
 | |
| 		# Randomly make it a range
 | |
| 		if [ "$((RANDOM % 2))" -eq "1" ]; then
 | |
| 			RANGE_END=$((BIT + $((1 + RANDOM % 10))))
 | |
| 			TEST_STR="${TEST_STR}-${RANGE_END}"
 | |
| 			BIT=$RANGE_END
 | |
| 		fi
 | |
| 	done
 | |
| 
 | |
| 	echo -n "Checking bitmap handler... "
 | |
| 	TEST_FILE=$(mktemp)
 | |
| 	echo -n "$TEST_STR" > $TEST_FILE
 | |
| 
 | |
| 	cat $TEST_FILE > $TARGET 2> /dev/null
 | |
| 	if [ $? -ne 0 ]; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 		test_rc
 | |
| 	fi
 | |
| 
 | |
| 	if ! verify_diff_proc_file "$TARGET" "$TEST_FILE"; then
 | |
| 		echo "FAIL" >&2
 | |
| 		rc=1
 | |
| 	else
 | |
| 		echo "ok"
 | |
| 		rc=0
 | |
| 	fi
 | |
| 	test_rc
 | |
| }
 | |
| 
 | |
| sysctl_test_0001()
 | |
| {
 | |
| 	TARGET="${SYSCTL}/$(get_test_target 0001)"
 | |
| 	reset_vals
 | |
| 	ORIG=$(cat "${TARGET}")
 | |
| 	TEST_STR=$(( $ORIG + 1 ))
 | |
| 
 | |
| 	run_numerictests
 | |
| 	run_wideint_tests
 | |
| 	run_limit_digit
 | |
| }
 | |
| 
 | |
| sysctl_test_0002()
 | |
| {
 | |
| 	TARGET="${SYSCTL}/$(get_test_target 0002)"
 | |
| 	reset_vals
 | |
| 	ORIG=$(cat "${TARGET}")
 | |
| 	TEST_STR="Testing sysctl"
 | |
| 	# Only string sysctls support seeking/appending.
 | |
| 	MAXLEN=65
 | |
| 
 | |
| 	run_numerictests
 | |
| 	run_stringtests
 | |
| }
 | |
| 
 | |
| sysctl_test_0003()
 | |
| {
 | |
| 	TARGET="${SYSCTL}/$(get_test_target 0003)"
 | |
| 	reset_vals
 | |
| 	ORIG=$(cat "${TARGET}")
 | |
| 	TEST_STR=$(( $ORIG + 1 ))
 | |
| 
 | |
| 	run_numerictests
 | |
| 	run_wideint_tests
 | |
| 	run_limit_digit
 | |
| 	run_limit_digit_int
 | |
| }
 | |
| 
 | |
| sysctl_test_0004()
 | |
| {
 | |
| 	TARGET="${SYSCTL}/$(get_test_target 0004)"
 | |
| 	reset_vals
 | |
| 	ORIG=$(cat "${TARGET}")
 | |
| 	TEST_STR=$(( $ORIG + 1 ))
 | |
| 
 | |
| 	run_numerictests
 | |
| 	run_wideint_tests
 | |
| 	run_limit_digit
 | |
| 	run_limit_digit_uint
 | |
| }
 | |
| 
 | |
| sysctl_test_0005()
 | |
| {
 | |
| 	TARGET="${SYSCTL}/$(get_test_target 0005)"
 | |
| 	reset_vals
 | |
| 	ORIG=$(cat "${TARGET}")
 | |
| 
 | |
| 	run_limit_digit_int_array
 | |
| }
 | |
| 
 | |
| sysctl_test_0006()
 | |
| {
 | |
| 	TARGET="${SYSCTL}/bitmap_0001"
 | |
| 	reset_vals
 | |
| 	ORIG=""
 | |
| 	run_bitmaptest
 | |
| }
 | |
| 
 | |
| sysctl_test_0007()
 | |
| {
 | |
| 	TARGET="${SYSCTL}/boot_int"
 | |
| 	if [ ! -f $TARGET ]; then
 | |
| 		echo "Skipping test for $TARGET as it is not present ..."
 | |
| 		return $ksft_skip
 | |
| 	fi
 | |
| 
 | |
| 	if [ -d $DIR ]; then
 | |
| 		echo "Boot param test only possible sysctl_test is built-in, not module:"
 | |
| 		cat $TEST_DIR/config >&2
 | |
| 		return $ksft_skip
 | |
| 	fi
 | |
| 
 | |
| 	echo -n "Testing if $TARGET is set to 1 ..."
 | |
| 	ORIG=$(cat "${TARGET}")
 | |
| 
 | |
| 	if [ x$ORIG = "x1" ]; then
 | |
| 		echo "ok"
 | |
| 		return 0
 | |
| 	fi
 | |
| 	echo "FAIL"
 | |
| 	echo "Checking if /proc/cmdline contains setting of the expected parameter ..."
 | |
| 	if [ ! -f /proc/cmdline ]; then
 | |
| 		echo "/proc/cmdline does not exist, test inconclusive"
 | |
| 		return 0
 | |
| 	fi
 | |
| 
 | |
| 	FOUND=$(grep -c "sysctl[./]debug[./]test_sysctl[./]boot_int=1" /proc/cmdline)
 | |
| 	if [ $FOUND = "1" ]; then
 | |
| 		echo "Kernel param found but $TARGET is not 1, TEST FAILED"
 | |
| 		rc=1
 | |
| 		test_rc
 | |
| 	fi
 | |
| 
 | |
| 	echo "Skipping test, expected kernel parameter missing."
 | |
| 	echo "To perform this test, make sure kernel is booted with parameter: sysctl.debug.test_sysctl.boot_int=1"
 | |
| 	return $ksft_skip
 | |
| }
 | |
| 
 | |
| list_tests()
 | |
| {
 | |
| 	echo "Test ID list:"
 | |
| 	echo
 | |
| 	echo "TEST_ID x NUM_TEST"
 | |
| 	echo "TEST_ID:   Test ID"
 | |
| 	echo "NUM_TESTS: Number of recommended times to run the test"
 | |
| 	echo
 | |
| 	echo "0001 x $(get_test_count 0001) - tests proc_dointvec_minmax()"
 | |
| 	echo "0002 x $(get_test_count 0002) - tests proc_dostring()"
 | |
| 	echo "0003 x $(get_test_count 0003) - tests proc_dointvec()"
 | |
| 	echo "0004 x $(get_test_count 0004) - tests proc_douintvec()"
 | |
| 	echo "0005 x $(get_test_count 0005) - tests proc_douintvec() array"
 | |
| 	echo "0006 x $(get_test_count 0006) - tests proc_do_large_bitmap()"
 | |
| 	echo "0007 x $(get_test_count 0007) - tests setting sysctl from kernel boot param"
 | |
| }
 | |
| 
 | |
| usage()
 | |
| {
 | |
| 	NUM_TESTS=$(grep -o ' ' <<<"$ALL_TESTS" | grep -c .)
 | |
| 	let NUM_TESTS=$NUM_TESTS+1
 | |
| 	MAX_TEST=$(printf "%04d\n" $NUM_TESTS)
 | |
| 	echo "Usage: $0 [ -t <4-number-digit> ] | [ -w <4-number-digit> ] |"
 | |
| 	echo "		 [ -s <4-number-digit> ] | [ -c <4-number-digit> <test- count>"
 | |
| 	echo "           [ all ] [ -h | --help ] [ -l ]"
 | |
| 	echo ""
 | |
| 	echo "Valid tests: 0001-$MAX_TEST"
 | |
| 	echo ""
 | |
| 	echo "    all     Runs all tests (default)"
 | |
| 	echo "    -t      Run test ID the number amount of times is recommended"
 | |
| 	echo "    -w      Watch test ID run until it runs into an error"
 | |
| 	echo "    -c      Run test ID once"
 | |
| 	echo "    -s      Run test ID x test-count number of times"
 | |
| 	echo "    -l      List all test ID list"
 | |
| 	echo " -h|--help  Help"
 | |
| 	echo
 | |
| 	echo "If an error every occurs execution will immediately terminate."
 | |
| 	echo "If you are adding a new test try using -w <test-ID> first to"
 | |
| 	echo "make sure the test passes a series of tests."
 | |
| 	echo
 | |
| 	echo Example uses:
 | |
| 	echo
 | |
| 	echo "$TEST_NAME.sh            -- executes all tests"
 | |
| 	echo "$TEST_NAME.sh -t 0002    -- Executes test ID 0002 number of times is recomended"
 | |
| 	echo "$TEST_NAME.sh -w 0002    -- Watch test ID 0002 run until an error occurs"
 | |
| 	echo "$TEST_NAME.sh -s 0002    -- Run test ID 0002 once"
 | |
| 	echo "$TEST_NAME.sh -c 0002 3  -- Run test ID 0002 three times"
 | |
| 	echo
 | |
| 	list_tests
 | |
| 	exit 1
 | |
| }
 | |
| 
 | |
| function test_num()
 | |
| {
 | |
| 	re='^[0-9]+$'
 | |
| 	if ! [[ $1 =~ $re ]]; then
 | |
| 		usage
 | |
| 	fi
 | |
| }
 | |
| 
 | |
| function get_test_count()
 | |
| {
 | |
| 	test_num $1
 | |
| 	TEST_DATA=$(echo $ALL_TESTS | awk '{print $'$1'}')
 | |
| 	echo ${TEST_DATA} | awk -F":" '{print $2}'
 | |
| }
 | |
| 
 | |
| function get_test_enabled()
 | |
| {
 | |
| 	test_num $1
 | |
| 	TEST_DATA=$(echo $ALL_TESTS | awk '{print $'$1'}')
 | |
| 	echo ${TEST_DATA} | awk -F":" '{print $3}'
 | |
| }
 | |
| 
 | |
| function get_test_target()
 | |
| {
 | |
| 	test_num $1
 | |
| 	TEST_DATA=$(echo $ALL_TESTS | awk '{print $'$1'}')
 | |
| 	echo ${TEST_DATA} | awk -F":" '{print $4}'
 | |
| }
 | |
| 
 | |
| function run_all_tests()
 | |
| {
 | |
| 	for i in $ALL_TESTS ; do
 | |
| 		TEST_ID=${i%:*:*:*}
 | |
| 		ENABLED=$(get_test_enabled $TEST_ID)
 | |
| 		TEST_COUNT=$(get_test_count $TEST_ID)
 | |
| 		TEST_TARGET=$(get_test_target $TEST_ID)
 | |
| 		if target_exists $TEST_TARGET $TEST_ID; then
 | |
| 			continue
 | |
| 		fi
 | |
| 		if [[ $ENABLED -eq "1" ]]; then
 | |
| 			test_case $TEST_ID $TEST_COUNT $TEST_TARGET
 | |
| 		fi
 | |
| 	done
 | |
| }
 | |
| 
 | |
| function watch_log()
 | |
| {
 | |
| 	if [ $# -ne 3 ]; then
 | |
| 		clear
 | |
| 	fi
 | |
| 	date
 | |
| 	echo "Running test: $2 - run #$1"
 | |
| }
 | |
| 
 | |
| function watch_case()
 | |
| {
 | |
| 	i=0
 | |
| 	while [ 1 ]; do
 | |
| 
 | |
| 		if [ $# -eq 1 ]; then
 | |
| 			test_num $1
 | |
| 			watch_log $i ${TEST_NAME}_test_$1
 | |
| 			${TEST_NAME}_test_$1
 | |
| 		else
 | |
| 			watch_log $i all
 | |
| 			run_all_tests
 | |
| 		fi
 | |
| 		let i=$i+1
 | |
| 	done
 | |
| }
 | |
| 
 | |
| function test_case()
 | |
| {
 | |
| 	NUM_TESTS=$2
 | |
| 
 | |
| 	i=0
 | |
| 
 | |
| 	if target_exists $3 $1; then
 | |
| 		continue
 | |
| 	fi
 | |
| 
 | |
| 	while [ $i -lt $NUM_TESTS ]; do
 | |
| 		test_num $1
 | |
| 		watch_log $i ${TEST_NAME}_test_$1 noclear
 | |
| 		RUN_TEST=${TEST_NAME}_test_$1
 | |
| 		$RUN_TEST
 | |
| 		let i=$i+1
 | |
| 	done
 | |
| }
 | |
| 
 | |
| function parse_args()
 | |
| {
 | |
| 	if [ $# -eq 0 ]; then
 | |
| 		run_all_tests
 | |
| 	else
 | |
| 		if [[ "$1" = "all" ]]; then
 | |
| 			run_all_tests
 | |
| 		elif [[ "$1" = "-w" ]]; then
 | |
| 			shift
 | |
| 			watch_case $@
 | |
| 		elif [[ "$1" = "-t" ]]; then
 | |
| 			shift
 | |
| 			test_num $1
 | |
| 			test_case $1 $(get_test_count $1) $(get_test_target $1)
 | |
| 		elif [[ "$1" = "-c" ]]; then
 | |
| 			shift
 | |
| 			test_num $1
 | |
| 			test_num $2
 | |
| 			test_case $1 $2 $(get_test_target $1)
 | |
| 		elif [[ "$1" = "-s" ]]; then
 | |
| 			shift
 | |
| 			test_case $1 1 $(get_test_target $1)
 | |
| 		elif [[ "$1" = "-l" ]]; then
 | |
| 			list_tests
 | |
| 		elif [[ "$1" = "-h" || "$1" = "--help" ]]; then
 | |
| 			usage
 | |
| 		else
 | |
| 			usage
 | |
| 		fi
 | |
| 	fi
 | |
| }
 | |
| 
 | |
| test_reqs
 | |
| allow_user_defaults
 | |
| check_production_sysctl_writes_strict
 | |
| load_req_mod
 | |
| 
 | |
| trap "test_finish" EXIT
 | |
| 
 | |
| parse_args $@
 | |
| 
 | |
| exit 0
 |