xref: /freebsd/tools/boot/ci-qemu-test.sh (revision 0a36787e4c1fa0cf77dcf83be0867178476e372b)
1#!/bin/sh
2
3# Install pkgbase packages for loader, kernel, and enough of userland to boot
4# in QEMU and echo "Hello world." from init, as a very quick smoke test for CI.
5# Uses QEMU's virtual FAT filesystem to avoid the need to create a disk image.
6# While designed for CI automated testing, this script can also be run by hand
7# as a quick smoke-test as long as pkgbase packages have been built.  The
8# rootgen.sh and related scripts generate much more extensive tests for many
9# combinations of boot env (ufs, zfs, geli, etc).
10#
11# $FreeBSD$
12
13set -e
14
15die()
16{
17	echo "$*" 1>&2
18	exit 1
19}
20
21tempdir_cleanup()
22{
23	trap - EXIT SIGINT SIGHUP SIGTERM SIGQUIT
24	rm -rf ${ROOTDIR}
25}
26
27tempdir_setup()
28{
29	# Create minimal directory structure and populate it.
30
31	for dir in dev bin efi/boot etc lib libexec sbin usr/lib usr/libexec; do
32		mkdir -p ${ROOTDIR}/${dir}
33	done
34
35	# Install kernel, loader and minimal userland.
36	cat<<EOF >${ROOTDIR}/pkg.conf
37REPOS_DIR=[]
38repositories={local {url = file://$(dirname $OBJTOP)/repo/\${ABI}/latest}}
39EOF
40	ASSUME_ALWAYS_YES=true INSTALL_AS_USER=true pkg \
41	    -o ABI_FILE=$OBJTOP/bin/sh/sh \
42	    -C ${ROOTDIR}/pkg.conf -r ${ROOTDIR} install \
43	    FreeBSD-kernel-generic FreeBSD-bootloader \
44	    FreeBSD-clibs FreeBSD-runtime
45
46	# Put loader in standard EFI location.
47	mv ${ROOTDIR}/boot/loader.efi ${ROOTDIR}/efi/boot/$EFIBOOT
48
49	# Configuration files.
50	cat > ${ROOTDIR}/boot/loader.conf <<EOF
51vfs.root.mountfrom="msdosfs:/dev/$ROOTDEV"
52autoboot_delay=-1
53boot_verbose=YES
54EOF
55	cat > ${ROOTDIR}/etc/rc <<EOF
56#!/bin/sh
57
58echo "Hello world."
59/sbin/sysctl vm.stats.vm.v_wire_count
60/sbin/shutdown -p now
61EOF
62
63	# Entropy needed to boot, see r346250 and followup commits/discussion.
64	dd if=/dev/random of=${ROOTDIR}/boot/entropy bs=4k count=1
65
66	# Remove unnecessary files to keep FAT filesystem size down.
67	rm -rf ${ROOTDIR}/METALOG ${ROOTDIR}/usr/lib
68}
69
70# Locate the top of the source tree, to run make install from.
71: ${SRCTOP:=$(make -V SRCTOP)}
72if [ -z "${SRCTOP}" ]; then
73	die "Cannot locate top of source tree"
74fi
75: ${OBJTOP:=$(make -V OBJTOP)}
76if [ -z "${OBJTOP}" ]; then
77	die "Cannot locate top of object tree"
78fi
79
80: ${TARGET:=$(uname -m)}
81case $TARGET in
82amd64)
83	# Locate the uefi firmware file used by qemu.
84	: ${OVMF:=/usr/local/share/qemu/edk2-x86_64-code.fd}
85	if [ ! -r "${OVMF}" ]; then
86		die "Cannot read UEFI firmware file ${OVMF}"
87	fi
88	QEMU="qemu-system-x86_64 -drive if=pflash,format=raw,readonly,file=${OVMF}"
89	EFIBOOT=BOOTx64.EFI
90	ROOTDEV=ada0s1
91	;;
92arm64)
93	QEMU="qemu-system-aarch64 -cpu cortex-a57 -M virt -bios edk2-aarch64-code.fd"
94	EFIBOOT=BOOTAA64.EFI
95	ROOTDEV=vtbd0s1
96	;;
97*)
98	die "Unknown TARGET:TARGET_ARCH $TARGET:$TARGET_ARCH"
99esac
100
101# Create a temp dir to hold the boot image.
102ROOTDIR=$(mktemp -d -t ci-qemu-test-fat-root)
103trap tempdir_cleanup EXIT SIGINT SIGHUP SIGTERM SIGQUIT
104
105# Populate the boot image in a temp dir.
106( cd ${SRCTOP} && tempdir_setup )
107
108# And, boot in QEMU.
109: ${BOOTLOG:=${TMPDIR:-/tmp}/ci-qemu-test-boot.log}
110timeout 300 \
111    $QEMU -m 256M -nodefaults \
112        -serial stdio -vga none -nographic -monitor none \
113        -snapshot -hda fat:${ROOTDIR} 2>&1 | tee ${BOOTLOG}
114
115# Check whether we succesfully booted...
116if grep -q 'Hello world.' ${BOOTLOG}; then
117	echo "OK"
118else
119	die "Did not boot successfully, see ${BOOTLOG}"
120fi
121