129a1c00cSLuis R. Rodriguez#!/bin/bash
229a1c00cSLuis R. Rodriguez# SPDX-License-Identifier: GPL-2.0
329a1c00cSLuis R. Rodriguez
429a1c00cSLuis R. Rodriguez# Library of helpers for test scripts.
529a1c00cSLuis R. Rodriguezset -e
629a1c00cSLuis R. Rodriguez
729a1c00cSLuis R. RodriguezDIR=/sys/devices/virtual/misc/test_firmware
829a1c00cSLuis R. Rodriguez
929a1c00cSLuis R. RodriguezPROC_CONFIG="/proc/config.gz"
1029a1c00cSLuis R. RodriguezTEST_DIR=$(dirname $0)
1129a1c00cSLuis R. Rodriguez
12a6a9be92SShuah Khan (Samsung OSG)# Kselftest framework requirement - SKIP code is 4.
13a6a9be92SShuah Khan (Samsung OSG)ksft_skip=4
14a6a9be92SShuah Khan (Samsung OSG)
1529a1c00cSLuis R. Rodriguezprint_reqs_exit()
1629a1c00cSLuis R. Rodriguez{
1729a1c00cSLuis R. Rodriguez	echo "You must have the following enabled in your kernel:" >&2
1829a1c00cSLuis R. Rodriguez	cat $TEST_DIR/config >&2
19a6a9be92SShuah Khan (Samsung OSG)	exit $ksft_skip
2029a1c00cSLuis R. Rodriguez}
2129a1c00cSLuis R. Rodriguez
2229a1c00cSLuis R. Rodrigueztest_modprobe()
2329a1c00cSLuis R. Rodriguez{
2429a1c00cSLuis R. Rodriguez	if [ ! -d $DIR ]; then
2529a1c00cSLuis R. Rodriguez		print_reqs_exit
2629a1c00cSLuis R. Rodriguez	fi
2729a1c00cSLuis R. Rodriguez}
2829a1c00cSLuis R. Rodriguez
2929a1c00cSLuis R. Rodriguezcheck_mods()
3029a1c00cSLuis R. Rodriguez{
3129a1c00cSLuis R. Rodriguez	trap "test_modprobe" EXIT
3229a1c00cSLuis R. Rodriguez	if [ ! -d $DIR ]; then
3329a1c00cSLuis R. Rodriguez		modprobe test_firmware
3429a1c00cSLuis R. Rodriguez	fi
3529a1c00cSLuis R. Rodriguez	if [ ! -f $PROC_CONFIG ]; then
3629a1c00cSLuis R. Rodriguez		if modprobe configs 2>/dev/null; then
3729a1c00cSLuis R. Rodriguez			echo "Loaded configs module"
3829a1c00cSLuis R. Rodriguez			if [ ! -f $PROC_CONFIG ]; then
3929a1c00cSLuis R. Rodriguez				echo "You must have the following enabled in your kernel:" >&2
4029a1c00cSLuis R. Rodriguez				cat $TEST_DIR/config >&2
4129a1c00cSLuis R. Rodriguez				echo "Resorting to old heuristics" >&2
4229a1c00cSLuis R. Rodriguez			fi
4329a1c00cSLuis R. Rodriguez		else
4429a1c00cSLuis R. Rodriguez			echo "Failed to load configs module, using old heuristics" >&2
4529a1c00cSLuis R. Rodriguez		fi
4629a1c00cSLuis R. Rodriguez	fi
4729a1c00cSLuis R. Rodriguez}
48ef557787SLuis R. Rodriguez
49f5a61451SLuis R. Rodriguezcheck_setup()
50f5a61451SLuis R. Rodriguez{
51f5a61451SLuis R. Rodriguez	HAS_FW_LOADER_USER_HELPER="$(kconfig_has CONFIG_FW_LOADER_USER_HELPER=y)"
52f5a61451SLuis R. Rodriguez	HAS_FW_LOADER_USER_HELPER_FALLBACK="$(kconfig_has CONFIG_FW_LOADER_USER_HELPER_FALLBACK=y)"
53b3cf21faSLuis R. Rodriguez	PROC_FW_IGNORE_SYSFS_FALLBACK="0"
54b3cf21faSLuis R. Rodriguez	PROC_FW_FORCE_SYSFS_FALLBACK="0"
55b3cf21faSLuis R. Rodriguez
56b3cf21faSLuis R. Rodriguez	if [ -z $PROC_SYS_DIR ]; then
57b3cf21faSLuis R. Rodriguez		PROC_SYS_DIR="/proc/sys/kernel"
58b3cf21faSLuis R. Rodriguez	fi
59b3cf21faSLuis R. Rodriguez
60b3cf21faSLuis R. Rodriguez	FW_PROC="${PROC_SYS_DIR}/firmware_config"
61b3cf21faSLuis R. Rodriguez	FW_FORCE_SYSFS_FALLBACK="$FW_PROC/force_sysfs_fallback"
62b3cf21faSLuis R. Rodriguez	FW_IGNORE_SYSFS_FALLBACK="$FW_PROC/ignore_sysfs_fallback"
63b3cf21faSLuis R. Rodriguez
64b3cf21faSLuis R. Rodriguez	if [ -f $FW_FORCE_SYSFS_FALLBACK ]; then
65b3cf21faSLuis R. Rodriguez		PROC_FW_FORCE_SYSFS_FALLBACK="$(cat $FW_FORCE_SYSFS_FALLBACK)"
66b3cf21faSLuis R. Rodriguez	fi
67b3cf21faSLuis R. Rodriguez
68b3cf21faSLuis R. Rodriguez	if [ -f $FW_IGNORE_SYSFS_FALLBACK ]; then
69b3cf21faSLuis R. Rodriguez		PROC_FW_IGNORE_SYSFS_FALLBACK="$(cat $FW_IGNORE_SYSFS_FALLBACK)"
70b3cf21faSLuis R. Rodriguez	fi
71b3cf21faSLuis R. Rodriguez
72b3cf21faSLuis R. Rodriguez	if [ "$PROC_FW_FORCE_SYSFS_FALLBACK" = "1" ]; then
73b3cf21faSLuis R. Rodriguez		HAS_FW_LOADER_USER_HELPER="yes"
74b3cf21faSLuis R. Rodriguez		HAS_FW_LOADER_USER_HELPER_FALLBACK="yes"
75b3cf21faSLuis R. Rodriguez	fi
76b3cf21faSLuis R. Rodriguez
77b3cf21faSLuis R. Rodriguez	if [ "$PROC_FW_IGNORE_SYSFS_FALLBACK" = "1" ]; then
78b3cf21faSLuis R. Rodriguez		HAS_FW_LOADER_USER_HELPER_FALLBACK="no"
79b3cf21faSLuis R. Rodriguez		HAS_FW_LOADER_USER_HELPER="no"
80b3cf21faSLuis R. Rodriguez	fi
81f5a61451SLuis R. Rodriguez
82f5a61451SLuis R. Rodriguez	if [ "$HAS_FW_LOADER_USER_HELPER" = "yes" ]; then
83f5a61451SLuis R. Rodriguez	       OLD_TIMEOUT="$(cat /sys/class/firmware/timeout)"
84f5a61451SLuis R. Rodriguez	fi
85f5a61451SLuis R. Rodriguez
86f5a61451SLuis R. Rodriguez	OLD_FWPATH="$(cat /sys/module/firmware_class/parameters/path)"
87f5a61451SLuis R. Rodriguez}
88f5a61451SLuis R. Rodriguez
89f5a61451SLuis R. Rodriguezverify_reqs()
90f5a61451SLuis R. Rodriguez{
91f5a61451SLuis R. Rodriguez	if [ "$TEST_REQS_FW_SYSFS_FALLBACK" = "yes" ]; then
92f5a61451SLuis R. Rodriguez		if [ ! "$HAS_FW_LOADER_USER_HELPER" = "yes" ]; then
93f5a61451SLuis R. Rodriguez			echo "usermode helper disabled so ignoring test"
94344c0152SLuis Chamberlain			exit 0
95f5a61451SLuis R. Rodriguez		fi
96f5a61451SLuis R. Rodriguez	fi
97f5a61451SLuis R. Rodriguez}
98f5a61451SLuis R. Rodriguez
99f5a61451SLuis R. Rodriguezsetup_tmp_file()
100f5a61451SLuis R. Rodriguez{
101f5a61451SLuis R. Rodriguez	FWPATH=$(mktemp -d)
102f5a61451SLuis R. Rodriguez	FW="$FWPATH/test-firmware.bin"
103f5a61451SLuis R. Rodriguez	echo "ABCD0123" >"$FW"
104f5a61451SLuis R. Rodriguez	NAME=$(basename "$FW")
105f5a61451SLuis R. Rodriguez	if [ "$TEST_REQS_FW_SET_CUSTOM_PATH" = "yes" ]; then
106f5a61451SLuis R. Rodriguez		echo -n "$FWPATH" >/sys/module/firmware_class/parameters/path
107f5a61451SLuis R. Rodriguez	fi
108f5a61451SLuis R. Rodriguez}
109f5a61451SLuis R. Rodriguez
1109952db75SLuis R. Rodriguez__setup_random_file()
1119952db75SLuis R. Rodriguez{
1129952db75SLuis R. Rodriguez	RANDOM_FILE_PATH="$(mktemp -p $FWPATH)"
1139952db75SLuis R. Rodriguez	# mktemp says dry-run -n is unsafe, so...
1149952db75SLuis R. Rodriguez	if [[ "$1" = "fake" ]]; then
1159952db75SLuis R. Rodriguez		rm -rf $RANDOM_FILE_PATH
1169952db75SLuis R. Rodriguez		sync
1179952db75SLuis R. Rodriguez	else
1189952db75SLuis R. Rodriguez		echo "ABCD0123" >"$RANDOM_FILE_PATH"
1199952db75SLuis R. Rodriguez	fi
1209952db75SLuis R. Rodriguez	echo $RANDOM_FILE_PATH
1219952db75SLuis R. Rodriguez}
1229952db75SLuis R. Rodriguez
1239952db75SLuis R. Rodriguezsetup_random_file()
1249952db75SLuis R. Rodriguez{
1259952db75SLuis R. Rodriguez	echo $(__setup_random_file)
1269952db75SLuis R. Rodriguez}
1279952db75SLuis R. Rodriguez
1289952db75SLuis R. Rodriguezsetup_random_file_fake()
1299952db75SLuis R. Rodriguez{
1309952db75SLuis R. Rodriguez	echo $(__setup_random_file fake)
1319952db75SLuis R. Rodriguez}
1329952db75SLuis R. Rodriguez
133b3cf21faSLuis R. Rodriguezproc_set_force_sysfs_fallback()
134b3cf21faSLuis R. Rodriguez{
135b3cf21faSLuis R. Rodriguez	if [ -f $FW_FORCE_SYSFS_FALLBACK ]; then
136b3cf21faSLuis R. Rodriguez		echo -n $1 > $FW_FORCE_SYSFS_FALLBACK
137b3cf21faSLuis R. Rodriguez		check_setup
138b3cf21faSLuis R. Rodriguez	fi
139b3cf21faSLuis R. Rodriguez}
140b3cf21faSLuis R. Rodriguez
141b3cf21faSLuis R. Rodriguezproc_set_ignore_sysfs_fallback()
142b3cf21faSLuis R. Rodriguez{
143b3cf21faSLuis R. Rodriguez	if [ -f $FW_IGNORE_SYSFS_FALLBACK ]; then
144b3cf21faSLuis R. Rodriguez		echo -n $1 > $FW_IGNORE_SYSFS_FALLBACK
145b3cf21faSLuis R. Rodriguez		check_setup
146b3cf21faSLuis R. Rodriguez	fi
147b3cf21faSLuis R. Rodriguez}
148b3cf21faSLuis R. Rodriguez
149b3cf21faSLuis R. Rodriguezproc_restore_defaults()
150b3cf21faSLuis R. Rodriguez{
151b3cf21faSLuis R. Rodriguez	proc_set_force_sysfs_fallback 0
152b3cf21faSLuis R. Rodriguez	proc_set_ignore_sysfs_fallback 0
153b3cf21faSLuis R. Rodriguez}
154b3cf21faSLuis R. Rodriguez
155f5a61451SLuis R. Rodrigueztest_finish()
156f5a61451SLuis R. Rodriguez{
157f5a61451SLuis R. Rodriguez	if [ "$HAS_FW_LOADER_USER_HELPER" = "yes" ]; then
158f5a61451SLuis R. Rodriguez		echo "$OLD_TIMEOUT" >/sys/class/firmware/timeout
159f5a61451SLuis R. Rodriguez	fi
160f5a61451SLuis R. Rodriguez	if [ "$TEST_REQS_FW_SET_CUSTOM_PATH" = "yes" ]; then
161e5384092SBen Hutchings		if [ "$OLD_FWPATH" = "" ]; then
162e5384092SBen Hutchings			# A zero-length write won't work; write a null byte
163e5384092SBen Hutchings			printf '\000' >/sys/module/firmware_class/parameters/path
164e5384092SBen Hutchings		else
165f5a61451SLuis R. Rodriguez			echo -n "$OLD_FWPATH" >/sys/module/firmware_class/parameters/path
166f5a61451SLuis R. Rodriguez		fi
167e5384092SBen Hutchings	fi
168f5a61451SLuis R. Rodriguez	if [ -f $FW ]; then
169f5a61451SLuis R. Rodriguez		rm -f "$FW"
170f5a61451SLuis R. Rodriguez	fi
171f5a61451SLuis R. Rodriguez	if [ -d $FWPATH ]; then
172f5a61451SLuis R. Rodriguez		rm -rf "$FWPATH"
173f5a61451SLuis R. Rodriguez	fi
174b3cf21faSLuis R. Rodriguez	proc_restore_defaults
175f5a61451SLuis R. Rodriguez}
176f5a61451SLuis R. Rodriguez
177ef557787SLuis R. Rodriguezkconfig_has()
178ef557787SLuis R. Rodriguez{
179ef557787SLuis R. Rodriguez	if [ -f $PROC_CONFIG ]; then
180ef557787SLuis R. Rodriguez		if zgrep -q $1 $PROC_CONFIG 2>/dev/null; then
181ef557787SLuis R. Rodriguez			echo "yes"
182ef557787SLuis R. Rodriguez		else
183ef557787SLuis R. Rodriguez			echo "no"
184ef557787SLuis R. Rodriguez		fi
185ef557787SLuis R. Rodriguez	else
186ef557787SLuis R. Rodriguez		# We currently don't have easy heuristics to infer this
187ef557787SLuis R. Rodriguez		# so best we can do is just try to use the kernel assuming
188ef557787SLuis R. Rodriguez		# you had enabled it. This matches the old behaviour.
189ef557787SLuis R. Rodriguez		if [ "$1" = "CONFIG_FW_LOADER_USER_HELPER_FALLBACK=y" ]; then
190ef557787SLuis R. Rodriguez			echo "yes"
191ef557787SLuis R. Rodriguez		elif [ "$1" = "CONFIG_FW_LOADER_USER_HELPER=y" ]; then
192ef557787SLuis R. Rodriguez			if [ -d /sys/class/firmware/ ]; then
193ef557787SLuis R. Rodriguez				echo yes
194ef557787SLuis R. Rodriguez			else
195ef557787SLuis R. Rodriguez				echo no
196ef557787SLuis R. Rodriguez			fi
197ef557787SLuis R. Rodriguez		fi
198ef557787SLuis R. Rodriguez	fi
199ef557787SLuis R. Rodriguez}
200