123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546 |
- #!/bin/bash
- # SPDX-License-Identifier: GPL-2.0
- # This validates that the kernel will load firmware out of its list of
- # firmware locations on disk. Since the user helper does similar work,
- # we reset the custom load directory to a location the user helper doesn't
- # know so we can be sure we're not accidentally testing the user helper.
- set -e
- TEST_REQS_FW_SYSFS_FALLBACK="no"
- TEST_REQS_FW_SET_CUSTOM_PATH="yes"
- TEST_DIR=$(dirname $0)
- source $TEST_DIR/fw_lib.sh
- RUN_XZ="xz -C crc32 --lzma2=dict=2MiB"
- RUN_ZSTD="zstd -q"
- check_mods
- check_setup
- verify_reqs
- setup_tmp_file
- trap "test_finish" EXIT
- if [ "$HAS_FW_LOADER_USER_HELPER" = "yes" ]; then
- # Turn down the timeout so failures don't take so long.
- echo 1 >/sys/class/firmware/timeout
- fi
- if printf '\000' >"$DIR"/trigger_request 2> /dev/null; then
- echo "$0: empty filename should not succeed" >&2
- exit 1
- fi
- if [ ! -e "$DIR"/trigger_async_request ]; then
- echo "$0: empty filename: async trigger not present, ignoring test" >&2
- exit $ksft_skip
- else
- if printf '\000' >"$DIR"/trigger_async_request 2> /dev/null; then
- echo "$0: empty filename should not succeed (async)" >&2
- exit 1
- fi
- fi
- # Request a firmware that doesn't exist, it should fail.
- if echo -n "nope-$NAME" >"$DIR"/trigger_request 2> /dev/null; then
- echo "$0: firmware shouldn't have loaded" >&2
- exit 1
- fi
- if diff -q "$FW" /dev/test_firmware >/dev/null ; then
- echo "$0: firmware was not expected to match" >&2
- exit 1
- else
- if [ "$HAS_FW_LOADER_USER_HELPER" = "yes" ]; then
- echo "$0: timeout works"
- fi
- fi
- # This should succeed via kernel load or will fail after 1 second after
- # being handed over to the user helper, which won't find the fw either.
- if ! echo -n "$NAME" >"$DIR"/trigger_request ; then
- echo "$0: could not trigger request" >&2
- exit 1
- fi
- # Verify the contents are what we expect.
- if ! diff -q "$FW" /dev/test_firmware >/dev/null ; then
- echo "$0: firmware was not loaded" >&2
- exit 1
- else
- echo "$0: filesystem loading works"
- fi
- # Try the asynchronous version too
- if [ ! -e "$DIR"/trigger_async_request ]; then
- echo "$0: firmware loading: async trigger not present, ignoring test" >&2
- exit $ksft_skip
- else
- if ! echo -n "$NAME" >"$DIR"/trigger_async_request ; then
- echo "$0: could not trigger async request" >&2
- exit 1
- fi
- # Verify the contents are what we expect.
- if ! diff -q "$FW" /dev/test_firmware >/dev/null ; then
- echo "$0: firmware was not loaded (async)" >&2
- exit 1
- else
- echo "$0: async filesystem loading works"
- fi
- fi
- # Try platform (EFI embedded fw) loading too
- if [ ! -e "$DIR"/trigger_request_platform ]; then
- echo "$0: firmware loading: platform trigger not present, ignoring test" >&2
- else
- if printf '\000' >"$DIR"/trigger_request_platform 2> /dev/null; then
- echo "$0: empty filename should not succeed (platform)" >&2
- exit 1
- fi
- # Note we echo a non-existing name, since files on the file-system
- # are preferred over firmware embedded inside the platform's firmware
- # The test adds a fake entry with the requested name to the platform's
- # fw list, so the name does not matter as long as it does not exist
- if ! echo -n "nope-$NAME" >"$DIR"/trigger_request_platform ; then
- echo "$0: could not trigger request platform" >&2
- exit 1
- fi
- # The test verifies itself that the loaded firmware contents matches
- # the contents for the fake platform fw entry it added.
- echo "$0: platform loading works"
- fi
- ### Batched requests tests
- test_config_present()
- {
- if [ ! -f $DIR/reset ]; then
- echo "Configuration triggers not present, ignoring test"
- exit $ksft_skip
- fi
- }
- # Defaults :
- #
- # send_uevent: 1
- # sync_direct: 0
- # name: test-firmware.bin
- # num_requests: 4
- config_reset()
- {
- echo 1 > $DIR/reset
- }
- release_all_firmware()
- {
- echo 1 > $DIR/release_all_firmware
- }
- config_set_name()
- {
- echo -n $1 > $DIR/config_name
- }
- config_set_into_buf()
- {
- echo 1 > $DIR/config_into_buf
- }
- config_unset_into_buf()
- {
- echo 0 > $DIR/config_into_buf
- }
- config_set_buf_size()
- {
- echo $1 > $DIR/config_buf_size
- }
- config_set_file_offset()
- {
- echo $1 > $DIR/config_file_offset
- }
- config_set_partial()
- {
- echo 1 > $DIR/config_partial
- }
- config_unset_partial()
- {
- echo 0 > $DIR/config_partial
- }
- config_set_sync_direct()
- {
- echo 1 > $DIR/config_sync_direct
- }
- config_unset_sync_direct()
- {
- echo 0 > $DIR/config_sync_direct
- }
- config_set_uevent()
- {
- echo 1 > $DIR/config_send_uevent
- }
- config_unset_uevent()
- {
- echo 0 > $DIR/config_send_uevent
- }
- config_trigger_sync()
- {
- echo -n 1 > $DIR/trigger_batched_requests 2>/dev/null
- }
- config_trigger_async()
- {
- echo -n 1 > $DIR/trigger_batched_requests_async 2> /dev/null
- }
- config_set_read_fw_idx()
- {
- echo -n $1 > $DIR/config_read_fw_idx 2> /dev/null
- }
- read_firmwares()
- {
- if [ "$(cat $DIR/config_into_buf)" == "1" ]; then
- fwfile="$FW_INTO_BUF"
- else
- fwfile="$FW"
- fi
- if [ "$1" = "componly" ]; then
- fwfile="${fwfile}-orig"
- fi
- for i in $(seq 0 3); do
- config_set_read_fw_idx $i
- # Verify the contents are what we expect.
- # -Z required for now -- check for yourself, md5sum
- # on $FW and DIR/read_firmware will yield the same. Even
- # cmp agrees, so something is off.
- if ! diff -q -Z "$fwfile" $DIR/read_firmware 2>/dev/null ; then
- echo "request #$i: firmware was not loaded" >&2
- exit 1
- fi
- done
- }
- read_partial_firmwares()
- {
- if [ "$(cat $DIR/config_into_buf)" == "1" ]; then
- fwfile="${FW_INTO_BUF}"
- else
- fwfile="${FW}"
- fi
- if [ "$1" = "componly" ]; then
- fwfile="${fwfile}-orig"
- fi
- # Strip fwfile down to match partial offset and length
- partial_data="$(cat $fwfile)"
- partial_data="${partial_data:$2:$3}"
- for i in $(seq 0 3); do
- config_set_read_fw_idx $i
- read_firmware="$(cat $DIR/read_firmware)"
- # Verify the contents are what we expect.
- if [ $read_firmware != $partial_data ]; then
- echo "request #$i: partial firmware was not loaded" >&2
- exit 1
- fi
- done
- }
- read_firmwares_expect_nofile()
- {
- for i in $(seq 0 3); do
- config_set_read_fw_idx $i
- # Ensures contents differ
- if diff -q -Z "$FW" $DIR/read_firmware 2>/dev/null ; then
- echo "request $i: file was not expected to match" >&2
- exit 1
- fi
- done
- }
- test_batched_request_firmware_nofile()
- {
- echo -n "Batched request_firmware() nofile try #$1: "
- config_reset
- config_set_name nope-test-firmware.bin
- config_trigger_sync
- read_firmwares_expect_nofile
- release_all_firmware
- echo "OK"
- }
- test_batched_request_firmware_into_buf_nofile()
- {
- echo -n "Batched request_firmware_into_buf() nofile try #$1: "
- config_reset
- config_set_name nope-test-firmware.bin
- config_set_into_buf
- config_trigger_sync
- read_firmwares_expect_nofile
- release_all_firmware
- echo "OK"
- }
- test_request_partial_firmware_into_buf_nofile()
- {
- echo -n "Test request_partial_firmware_into_buf() off=$1 size=$2 nofile: "
- config_reset
- config_set_name nope-test-firmware.bin
- config_set_into_buf
- config_set_partial
- config_set_buf_size $2
- config_set_file_offset $1
- config_trigger_sync
- read_firmwares_expect_nofile
- release_all_firmware
- echo "OK"
- }
- test_batched_request_firmware_direct_nofile()
- {
- echo -n "Batched request_firmware_direct() nofile try #$1: "
- config_reset
- config_set_name nope-test-firmware.bin
- config_set_sync_direct
- config_trigger_sync
- release_all_firmware
- echo "OK"
- }
- test_request_firmware_nowait_uevent_nofile()
- {
- echo -n "Batched request_firmware_nowait(uevent=true) nofile try #$1: "
- config_reset
- config_set_name nope-test-firmware.bin
- config_trigger_async
- release_all_firmware
- echo "OK"
- }
- test_wait_and_cancel_custom_load()
- {
- if [ "$HAS_FW_LOADER_USER_HELPER" != "yes" ]; then
- return
- fi
- local timeout=10
- name=$1
- while [ ! -e "$DIR"/"$name"/loading ]; do
- sleep 0.1
- timeout=$(( $timeout - 1 ))
- if [ "$timeout" -eq 0 ]; then
- echo "firmware interface never appeared:" >&2
- echo "$DIR/$name/loading" >&2
- exit 1
- fi
- done
- echo -1 >"$DIR"/"$name"/loading
- }
- test_request_firmware_nowait_custom_nofile()
- {
- echo -n "Batched request_firmware_nowait(uevent=false) nofile try #$1: "
- config_reset
- config_unset_uevent
- RANDOM_FILE_PATH=$(setup_random_file_fake)
- RANDOM_FILE="$(basename $RANDOM_FILE_PATH)"
- config_set_name $RANDOM_FILE
- config_trigger_async &
- test_wait_and_cancel_custom_load $RANDOM_FILE
- wait
- release_all_firmware
- echo "OK"
- }
- test_batched_request_firmware()
- {
- echo -n "Batched request_firmware() $2 try #$1: "
- config_reset
- config_trigger_sync
- read_firmwares $2
- release_all_firmware
- echo "OK"
- }
- test_batched_request_firmware_into_buf()
- {
- echo -n "Batched request_firmware_into_buf() $2 try #$1: "
- config_reset
- config_set_name $TEST_FIRMWARE_INTO_BUF_FILENAME
- config_set_into_buf
- config_trigger_sync
- read_firmwares $2
- release_all_firmware
- echo "OK"
- }
- test_batched_request_firmware_direct()
- {
- echo -n "Batched request_firmware_direct() $2 try #$1: "
- config_reset
- config_set_sync_direct
- config_trigger_sync
- release_all_firmware
- echo "OK"
- }
- test_request_firmware_nowait_uevent()
- {
- echo -n "Batched request_firmware_nowait(uevent=true) $2 try #$1: "
- config_reset
- config_trigger_async
- release_all_firmware
- echo "OK"
- }
- test_request_firmware_nowait_custom()
- {
- echo -n "Batched request_firmware_nowait(uevent=false) $2 try #$1: "
- config_reset
- config_unset_uevent
- RANDOM_FILE_PATH=$(setup_random_file)
- RANDOM_FILE="$(basename $RANDOM_FILE_PATH)"
- if [ -n "$2" -a "$2" != "normal" ]; then
- compress_"$2"_"$COMPRESS_FORMAT" $RANDOM_FILE_PATH
- fi
- config_set_name $RANDOM_FILE
- config_trigger_async
- release_all_firmware
- echo "OK"
- }
- test_request_partial_firmware_into_buf()
- {
- echo -n "Test request_partial_firmware_into_buf() off=$1 size=$2: "
- config_reset
- config_set_name $TEST_FIRMWARE_INTO_BUF_FILENAME
- config_set_into_buf
- config_set_partial
- config_set_buf_size $2
- config_set_file_offset $1
- config_trigger_sync
- read_partial_firmwares normal $1 $2
- release_all_firmware
- echo "OK"
- }
- do_tests ()
- {
- mode="$1"
- suffix="$2"
- for i in $(seq 1 5); do
- test_batched_request_firmware$suffix $i $mode
- done
- for i in $(seq 1 5); do
- test_batched_request_firmware_into_buf$suffix $i $mode
- done
- for i in $(seq 1 5); do
- test_batched_request_firmware_direct$suffix $i $mode
- done
- for i in $(seq 1 5); do
- test_request_firmware_nowait_uevent$suffix $i $mode
- done
- for i in $(seq 1 5); do
- test_request_firmware_nowait_custom$suffix $i $mode
- done
- }
- # Only continue if batched request triggers are present on the
- # test-firmware driver
- test_config_present
- # test with the file present
- echo
- echo "Testing with the file present..."
- do_tests normal
- # Partial loads cannot use fallback, so do not repeat tests.
- test_request_partial_firmware_into_buf 0 10
- test_request_partial_firmware_into_buf 0 5
- test_request_partial_firmware_into_buf 1 6
- test_request_partial_firmware_into_buf 2 10
- # Test for file not found, errors are expected, the failure would be
- # a hung task, which would require a hard reset.
- echo
- echo "Testing with the file missing..."
- do_tests nofile _nofile
- # Partial loads cannot use fallback, so do not repeat tests.
- test_request_partial_firmware_into_buf_nofile 0 10
- test_request_partial_firmware_into_buf_nofile 0 5
- test_request_partial_firmware_into_buf_nofile 1 6
- test_request_partial_firmware_into_buf_nofile 2 10
- test_request_firmware_compressed ()
- {
- export COMPRESS_FORMAT="$1"
- # test with both files present
- compress_both_"$COMPRESS_FORMAT" $FW
- compress_both_"$COMPRESS_FORMAT" $FW_INTO_BUF
- config_set_name $NAME
- echo
- echo "Testing with both plain and $COMPRESS_FORMAT files present..."
- do_tests both
- # test with only compressed file present
- mv "$FW" "${FW}-orig"
- mv "$FW_INTO_BUF" "${FW_INTO_BUF}-orig"
- config_set_name $NAME
- echo
- echo "Testing with only $COMPRESS_FORMAT file present..."
- do_tests componly
- mv "${FW}-orig" "$FW"
- mv "${FW_INTO_BUF}-orig" "$FW_INTO_BUF"
- }
- compress_both_XZ ()
- {
- $RUN_XZ -k "$@"
- }
- compress_componly_XZ ()
- {
- $RUN_XZ "$@"
- }
- compress_both_ZSTD ()
- {
- $RUN_ZSTD -k "$@"
- }
- compress_componly_ZSTD ()
- {
- $RUN_ZSTD --rm "$@"
- }
- if test "$HAS_FW_LOADER_COMPRESS_XZ" = "yes"; then
- test_request_firmware_compressed XZ
- fi
- if test "$HAS_FW_LOADER_COMPRESS_ZSTD" = "yes"; then
- test_request_firmware_compressed ZSTD
- fi
- exit 0
|