| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474 |
- #!/bin/bash
- # SPDX-License-Identifier: GPL-2.0
- #
- # Copyright (c) 2025 Red Hat
- # Copyright (c) 2025 Meta Platforms, Inc. and affiliates
- #
- # Dependencies:
- # * virtme-ng
- # * busybox-static (used by virtme-ng)
- # * qemu (used by virtme-ng)
- readonly SCRIPT_DIR="$(cd -P -- "$(dirname -- "${BASH_SOURCE[0]}")" && pwd -P)"
- readonly KERNEL_CHECKOUT=$(realpath "${SCRIPT_DIR}"/../../../../)
- source "${SCRIPT_DIR}"/../kselftest/ktap_helpers.sh
- readonly HID_BPF_TEST="${SCRIPT_DIR}"/hid_bpf
- readonly HIDRAW_TEST="${SCRIPT_DIR}"/hidraw
- readonly HID_BPF_PROGS="${KERNEL_CHECKOUT}/drivers/hid/bpf/progs"
- readonly SSH_GUEST_PORT=22
- readonly WAIT_PERIOD=3
- readonly WAIT_PERIOD_MAX=60
- readonly WAIT_TOTAL=$(( WAIT_PERIOD * WAIT_PERIOD_MAX ))
- readonly QEMU_PIDFILE=$(mktemp /tmp/qemu_hid_vmtest_XXXX.pid)
- readonly QEMU_OPTS="\
- --pidfile ${QEMU_PIDFILE} \
- "
- readonly KERNEL_CMDLINE=""
- readonly LOG=$(mktemp /tmp/hid_vmtest_XXXX.log)
- readonly TEST_NAMES=(vm_hid_bpf vm_hidraw vm_pytest)
- readonly TEST_DESCS=(
- "Run hid_bpf tests in the VM."
- "Run hidraw tests in the VM."
- "Run the hid-tools test-suite in the VM."
- )
- VERBOSE=0
- SHELL_MODE=0
- BUILD_HOST=""
- BUILD_HOST_PODMAN_CONTAINER_NAME=""
- usage() {
- local name
- local desc
- local i
- echo
- echo "$0 [OPTIONS] [TEST]... [-- tests-args]"
- echo "If no TEST argument is given, all tests will be run."
- echo
- echo "Options"
- echo " -b: build the kernel from the current source tree and use it for guest VMs"
- echo " -H: hostname for remote build host (used with -b)"
- echo " -p: podman container name for remote build host (used with -b)"
- echo " Example: -H beefyserver -p vng"
- echo " -q: set the path to or name of qemu binary"
- echo " -s: start a shell in the VM instead of running tests"
- echo " -v: more verbose output (can be repeated multiple times)"
- echo
- echo "Available tests"
- for ((i = 0; i < ${#TEST_NAMES[@]}; i++)); do
- name=${TEST_NAMES[${i}]}
- desc=${TEST_DESCS[${i}]}
- printf "\t%-35s%-35s\n" "${name}" "${desc}"
- done
- echo
- exit 1
- }
- die() {
- echo "$*" >&2
- exit "${KSFT_FAIL}"
- }
- vm_ssh() {
- # vng --ssh-client keeps shouting "Warning: Permanently added 'virtme-ng%22'
- # (ED25519) to the list of known hosts.",
- # So replace the command with what's actually called and add the "-q" option
- stdbuf -oL ssh -q \
- -F ${HOME}/.cache/virtme-ng/.ssh/virtme-ng-ssh.conf \
- -l root virtme-ng%${SSH_GUEST_PORT} \
- "$@"
- return $?
- }
- cleanup() {
- if [[ -s "${QEMU_PIDFILE}" ]]; then
- pkill -SIGTERM -F "${QEMU_PIDFILE}" > /dev/null 2>&1
- fi
- # If failure occurred during or before qemu start up, then we need
- # to clean this up ourselves.
- if [[ -e "${QEMU_PIDFILE}" ]]; then
- rm "${QEMU_PIDFILE}"
- fi
- }
- check_args() {
- local found
- for arg in "$@"; do
- found=0
- for name in "${TEST_NAMES[@]}"; do
- if [[ "${name}" = "${arg}" ]]; then
- found=1
- break
- fi
- done
- if [[ "${found}" -eq 0 ]]; then
- echo "${arg} is not an available test" >&2
- usage
- fi
- done
- for arg in "$@"; do
- if ! command -v > /dev/null "test_${arg}"; then
- echo "Test ${arg} not found" >&2
- usage
- fi
- done
- }
- check_deps() {
- for dep in vng ${QEMU} busybox pkill ssh pytest; do
- if [[ ! -x $(command -v "${dep}") ]]; then
- echo -e "skip: dependency ${dep} not found!\n"
- exit "${KSFT_SKIP}"
- fi
- done
- if [[ ! -x $(command -v "${HID_BPF_TEST}") ]]; then
- printf "skip: %s not found!" "${HID_BPF_TEST}"
- printf " Please build the kselftest hid_bpf target.\n"
- exit "${KSFT_SKIP}"
- fi
- if [[ ! -x $(command -v "${HIDRAW_TEST}") ]]; then
- printf "skip: %s not found!" "${HIDRAW_TEST}"
- printf " Please build the kselftest hidraw target.\n"
- exit "${KSFT_SKIP}"
- fi
- }
- check_vng() {
- local tested_versions
- local version
- local ok
- tested_versions=("1.36" "1.37")
- version="$(vng --version)"
- ok=0
- for tv in "${tested_versions[@]}"; do
- if [[ "${version}" == *"${tv}"* ]]; then
- ok=1
- break
- fi
- done
- if [[ ! "${ok}" -eq 1 ]]; then
- printf "warning: vng version '%s' has not been tested and may " "${version}" >&2
- printf "not function properly.\n\tThe following versions have been tested: " >&2
- echo "${tested_versions[@]}" >&2
- fi
- }
- handle_build() {
- if [[ ! "${BUILD}" -eq 1 ]]; then
- return
- fi
- if [[ ! -d "${KERNEL_CHECKOUT}" ]]; then
- echo "-b requires vmtest.sh called from the kernel source tree" >&2
- exit 1
- fi
- pushd "${KERNEL_CHECKOUT}" &>/dev/null
- if ! vng --kconfig --config "${SCRIPT_DIR}"/config; then
- die "failed to generate .config for kernel source tree (${KERNEL_CHECKOUT})"
- fi
- local vng_args=("-v" "--config" "${SCRIPT_DIR}/config" "--build")
- if [[ -n "${BUILD_HOST}" ]]; then
- vng_args+=("--build-host" "${BUILD_HOST}")
- fi
- if [[ -n "${BUILD_HOST_PODMAN_CONTAINER_NAME}" ]]; then
- vng_args+=("--build-host-exec-prefix" \
- "podman exec -ti ${BUILD_HOST_PODMAN_CONTAINER_NAME}")
- fi
- if ! vng "${vng_args[@]}"; then
- die "failed to build kernel from source tree (${KERNEL_CHECKOUT})"
- fi
- if ! make -j$(nproc) -C "${HID_BPF_PROGS}"; then
- die "failed to build HID bpf objects from source tree (${HID_BPF_PROGS})"
- fi
- if ! make -j$(nproc) -C "${SCRIPT_DIR}"; then
- die "failed to build HID selftests from source tree (${SCRIPT_DIR})"
- fi
- popd &>/dev/null
- }
- vm_start() {
- local logfile=/dev/null
- local verbose_opt=""
- local kernel_opt=""
- local qemu
- qemu=$(command -v "${QEMU}")
- if [[ "${VERBOSE}" -eq 2 ]]; then
- verbose_opt="--verbose"
- logfile=/dev/stdout
- fi
- # If we are running from within the kernel source tree, use the kernel source tree
- # as the kernel to boot, otherwise use the currently running kernel.
- if [[ "$(realpath "$(pwd)")" == "${KERNEL_CHECKOUT}"* ]]; then
- kernel_opt="${KERNEL_CHECKOUT}"
- fi
- vng \
- --run \
- ${kernel_opt} \
- ${verbose_opt} \
- --qemu-opts="${QEMU_OPTS}" \
- --qemu="${qemu}" \
- --user root \
- --append "${KERNEL_CMDLINE}" \
- --ssh "${SSH_GUEST_PORT}" \
- --rw &> ${logfile} &
- local vng_pid=$!
- local elapsed=0
- while [[ ! -s "${QEMU_PIDFILE}" ]]; do
- if ! kill -0 "${vng_pid}" 2>/dev/null; then
- echo "vng process (PID ${vng_pid}) exited early, check logs for details" >&2
- die "failed to boot VM"
- fi
- if [[ ${elapsed} -ge ${WAIT_TOTAL} ]]; then
- echo "Timed out after ${WAIT_TOTAL} seconds waiting for VM to boot" >&2
- die "failed to boot VM"
- fi
- sleep 1
- elapsed=$((elapsed + 1))
- done
- }
- vm_wait_for_ssh() {
- local i
- i=0
- while true; do
- if [[ ${i} -gt ${WAIT_PERIOD_MAX} ]]; then
- die "Timed out waiting for guest ssh"
- fi
- if vm_ssh -- true; then
- break
- fi
- i=$(( i + 1 ))
- sleep ${WAIT_PERIOD}
- done
- }
- vm_mount_bpffs() {
- vm_ssh -- mount bpffs -t bpf /sys/fs/bpf
- }
- __log_stdin() {
- stdbuf -oL awk '{ printf "%s:\t%s\n","'"${prefix}"'", $0; fflush() }'
- }
- __log_args() {
- echo "$*" | awk '{ printf "%s:\t%s\n","'"${prefix}"'", $0 }'
- }
- log() {
- local verbose="$1"
- shift
- local prefix="$1"
- shift
- local redirect=
- if [[ ${verbose} -le 0 ]]; then
- redirect=/dev/null
- else
- redirect=/dev/stdout
- fi
- if [[ "$#" -eq 0 ]]; then
- __log_stdin | tee -a "${LOG}" > ${redirect}
- else
- __log_args "$@" | tee -a "${LOG}" > ${redirect}
- fi
- }
- log_setup() {
- log $((VERBOSE-1)) "setup" "$@"
- }
- log_host() {
- local testname=$1
- shift
- log $((VERBOSE-1)) "test:${testname}:host" "$@"
- }
- log_guest() {
- local testname=$1
- shift
- log ${VERBOSE} "# test:${testname}" "$@"
- }
- test_vm_hid_bpf() {
- local testname="${FUNCNAME[0]#test_}"
- vm_ssh -- "${HID_BPF_TEST}" \
- 2>&1 | log_guest "${testname}"
- return ${PIPESTATUS[0]}
- }
- test_vm_hidraw() {
- local testname="${FUNCNAME[0]#test_}"
- vm_ssh -- "${HIDRAW_TEST}" \
- 2>&1 | log_guest "${testname}"
- return ${PIPESTATUS[0]}
- }
- test_vm_pytest() {
- local testname="${FUNCNAME[0]#test_}"
- shift
- vm_ssh -- pytest ${SCRIPT_DIR}/tests --color=yes "$@" \
- 2>&1 | log_guest "${testname}"
- return ${PIPESTATUS[0]}
- }
- run_test() {
- local vm_oops_cnt_before
- local vm_warn_cnt_before
- local vm_oops_cnt_after
- local vm_warn_cnt_after
- local name
- local rc
- vm_oops_cnt_before=$(vm_ssh -- dmesg | grep -c -i 'Oops')
- vm_error_cnt_before=$(vm_ssh -- dmesg --level=err | wc -l)
- name=$(echo "${1}" | awk '{ print $1 }')
- eval test_"${name}" "$@"
- rc=$?
- vm_oops_cnt_after=$(vm_ssh -- dmesg | grep -i 'Oops' | wc -l)
- if [[ ${vm_oops_cnt_after} -gt ${vm_oops_cnt_before} ]]; then
- echo "FAIL: kernel oops detected on vm" | log_host "${name}"
- rc=$KSFT_FAIL
- fi
- vm_error_cnt_after=$(vm_ssh -- dmesg --level=err | wc -l)
- if [[ ${vm_error_cnt_after} -gt ${vm_error_cnt_before} ]]; then
- echo "FAIL: kernel error detected on vm" | log_host "${name}"
- vm_ssh -- dmesg --level=err | log_host "${name}"
- rc=$KSFT_FAIL
- fi
- return "${rc}"
- }
- QEMU="qemu-system-$(uname -m)"
- while getopts :hvsbq:H:p: o
- do
- case $o in
- v) VERBOSE=$((VERBOSE+1));;
- s) SHELL_MODE=1;;
- b) BUILD=1;;
- q) QEMU=$OPTARG;;
- H) BUILD_HOST=$OPTARG;;
- p) BUILD_HOST_PODMAN_CONTAINER_NAME=$OPTARG;;
- h|*) usage;;
- esac
- done
- shift $((OPTIND-1))
- trap cleanup EXIT
- PARAMS=""
- if [[ ${#} -eq 0 ]]; then
- ARGS=("${TEST_NAMES[@]}")
- else
- ARGS=()
- COUNT=0
- for arg in $@; do
- COUNT=$((COUNT+1))
- if [[ x"$arg" == x"--" ]]; then
- break
- fi
- ARGS+=($arg)
- done
- shift $COUNT
- PARAMS="$@"
- fi
- if [[ "${SHELL_MODE}" -eq 0 ]]; then
- check_args "${ARGS[@]}"
- echo "1..${#ARGS[@]}"
- fi
- check_deps
- check_vng
- handle_build
- log_setup "Booting up VM"
- vm_start
- vm_wait_for_ssh
- vm_mount_bpffs
- log_setup "VM booted up"
- if [[ "${SHELL_MODE}" -eq 1 ]]; then
- log_setup "Starting interactive shell in VM"
- echo "Starting shell in VM. Use 'exit' to quit and shutdown the VM."
- CURRENT_DIR="$(pwd)"
- vm_ssh -t -- "cd '${CURRENT_DIR}' && exec bash -l"
- exit "$KSFT_PASS"
- fi
- cnt_pass=0
- cnt_fail=0
- cnt_skip=0
- cnt_total=0
- for arg in "${ARGS[@]}"; do
- run_test "${arg}" "${PARAMS}"
- rc=$?
- if [[ ${rc} -eq $KSFT_PASS ]]; then
- cnt_pass=$(( cnt_pass + 1 ))
- echo "ok ${cnt_total} ${arg}"
- elif [[ ${rc} -eq $KSFT_SKIP ]]; then
- cnt_skip=$(( cnt_skip + 1 ))
- echo "ok ${cnt_total} ${arg} # SKIP"
- elif [[ ${rc} -eq $KSFT_FAIL ]]; then
- cnt_fail=$(( cnt_fail + 1 ))
- echo "not ok ${cnt_total} ${arg} # exit=$rc"
- fi
- cnt_total=$(( cnt_total + 1 ))
- done
- echo "SUMMARY: PASS=${cnt_pass} SKIP=${cnt_skip} FAIL=${cnt_fail}"
- echo "Log: ${LOG}"
- if [ $((cnt_pass + cnt_skip)) -eq ${cnt_total} ]; then
- exit "$KSFT_PASS"
- else
- exit "$KSFT_FAIL"
- fi
|