297 lines
6.9 KiB
Bash
Executable File
297 lines
6.9 KiB
Bash
Executable File
#!/bin/bash
|
|
# SPDX-License-Identifier: GPL-2.0
|
|
|
|
set -u
|
|
set -e
|
|
|
|
# This script currently only works for x86_64
|
|
ARCH="$(uname -m)"
|
|
case "${ARCH}" in
|
|
x86_64)
|
|
QEMU_BINARY=qemu-system-x86_64
|
|
BZIMAGE="arch/x86/boot/bzImage"
|
|
;;
|
|
*)
|
|
echo "Unsupported architecture"
|
|
exit 1
|
|
;;
|
|
esac
|
|
SCRIPT_DIR="$(dirname $(realpath $0))"
|
|
OUTPUT_DIR="$SCRIPT_DIR/results"
|
|
KCONFIG_REL_PATHS=("${SCRIPT_DIR}/config" "${SCRIPT_DIR}/config.common" "${SCRIPT_DIR}/config.${ARCH}")
|
|
B2C_URL="https://gitlab.freedesktop.org/gfx-ci/boot2container/-/raw/main/vm2c.py"
|
|
NUM_COMPILE_JOBS="$(nproc)"
|
|
LOG_FILE_BASE="$(date +"hid_selftests.%Y-%m-%d_%H-%M-%S")"
|
|
LOG_FILE="${LOG_FILE_BASE}.log"
|
|
EXIT_STATUS_FILE="${LOG_FILE_BASE}.exit_status"
|
|
CONTAINER_IMAGE="registry.freedesktop.org/bentiss/hid/fedora/39:2023-11-22.1"
|
|
|
|
TARGETS="${TARGETS:=$(basename ${SCRIPT_DIR})}"
|
|
DEFAULT_COMMAND="pip3 install hid-tools; make -C tools/testing/selftests TARGETS=${TARGETS} run_tests"
|
|
|
|
usage()
|
|
{
|
|
cat <<EOF
|
|
Usage: $0 [-j N] [-s] [-b] [-d <output_dir>] -- [<command>]
|
|
|
|
<command> is the command you would normally run when you are in
|
|
the source kernel direcory. e.g:
|
|
|
|
$0 -- ./tools/testing/selftests/hid/hid_bpf
|
|
|
|
If no command is specified and a debug shell (-s) is not requested,
|
|
"${DEFAULT_COMMAND}" will be run by default.
|
|
|
|
If you build your kernel using KBUILD_OUTPUT= or O= options, these
|
|
can be passed as environment variables to the script:
|
|
|
|
O=<kernel_build_path> $0 -- ./tools/testing/selftests/hid/hid_bpf
|
|
|
|
or
|
|
|
|
KBUILD_OUTPUT=<kernel_build_path> $0 -- ./tools/testing/selftests/hid/hid_bpf
|
|
|
|
Options:
|
|
|
|
-u) Update the boot2container script to a newer version.
|
|
-d) Update the output directory (default: ${OUTPUT_DIR})
|
|
-b) Run only the build steps for the kernel and the selftests
|
|
-j) Number of jobs for compilation, similar to -j in make
|
|
(default: ${NUM_COMPILE_JOBS})
|
|
-s) Instead of powering off the VM, start an interactive
|
|
shell. If <command> is specified, the shell runs after
|
|
the command finishes executing
|
|
EOF
|
|
}
|
|
|
|
download()
|
|
{
|
|
local file="$1"
|
|
|
|
echo "Downloading $file..." >&2
|
|
curl -Lsf "$file" -o "${@:2}"
|
|
}
|
|
|
|
recompile_kernel()
|
|
{
|
|
local kernel_checkout="$1"
|
|
local make_command="$2"
|
|
|
|
cd "${kernel_checkout}"
|
|
|
|
${make_command} olddefconfig
|
|
${make_command} headers
|
|
${make_command}
|
|
}
|
|
|
|
update_selftests()
|
|
{
|
|
local kernel_checkout="$1"
|
|
local selftests_dir="${kernel_checkout}/tools/testing/selftests/hid"
|
|
|
|
cd "${selftests_dir}"
|
|
${make_command}
|
|
}
|
|
|
|
run_vm()
|
|
{
|
|
local run_dir="$1"
|
|
local b2c="$2"
|
|
local kernel_bzimage="$3"
|
|
local command="$4"
|
|
local post_command=""
|
|
|
|
cd "${run_dir}"
|
|
|
|
if ! which "${QEMU_BINARY}" &> /dev/null; then
|
|
cat <<EOF
|
|
Could not find ${QEMU_BINARY}
|
|
Please install qemu or set the QEMU_BINARY environment variable.
|
|
EOF
|
|
exit 1
|
|
fi
|
|
|
|
# alpine (used in post-container requires the PATH to have /bin
|
|
export PATH=$PATH:/bin
|
|
|
|
if [[ "${debug_shell}" != "yes" ]]
|
|
then
|
|
touch ${OUTPUT_DIR}/${LOG_FILE}
|
|
command="mount bpffs -t bpf /sys/fs/bpf/; set -o pipefail ; ${command} 2>&1 | tee ${OUTPUT_DIR}/${LOG_FILE}"
|
|
post_command="cat ${OUTPUT_DIR}/${LOG_FILE}"
|
|
else
|
|
command="mount bpffs -t bpf /sys/fs/bpf/; ${command}"
|
|
fi
|
|
|
|
set +e
|
|
$b2c --command "${command}" \
|
|
--kernel ${kernel_bzimage} \
|
|
--workdir ${OUTPUT_DIR} \
|
|
--image ${CONTAINER_IMAGE}
|
|
|
|
echo $? > ${OUTPUT_DIR}/${EXIT_STATUS_FILE}
|
|
|
|
set -e
|
|
|
|
${post_command}
|
|
}
|
|
|
|
is_rel_path()
|
|
{
|
|
local path="$1"
|
|
|
|
[[ ${path:0:1} != "/" ]]
|
|
}
|
|
|
|
do_update_kconfig()
|
|
{
|
|
local kernel_checkout="$1"
|
|
local kconfig_file="$2"
|
|
|
|
rm -f "$kconfig_file" 2> /dev/null
|
|
|
|
for config in "${KCONFIG_REL_PATHS[@]}"; do
|
|
local kconfig_src="${config}"
|
|
cat "$kconfig_src" >> "$kconfig_file"
|
|
done
|
|
}
|
|
|
|
update_kconfig()
|
|
{
|
|
local kernel_checkout="$1"
|
|
local kconfig_file="$2"
|
|
|
|
if [[ -f "${kconfig_file}" ]]; then
|
|
local local_modified="$(stat -c %Y "${kconfig_file}")"
|
|
|
|
for config in "${KCONFIG_REL_PATHS[@]}"; do
|
|
local kconfig_src="${config}"
|
|
local src_modified="$(stat -c %Y "${kconfig_src}")"
|
|
# Only update the config if it has been updated after the
|
|
# previously cached config was created. This avoids
|
|
# unnecessarily compiling the kernel and selftests.
|
|
if [[ "${src_modified}" -gt "${local_modified}" ]]; then
|
|
do_update_kconfig "$kernel_checkout" "$kconfig_file"
|
|
# Once we have found one outdated configuration
|
|
# there is no need to check other ones.
|
|
break
|
|
fi
|
|
done
|
|
else
|
|
do_update_kconfig "$kernel_checkout" "$kconfig_file"
|
|
fi
|
|
}
|
|
|
|
main()
|
|
{
|
|
local script_dir="$(cd -P -- "$(dirname -- "${BASH_SOURCE[0]}")" && pwd -P)"
|
|
local kernel_checkout=$(realpath "${script_dir}"/../../../../)
|
|
# By default the script searches for the kernel in the checkout directory but
|
|
# it also obeys environment variables O= and KBUILD_OUTPUT=
|
|
local kernel_bzimage="${kernel_checkout}/${BZIMAGE}"
|
|
local command="${DEFAULT_COMMAND}"
|
|
local update_b2c="no"
|
|
local debug_shell="no"
|
|
local build_only="no"
|
|
|
|
while getopts ':hsud:j:b' opt; do
|
|
case ${opt} in
|
|
u)
|
|
update_b2c="yes"
|
|
;;
|
|
d)
|
|
OUTPUT_DIR="$OPTARG"
|
|
;;
|
|
j)
|
|
NUM_COMPILE_JOBS="$OPTARG"
|
|
;;
|
|
s)
|
|
command="/bin/sh"
|
|
debug_shell="yes"
|
|
;;
|
|
b)
|
|
build_only="yes"
|
|
;;
|
|
h)
|
|
usage
|
|
exit 0
|
|
;;
|
|
\? )
|
|
echo "Invalid Option: -$OPTARG"
|
|
usage
|
|
exit 1
|
|
;;
|
|
: )
|
|
echo "Invalid Option: -$OPTARG requires an argument"
|
|
usage
|
|
exit 1
|
|
;;
|
|
esac
|
|
done
|
|
shift $((OPTIND -1))
|
|
|
|
# trap 'catch "$?"' EXIT
|
|
if [[ "${build_only}" == "no" && "${debug_shell}" == "no" ]]; then
|
|
if [[ $# -eq 0 ]]; then
|
|
echo "No command specified, will run ${DEFAULT_COMMAND} in the vm"
|
|
else
|
|
command="$@"
|
|
|
|
if [[ "${command}" == "/bin/bash" || "${command}" == "bash" ]]
|
|
then
|
|
debug_shell="yes"
|
|
fi
|
|
fi
|
|
fi
|
|
|
|
local kconfig_file="${OUTPUT_DIR}/latest.config"
|
|
local make_command="make -j ${NUM_COMPILE_JOBS} KCONFIG_CONFIG=${kconfig_file}"
|
|
|
|
# Figure out where the kernel is being built.
|
|
# O takes precedence over KBUILD_OUTPUT.
|
|
if [[ "${O:=""}" != "" ]]; then
|
|
if is_rel_path "${O}"; then
|
|
O="$(realpath "${PWD}/${O}")"
|
|
fi
|
|
kernel_bzimage="${O}/${BZIMAGE}"
|
|
make_command="${make_command} O=${O}"
|
|
elif [[ "${KBUILD_OUTPUT:=""}" != "" ]]; then
|
|
if is_rel_path "${KBUILD_OUTPUT}"; then
|
|
KBUILD_OUTPUT="$(realpath "${PWD}/${KBUILD_OUTPUT}")"
|
|
fi
|
|
kernel_bzimage="${KBUILD_OUTPUT}/${BZIMAGE}"
|
|
make_command="${make_command} KBUILD_OUTPUT=${KBUILD_OUTPUT}"
|
|
fi
|
|
|
|
local b2c="${OUTPUT_DIR}/vm2c.py"
|
|
|
|
echo "Output directory: ${OUTPUT_DIR}"
|
|
|
|
mkdir -p "${OUTPUT_DIR}"
|
|
update_kconfig "${kernel_checkout}" "${kconfig_file}"
|
|
|
|
recompile_kernel "${kernel_checkout}" "${make_command}"
|
|
update_selftests "${kernel_checkout}" "${make_command}"
|
|
|
|
if [[ "${build_only}" == "no" ]]; then
|
|
if [[ "${update_b2c}" == "no" && ! -f "${b2c}" ]]; then
|
|
echo "vm2c script not found in ${b2c}"
|
|
update_b2c="yes"
|
|
fi
|
|
|
|
if [[ "${update_b2c}" == "yes" ]]; then
|
|
download $B2C_URL $b2c
|
|
chmod +x $b2c
|
|
fi
|
|
|
|
run_vm "${kernel_checkout}" $b2c "${kernel_bzimage}" "${command}"
|
|
if [[ "${debug_shell}" != "yes" ]]; then
|
|
echo "Logs saved in ${OUTPUT_DIR}/${LOG_FILE}"
|
|
fi
|
|
|
|
exit $(cat ${OUTPUT_DIR}/${EXIT_STATUS_FILE})
|
|
fi
|
|
}
|
|
|
|
main "$@"
|