linux/tools/testing/selftests/efivarfs/efivarfs.sh
James Bottomley 87e6cd7cdb selftests/efivarfs: add concurrent update tests
The delete on last close functionality can now only be tested properly
by using multiple threads to hold open the variable files and testing
what happens as they complete.

Signed-off-by: James Bottomley <James.Bottomley@HansenPartnership.com>
Signed-off-by: Ard Biesheuvel <ardb@kernel.org>
2025-01-21 16:34:41 +01:00

379 lines
7.7 KiB
Bash
Executable File

#!/bin/bash
# SPDX-License-Identifier: GPL-2.0
efivarfs_mount=/sys/firmware/efi/efivars
test_guid=210be57c-9849-4fc7-a635-e6382d1aec27
# Kselftest framework requirement - SKIP code is 4.
ksft_skip=4
file_cleanup()
{
chattr -i $1
rm -f $1
}
check_prereqs()
{
local msg="skip all tests:"
if [ $UID != 0 ]; then
echo $msg must be run as root >&2
exit $ksft_skip
fi
if ! grep -q "^\S\+ $efivarfs_mount efivarfs" /proc/mounts; then
echo $msg efivarfs is not mounted on $efivarfs_mount >&2
exit $ksft_skip
fi
}
run_test()
{
local test="$1"
echo "--------------------"
echo "running $test"
echo "--------------------"
if [ "$(type -t $test)" = 'function' ]; then
( $test )
else
( ./$test )
fi
if [ $? -ne 0 ]; then
echo " [FAIL]"
rc=1
else
echo " [PASS]"
fi
}
test_create()
{
local attrs='\x07\x00\x00\x00'
local file=$efivarfs_mount/$FUNCNAME-$test_guid
printf "$attrs\x00" > $file
if [ ! -e $file ]; then
echo "$file couldn't be created" >&2
exit 1
fi
if [ $(stat -c %s $file) -ne 5 ]; then
echo "$file has invalid size" >&2
file_cleanup $file
exit 1
fi
file_cleanup $file
}
test_create_empty()
{
local file=$efivarfs_mount/$FUNCNAME-$test_guid
: > $file
if [ -e $file ]; then
echo "$file can be created without writing" >&2
file_cleanup $file
exit 1
fi
}
test_create_read()
{
local file=$efivarfs_mount/$FUNCNAME-$test_guid
./create-read $file
if [ $? -ne 0 ]; then
echo "create and read $file failed"
exit 1
fi
if [ -e $file ]; then
echo "file still exists and should not"
file_cleanup $file
exit 1
fi
}
test_delete()
{
local attrs='\x07\x00\x00\x00'
local file=$efivarfs_mount/$FUNCNAME-$test_guid
printf "$attrs\x00" > $file
if [ ! -e $file ]; then
echo "$file couldn't be created" >&2
exit 1
fi
file_cleanup $file
if [ -e $file ]; then
echo "$file couldn't be deleted" >&2
exit 1
fi
}
# test that we can remove a variable by issuing a write with only
# attributes specified
test_zero_size_delete()
{
local attrs='\x07\x00\x00\x00'
local file=$efivarfs_mount/$FUNCNAME-$test_guid
printf "$attrs\x00" > $file
if [ ! -e $file ]; then
echo "$file does not exist" >&2
exit 1
fi
chattr -i $file
printf "$attrs" > $file
if [ -e $file ]; then
echo "$file should have been deleted" >&2
exit 1
fi
}
test_open_unlink()
{
local file=$efivarfs_mount/$FUNCNAME-$test_guid
./open-unlink $file
}
# test that we can create a range of filenames
test_valid_filenames()
{
local attrs='\x07\x00\x00\x00'
local ret=0
local file_list="abc dump-type0-11-1-1362436005 1234 -"
for f in $file_list; do
local file=$efivarfs_mount/$f-$test_guid
printf "$attrs\x00" > $file
if [ ! -e $file ]; then
echo "$file could not be created" >&2
ret=1
else
file_cleanup $file
fi
done
exit $ret
}
test_invalid_filenames()
{
local attrs='\x07\x00\x00\x00'
local ret=0
local file_list="
-1234-1234-1234-123456789abc
foo
foo-bar
-foo-
foo-barbazba-foob-foob-foob-foobarbazfoo
foo-------------------------------------
-12345678-1234-1234-1234-123456789abc
a-12345678=1234-1234-1234-123456789abc
a-12345678-1234=1234-1234-123456789abc
a-12345678-1234-1234=1234-123456789abc
a-12345678-1234-1234-1234=123456789abc
1112345678-1234-1234-1234-123456789abc"
for f in $file_list; do
local file=$efivarfs_mount/$f
printf "$attrs\x00" 2>/dev/null > $file
if [ -e $file ]; then
echo "Creating $file should have failed" >&2
file_cleanup $file
ret=1
fi
done
exit $ret
}
test_no_set_size()
{
local attrs='\x07\x00\x00\x00'
local file=$efivarfs_mount/$FUNCNAME-$test_guid
local ret=0
printf "$attrs\x00" > $file
[ -e $file -a -s $file ] || exit 1
chattr -i $file
: > $file
if [ $? != 0 ]; then
echo "variable file failed to accept truncation"
ret=1
elif [ -e $file -a ! -s $file ]; then
echo "file can be truncated to zero size"
ret=1
fi
rm $file || exit 1
exit $ret
}
setup_test_multiple()
{
##
# we're going to do multi-threaded tests, so create a set of
# pipes for synchronization. We use pipes 1..3 to start the
# stalled shell job and pipes 4..6 as indicators that the job
# has started. If you need more than 3 jobs the two +3's below
# need increasing
##
declare -ag p
# empty is because arrays number from 0 but jobs number from 1
p[0]=""
for f in 1 2 3 4 5 6; do
p[$f]=/tmp/efivarfs_pipe${f}
mknod ${p[$f]} p
done
declare -g var=$efivarfs_mount/test_multiple-$test_guid
cleanup() {
for f in ${p[@]}; do
rm -f ${f}
done
if [ -e $var ]; then
file_cleanup $var
fi
}
trap cleanup exit
waitstart() {
cat ${p[$[$1+3]]} > /dev/null
}
waitpipe() {
echo 1 > ${p[$[$1+3]]}
cat ${p[$1]} > /dev/null
}
endjob() {
echo 1 > ${p[$1]}
wait -n %$1
}
}
test_multiple_zero_size()
{
##
# check for remove on last close, set up three threads all
# holding the variable (one write and two reads) and then
# close them sequentially (waiting for completion) and check
# the state of the variable
##
{ waitpipe 1; echo 1; } > $var 2> /dev/null &
waitstart 1
# zero length file should exist
[ -e $var ] || exit 1
# second and third delayed close
{ waitpipe 2; } < $var &
waitstart 2
{ waitpipe 3; } < $var &
waitstart 3
# close first fd
endjob 1
# var should only be deleted on last close
[ -e $var ] || exit 1
# close second fd
endjob 2
[ -e $var ] || exit 1
# file should go on last close
endjob 3
[ ! -e $var ] || exit 1
}
test_multiple_create()
{
##
# set multiple threads to access the variable but delay
# the final write to check the close of 2 and 3. The
# final write should succeed in creating the variable
##
{ waitpipe 1; printf '\x07\x00\x00\x00\x54'; } > $var &
waitstart 1
[ -e $var -a ! -s $var ] || exit 1
{ waitpipe 2; } < $var &
waitstart 2
{ waitpipe 3; } < $var &
waitstart 3
# close second and third fds
endjob 2
# var should only be created (have size) on last close
[ -e $var -a ! -s $var ] || exit 1
endjob 3
[ -e $var -a ! -s $var ] || exit 1
# close first fd
endjob 1
# variable should still exist
[ -s $var ] || exit 1
file_cleanup $var
}
test_multiple_delete_on_write() {
##
# delete the variable on final write; seqencing similar
# to test_multiple_create()
##
printf '\x07\x00\x00\x00\x54' > $var
chattr -i $var
{ waitpipe 1; printf '\x07\x00\x00\x00'; } > $var &
waitstart 1
[ -e $var -a -s $var ] || exit 1
{ waitpipe 2; } < $var &
waitstart 2
{ waitpipe 3; } < $var &
waitstart 3
# close first fd; write should set variable size to zero
endjob 1
# var should only be deleted on last close
[ -e $var -a ! -s $var ] || exit 1
endjob 2
[ -e $var ] || exit 1
# close last fd
endjob 3
# variable should now be removed
[ ! -e $var ] || exit 1
}
check_prereqs
rc=0
run_test test_create
run_test test_create_empty
run_test test_create_read
run_test test_delete
run_test test_zero_size_delete
run_test test_open_unlink
run_test test_valid_filenames
run_test test_invalid_filenames
run_test test_no_set_size
setup_test_multiple
run_test test_multiple_zero_size
run_test test_multiple_create
run_test test_multiple_delete_on_write
exit $rc