fw_lib.sh 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236
  1. #!/bin/bash
  2. # SPDX-License-Identifier: GPL-2.0
  3. # Library of helpers for test scripts.
  4. set -e
  5. DIR=/sys/devices/virtual/misc/test_firmware
  6. PROC_CONFIG="/proc/config.gz"
  7. TEST_DIR=$(dirname $0)
  8. # We need to load a different file to test request_firmware_into_buf
  9. # I believe the issue is firmware loaded cached vs. non-cached
  10. # with same filename is bungled.
  11. # To reproduce rename this to test-firmware.bin
  12. TEST_FIRMWARE_INTO_BUF_FILENAME=test-firmware-into-buf.bin
  13. # Kselftest framework requirement - SKIP code is 4.
  14. ksft_skip=4
  15. print_reqs_exit()
  16. {
  17. echo "You must have the following enabled in your kernel:" >&2
  18. cat $TEST_DIR/config >&2
  19. exit $ksft_skip
  20. }
  21. test_modprobe()
  22. {
  23. if [ ! -d $DIR ]; then
  24. print_reqs_exit
  25. fi
  26. }
  27. check_mods()
  28. {
  29. local uid=$(id -u)
  30. if [ $uid -ne 0 ]; then
  31. echo "skip all tests: must be run as root" >&2
  32. exit $ksft_skip
  33. fi
  34. trap "test_modprobe" EXIT
  35. if [ ! -d $DIR ]; then
  36. modprobe test_firmware
  37. fi
  38. if [ ! -f $PROC_CONFIG ]; then
  39. if modprobe configs 2>/dev/null; then
  40. echo "Loaded configs module"
  41. if [ ! -f $PROC_CONFIG ]; then
  42. echo "You must have the following enabled in your kernel:" >&2
  43. cat $TEST_DIR/config >&2
  44. echo "Resorting to old heuristics" >&2
  45. fi
  46. else
  47. echo "Failed to load configs module, using old heuristics" >&2
  48. fi
  49. fi
  50. }
  51. check_setup()
  52. {
  53. HAS_FW_LOADER_USER_HELPER="$(kconfig_has CONFIG_FW_LOADER_USER_HELPER=y)"
  54. HAS_FW_LOADER_USER_HELPER_FALLBACK="$(kconfig_has CONFIG_FW_LOADER_USER_HELPER_FALLBACK=y)"
  55. HAS_FW_LOADER_COMPRESS_XZ="$(kconfig_has CONFIG_FW_LOADER_COMPRESS_XZ=y)"
  56. HAS_FW_LOADER_COMPRESS_ZSTD="$(kconfig_has CONFIG_FW_LOADER_COMPRESS_ZSTD=y)"
  57. HAS_FW_UPLOAD="$(kconfig_has CONFIG_FW_UPLOAD=y)"
  58. PROC_FW_IGNORE_SYSFS_FALLBACK="0"
  59. PROC_FW_FORCE_SYSFS_FALLBACK="0"
  60. if [ -z $PROC_SYS_DIR ]; then
  61. PROC_SYS_DIR="/proc/sys/kernel"
  62. fi
  63. FW_PROC="${PROC_SYS_DIR}/firmware_config"
  64. FW_FORCE_SYSFS_FALLBACK="$FW_PROC/force_sysfs_fallback"
  65. FW_IGNORE_SYSFS_FALLBACK="$FW_PROC/ignore_sysfs_fallback"
  66. if [ -f $FW_FORCE_SYSFS_FALLBACK ]; then
  67. PROC_FW_FORCE_SYSFS_FALLBACK="$(cat $FW_FORCE_SYSFS_FALLBACK)"
  68. fi
  69. if [ -f $FW_IGNORE_SYSFS_FALLBACK ]; then
  70. PROC_FW_IGNORE_SYSFS_FALLBACK="$(cat $FW_IGNORE_SYSFS_FALLBACK)"
  71. fi
  72. if [ "$PROC_FW_FORCE_SYSFS_FALLBACK" = "1" ]; then
  73. HAS_FW_LOADER_USER_HELPER="yes"
  74. HAS_FW_LOADER_USER_HELPER_FALLBACK="yes"
  75. fi
  76. if [ "$PROC_FW_IGNORE_SYSFS_FALLBACK" = "1" ]; then
  77. HAS_FW_LOADER_USER_HELPER_FALLBACK="no"
  78. HAS_FW_LOADER_USER_HELPER="no"
  79. fi
  80. if [ "$HAS_FW_LOADER_USER_HELPER" = "yes" ]; then
  81. OLD_TIMEOUT="$(cat /sys/class/firmware/timeout)"
  82. fi
  83. OLD_FWPATH="$(cat /sys/module/firmware_class/parameters/path)"
  84. if [ "$HAS_FW_LOADER_COMPRESS_XZ" = "yes" ]; then
  85. if ! which xz 2> /dev/null > /dev/null; then
  86. HAS_FW_LOADER_COMPRESS_XZ=""
  87. fi
  88. fi
  89. if [ "$HAS_FW_LOADER_COMPRESS_ZSTD" = "yes" ]; then
  90. if ! which zstd 2> /dev/null > /dev/null; then
  91. HAS_FW_LOADER_COMPRESS_ZSTD=""
  92. fi
  93. fi
  94. }
  95. verify_reqs()
  96. {
  97. if [ "$TEST_REQS_FW_SYSFS_FALLBACK" = "yes" ]; then
  98. if [ ! "$HAS_FW_LOADER_USER_HELPER" = "yes" ]; then
  99. echo "usermode helper disabled so ignoring test"
  100. exit 0
  101. fi
  102. fi
  103. if [ "$TEST_REQS_FW_UPLOAD" = "yes" ]; then
  104. if [ ! "$HAS_FW_UPLOAD" = "yes" ]; then
  105. echo "firmware upload disabled so ignoring test"
  106. exit 0
  107. fi
  108. fi
  109. }
  110. setup_tmp_file()
  111. {
  112. FWPATH=$(mktemp -d)
  113. FW="$FWPATH/test-firmware.bin"
  114. echo "ABCD0123" >"$FW"
  115. FW_INTO_BUF="$FWPATH/$TEST_FIRMWARE_INTO_BUF_FILENAME"
  116. echo "EFGH4567" >"$FW_INTO_BUF"
  117. NAME=$(basename "$FW")
  118. if [ "$TEST_REQS_FW_SET_CUSTOM_PATH" = "yes" ]; then
  119. echo -n "$FWPATH" >/sys/module/firmware_class/parameters/path
  120. fi
  121. }
  122. __setup_random_file()
  123. {
  124. RANDOM_FILE_PATH="$(mktemp -p $FWPATH)"
  125. # mktemp says dry-run -n is unsafe, so...
  126. if [[ "$1" = "fake" ]]; then
  127. rm -rf $RANDOM_FILE_PATH
  128. sync
  129. else
  130. echo "ABCD0123" >"$RANDOM_FILE_PATH"
  131. fi
  132. echo $RANDOM_FILE_PATH
  133. }
  134. setup_random_file()
  135. {
  136. echo $(__setup_random_file)
  137. }
  138. setup_random_file_fake()
  139. {
  140. echo $(__setup_random_file fake)
  141. }
  142. proc_set_force_sysfs_fallback()
  143. {
  144. if [ -f $FW_FORCE_SYSFS_FALLBACK ]; then
  145. echo -n $1 > $FW_FORCE_SYSFS_FALLBACK
  146. check_setup
  147. fi
  148. }
  149. proc_set_ignore_sysfs_fallback()
  150. {
  151. if [ -f $FW_IGNORE_SYSFS_FALLBACK ]; then
  152. echo -n $1 > $FW_IGNORE_SYSFS_FALLBACK
  153. check_setup
  154. fi
  155. }
  156. proc_restore_defaults()
  157. {
  158. proc_set_force_sysfs_fallback 0
  159. proc_set_ignore_sysfs_fallback 0
  160. }
  161. test_finish()
  162. {
  163. if [ "$HAS_FW_LOADER_USER_HELPER" = "yes" ]; then
  164. echo "$OLD_TIMEOUT" >/sys/class/firmware/timeout
  165. fi
  166. if [ "$TEST_REQS_FW_SET_CUSTOM_PATH" = "yes" ]; then
  167. if [ "$OLD_FWPATH" = "" ]; then
  168. # A zero-length write won't work; write a null byte
  169. printf '\000' >/sys/module/firmware_class/parameters/path
  170. else
  171. echo -n "$OLD_FWPATH" >/sys/module/firmware_class/parameters/path
  172. fi
  173. fi
  174. if [ -f $FW ]; then
  175. rm -f "$FW"
  176. fi
  177. if [ -f $FW_INTO_BUF ]; then
  178. rm -f "$FW_INTO_BUF"
  179. fi
  180. if [ -d $FWPATH ]; then
  181. rm -rf "$FWPATH"
  182. fi
  183. proc_restore_defaults
  184. }
  185. kconfig_has()
  186. {
  187. if [ -f $PROC_CONFIG ]; then
  188. if zgrep -q $1 $PROC_CONFIG 2>/dev/null; then
  189. echo "yes"
  190. else
  191. echo "no"
  192. fi
  193. else
  194. # We currently don't have easy heuristics to infer this
  195. # so best we can do is just try to use the kernel assuming
  196. # you had enabled it. This matches the old behaviour.
  197. if [ "$1" = "CONFIG_FW_LOADER_USER_HELPER_FALLBACK=y" ]; then
  198. echo "yes"
  199. elif [ "$1" = "CONFIG_FW_LOADER_USER_HELPER=y" ]; then
  200. if [ -d /sys/class/firmware/ ]; then
  201. echo yes
  202. else
  203. echo no
  204. fi
  205. fi
  206. fi
  207. }