Skip to content

Commit

Permalink
[vmtest] using danobi/vmtest to run selftest
Browse files Browse the repository at this point in the history
https://github.com/danobi/vmtest make it easy to run a VM with a given kernel
and using the host rootfs (RO) and a shared directory (RW).

This will help us with a couple of issue:
- we don't have to maintain a rootfs anymore
- when running the test, we do not need to perform a dance with libguestfs, which allows us to save time during test, and make it trivial to exchange files between the host and the guest.
- we won't have anymore discrepency between the libraries used in the host and guest (like explained in libbpf/ci#84 and also libbpf/ci#103)

This change migrates the test runs from using `prepare-rootfs` + `run-qemu` actions to using `run-vmtest` instead.
  • Loading branch information
chantra committed Jan 25, 2024
1 parent ba369dd commit aa82f29
Show file tree
Hide file tree
Showing 2 changed files with 194 additions and 11 deletions.
12 changes: 1 addition & 11 deletions .github/workflows/kernel-test.yml
Original file line number Diff line number Diff line change
Expand Up @@ -54,18 +54,8 @@ jobs:
- name: Untar artifacts
# zstd is installed by default in the runner images.
run: zstd -d -T0 vmlinux-${{ inputs.arch }}-${{ inputs.toolchain_full }}.tar.zst --stdout | tar -xf -
- name: Prepare rootfs
uses: libbpf/ci/prepare-rootfs@main
with:
project-name: 'libbpf'
arch: ${{ inputs.arch }}
kernel: ${{ inputs.kernel }}
kernel-root: '.'
kbuild-output: ${{ env.KBUILD_OUTPUT }}
image-output: '/tmp/root.img'
test: ${{ inputs.test }}
- name: Run selftests
uses: libbpf/ci/run-qemu@main
uses: libbpf/ci/run-vmtest@main
# https://github.com/actions/runner/issues/1483#issuecomment-1031671517
# booleans are weird in GH.
continue-on-error: ${{ fromJSON(env.CONTINUE_ON_ERROR) }}
Expand Down
193 changes: 193 additions & 0 deletions ci/vmtest/vmtest_selftests.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,193 @@
#!/bin/bash

# run_selftest.sh will run the tests within /${PROJECT_NAME}/selftests/bpf
# If no specific test names are given, all test will be ran, otherwise, it will
# run the test passed as parameters.
# There is 2 ways to pass test names.
# 1) command-line arguments to this script
# 2) a comma-separated list of test names passed as `run_tests` boot parameters.
# test names passed as any of those methods will be ran.

set -euo pipefail

source "$(cd "$(dirname "$0")" && pwd)/helpers.sh"

ARCH=$(uname -m)

STATUS_FILE=/mnt/vmtest/exitstatus
OUTPUT_DIR=/mnt/vmtest

BPF_SELFTESTS_DIR="/${PROJECT_NAME}/selftests/bpf"
VMTEST_CONFIGS_PATH="/${PROJECT_NAME}/vmtest/configs"

read_lists() {
(for path in "$@"; do
if [[ -s "$path" ]]; then
cat "$path"
fi;
done) | cut -d'#' -f1 | sed -e 's/^[[:space:]]*//' -e 's/[[:space:]]*$//' | tr -s '\n' ','
}

DENYLIST=$(read_lists \
"$BPF_SELFTESTS_DIR/DENYLIST" \
"$BPF_SELFTESTS_DIR/DENYLIST.${ARCH}" \
"$VMTEST_CONFIGS_PATH/DENYLIST" \
"$VMTEST_CONFIGS_PATH/DENYLIST.${ARCH}" \
)
ALLOWLIST=$(read_lists \
"$BPF_SELFTESTS_DIR/ALLOWLIST" \
"$BPF_SELFTESTS_DIR/ALLOWLIST.${ARCH}" \
"$VMTEST_CONFIGS_PATH/ALLOWLIST" \
"$VMTEST_CONFIGS_PATH/ALLOWLIST.${ARCH}" \
)

declare -a TEST_NAMES=()

read_test_names() {
foldable start read_test_names "Reading test names from boot parameters and command line arguments"
# Check if test names were passed as boot parameter.
# We expect `run_tests` to be a comma-separated list of test names.
IFS=',' read -r -a test_names_from_boot <<< \
"$(sed -n 's/.*run_tests=\([^ ]*\).*/\1/p' /proc/cmdline)"
echo "${#test_names_from_boot[@]} tests extracted from boot parameters: ${test_names_from_boot[*]}"
# Sort and only keep unique test names from both boot params and arguments
# TEST_NAMES will contain a sorted list of uniq tests to be ran.
# Only do this if any of $test_names_from_boot[@] or $@ has elements as
# "printf '%s\0'" will otherwise generate an empty element.
if [[ ${#test_names_from_boot[@]} -gt 0 || $# -gt 0 ]]
then
readarray -t TEST_NAMES < \
<(printf '%s\0' "${test_names_from_boot[@]}" "$@" | \
sort --zero-terminated --unique | \
xargs --null --max-args=1)
fi
foldable end read_test_names
}
test_progs_helper() {
local selftest="test_progs${1}"
local args="$2"
json_file=${selftest/-/_}
if [ "$2" == "-j" ]
then
json_file+="_parallel"
fi
json_file="${OUTPUT_DIR}/${json_file}.json"
foldable start ${selftest} "Testing ${selftest}"
# "&& true" does not change the return code (it is not executed
# if the Python script fails), but it prevents exiting on a
# failure due to the "set -e".
./${selftest} ${args} ${DENYLIST:+-d"$DENYLIST"} ${ALLOWLIST:+-a"$ALLOWLIST"} --json-summary "${json_file}" && true
echo "${selftest}:$?" >>"${STATUS_FILE}"
foldable end ${selftest}
}
test_progs() {
test_progs_helper "" ""
}
test_progs_parallel() {
test_progs_helper "" "-j"
}
test_progs_no_alu32() {
test_progs_helper "-no_alu32" ""
}
test_progs_no_alu32_parallel() {
test_progs_helper "-no_alu32" "-j"
}
test_progs_cpuv4() {
test_progs_helper "-cpuv4" ""
}
test_maps() {
foldable start test_maps "Testing test_maps"
taskset 0xF ./test_maps && true
echo "test_maps:$?" >>"${STATUS_FILE}"
foldable end test_maps
}
test_verifier() {
foldable start test_verifier "Testing test_verifier"
./test_verifier && true
echo "test_verifier:$?" >>"${STATUS_FILE}"
foldable end test_verifier
}
run_veristat_helper() {
local mode="${1}"
# Make veristat commands visible in the log
if [ -o xtrace ]; then
xtrace_was_on="1"
else
xtrace_was_on=""
set -x
fi
(
# shellcheck source=ci/vmtest/configs/run_veristat.default.cfg
# shellcheck source=ci/vmtest/configs/run_veristat.meta.cfg
source "${VMTEST_CONFIGS_PATH}/run_veristat.${mode}.cfg"
pushd "${VERISTAT_OBJECTS_DIR}"
"${BPF_SELFTESTS_DIR}/veristat" -o csv -q -e file,prog,verdict,states ${VERISTAT_OBJECTS_GLOB} > \
"${OUTPUT_DIR}/${VERISTAT_OUTPUT}"
echo "run_veristat_${mode}:$?" >> ${STATUS_FILE}
popd
)
# Hide commands again
if [ -z "$xtrace_was_on" ]; then
set +x
fi
}
run_veristat_kernel() {
foldable start run_veristat_kernel "Running veristat.kernel"
run_veristat_helper "kernel"
foldable end run_veristat_kernel
}
run_veristat_meta() {
foldable start run_veristat_meta "Running veristat.meta"
run_veristat_helper "meta"
foldable end run_veristat_meta
}
foldable end vm_init
foldable start kernel_config "Kconfig"
zcat /proc/config.gz
foldable end kernel_config
echo "DENYLIST: ${DENYLIST}"
echo "ALLOWLIST: ${ALLOWLIST}"
cd ${PROJECT_NAME}/selftests/bpf
# populate TEST_NAMES
read_test_names "$@"
# if we don't have any test name provided to the script, we run all tests.
if [ ${#TEST_NAMES[@]} -eq 0 ]; then
test_progs
test_progs_no_alu32
test_progs_cpuv4
test_maps
test_verifier
else
# else we run the tests passed as command-line arguments and through boot
# parameter.
for test_name in "${TEST_NAMES[@]}"; do
"${test_name}"
done
fi

0 comments on commit aa82f29

Please sign in to comment.