xref: /kvm-unit-tests/scripts/arch-run.bash (revision 34f728696e06d0830e793bb38b0b54ce370b62b0)
1##############################################################################
2# run_qemu translates the ambiguous exit status in Table1 to that in Table2.
3# Table3 simply documents the complete status table.
4#
5# Table1: Before fixup
6# --------------------
7# 0      - Unexpected exit from QEMU (possible signal), or the unittest did
8#          not use debug-exit
9# 1      - most likely unittest succeeded, or QEMU failed
10#
11# Table2: After fixup
12# -------------------
13# 0      - Everything succeeded
14# 1      - most likely QEMU failed
15#
16# Table3: Complete table
17# ----------------------
18# 0      - SUCCESS
19# 1      - most likely QEMU failed
20# 2      - most likely a run script failed
21# 3      - most likely the unittest failed
22# 124    - most likely the unittest timed out
23# 127    - most likely the unittest called abort()
24# 1..127 - FAILURE (could be QEMU, a run script, or the unittest)
25# >= 128 - Signal (signum = status - 128)
26##############################################################################
27run_qemu ()
28{
29	local stdout errors ret sig
30
31	initrd_create || return $?
32	echo -n "$@"
33	[ "$ENVIRON_DEFAULT" = "yes" ] && echo -n " #"
34	echo " $INITRD"
35
36	# stdout to {stdout}, stderr to $errors and stderr
37	exec {stdout}>&1
38	errors=$("${@}" $INITRD </dev/null 2> >(tee /dev/stderr) > /dev/fd/$stdout)
39	ret=$?
40	exec {stdout}>&-
41
42	[ $ret -eq 134 ] && echo "QEMU Aborted" >&2
43
44	if [ "$errors" ]; then
45		sig=$(grep 'terminating on signal' <<<"$errors")
46		if [ "$sig" ]; then
47			sig=$(sed 's/.*terminating on signal \([0-9][0-9]*\).*/\1/' <<<"$sig")
48		fi
49	fi
50
51	if [ $ret -eq 0 ]; then
52		# Some signals result in a zero return status, but the
53		# error log tells the truth.
54		if [ "$sig" ]; then
55			((ret=sig+128))
56		else
57			# Exiting with zero (non-debugexit) is an error
58			ret=1
59		fi
60	elif [ $ret -eq 1 ]; then
61		# Even when ret==1 (unittest success) if we also got stderr
62		# logs, then we assume a QEMU failure. Otherwise we translate
63		# status of 1 to 0 (SUCCESS)
64		if [ -z "$(echo "$errors" | grep -vi warning)" ]; then
65			ret=0
66		fi
67	fi
68
69	return $ret
70}
71
72run_qemu_status ()
73{
74	local stdout ret
75
76	exec {stdout}>&1
77	lines=$(run_qemu "$@" > >(tee /dev/fd/$stdout))
78	ret=$?
79	exec {stdout}>&-
80
81	if [ $ret -eq 1 ]; then
82		testret=$(grep '^EXIT: ' <<<"$lines" | sed 's/.*STATUS=\([0-9][0-9]*\).*/\1/')
83		if [ "$testret" ]; then
84			if [ $testret -eq 1 ]; then
85				ret=0
86			else
87				ret=$testret
88			fi
89		fi
90	fi
91
92	return $ret
93}
94
95timeout_cmd ()
96{
97	local s
98
99	if [ "$TIMEOUT" ] && [ "$TIMEOUT" != "0" ]; then
100		if [ "$CONFIG_EFI" = 'y' ]; then
101			s=${TIMEOUT: -1}
102			if [ "$s" = 's' ]; then
103				TIMEOUT=${TIMEOUT:0:-1}
104				((TIMEOUT += 10)) # Add 10 seconds for booting UEFI
105				TIMEOUT="${TIMEOUT}s"
106			fi
107		fi
108		echo "timeout -k 1s --foreground $TIMEOUT"
109	fi
110}
111
112qmp ()
113{
114	echo '{ "execute": "qmp_capabilities" }{ "execute":' "$2" '}' | ncat -U $1
115}
116
117qmp_events ()
118{
119	while ! test -S "$1"; do sleep 0.1; done
120	echo '{ "execute": "qmp_capabilities" }{ "execute": "cont" }' |
121		ncat --no-shutdown -U $1 |
122		jq -c 'select(has("event"))'
123}
124
125run_migration ()
126{
127	if ! command -v ncat >/dev/null 2>&1; then
128		echo "${FUNCNAME[0]} needs ncat (netcat)" >&2
129		return 77
130	fi
131
132	migcmdline=$@
133
134	trap 'trap - TERM ; kill 0 ; exit 2' INT TERM
135	trap 'rm -f ${src_out} ${dst_out} ${src_outfifo} ${dst_outfifo} ${dst_incoming} ${src_qmp} ${dst_qmp} ${dst_infifo}' RETURN EXIT
136
137	dst_incoming=$(mktemp -u -t mig-helper-socket-incoming.XXXXXXXXXX)
138	src_out=$(mktemp -t mig-helper-stdout1.XXXXXXXXXX)
139	src_outfifo=$(mktemp -u -t mig-helper-fifo-stdout1.XXXXXXXXXX)
140	dst_out=$(mktemp -t mig-helper-stdout2.XXXXXXXXXX)
141	dst_outfifo=$(mktemp -u -t mig-helper-fifo-stdout2.XXXXXXXXXX)
142	src_qmp=$(mktemp -u -t mig-helper-qmp1.XXXXXXXXXX)
143	dst_qmp=$(mktemp -u -t mig-helper-qmp2.XXXXXXXXXX)
144	dst_infifo=$(mktemp -u -t mig-helper-fifo-stdin.XXXXXXXXXX)
145	src_qmpout=/dev/null
146	dst_qmpout=/dev/null
147
148	mkfifo ${src_outfifo}
149	mkfifo ${dst_outfifo}
150
151	eval "$migcmdline" \
152		-chardev socket,id=mon,path=${src_qmp},server=on,wait=off \
153		-mon chardev=mon,mode=control > ${src_outfifo} &
154	live_pid=$!
155	cat ${src_outfifo} | tee ${src_out} | grep -v "Now migrate the VM (quiet)" &
156
157	# Start the first destination QEMU machine in advance of the test
158	# reaching the migration point, since we expect at least one migration.
159	# Then destination machines are started after the test outputs
160	# subsequent "Now migrate the VM" messages.
161	do_migration || return $?
162
163	while ps -p ${live_pid} > /dev/null ; do
164		# Wait for test exit or further migration messages.
165		if ! grep -q -i "Now migrate the VM" < ${src_out} ; then
166			sleep 0.1
167		else
168			do_migration || return $?
169		fi
170	done
171
172	wait ${live_pid}
173	ret=$?
174
175	while (( $(jobs -r | wc -l) > 0 )); do
176		sleep 0.1
177	done
178
179	return $ret
180}
181
182do_migration ()
183{
184	# We have to use cat to open the named FIFO, because named FIFO's,
185	# unlike pipes, will block on open() until the other end is also
186	# opened, and that totally breaks QEMU...
187	mkfifo ${dst_infifo}
188	eval "$migcmdline" \
189		-chardev socket,id=mon,path=${dst_qmp},server=on,wait=off \
190		-mon chardev=mon,mode=control -incoming unix:${dst_incoming} \
191		< <(cat ${dst_infifo}) > ${dst_outfifo} &
192	incoming_pid=$!
193	cat ${dst_outfifo} | tee ${dst_out} | grep -v "Now migrate the VM (quiet)" &
194
195	# The test must prompt the user to migrate, so wait for the
196	# "Now migrate VM" console message.
197	while ! grep -q -i "Now migrate the VM" < ${src_out} ; do
198		if ! ps -p ${live_pid} > /dev/null ; then
199			echo "ERROR: Test exit before migration point." >&2
200			echo > ${dst_infifo}
201			qmp ${src_qmp} '"quit"'> ${src_qmpout} 2>/dev/null
202			qmp ${dst_qmp} '"quit"'> ${dst_qmpout} 2>/dev/null
203			return 3
204		fi
205		sleep 0.1
206	done
207
208	# Wait until the destination has created the incoming and qmp sockets
209	while ! [ -S ${dst_incoming} ] ; do sleep 0.1 ; done
210	while ! [ -S ${dst_qmp} ] ; do sleep 0.1 ; done
211
212	qmp ${src_qmp} '"migrate", "arguments": { "uri": "unix:'${dst_incoming}'" }' > ${src_qmpout}
213
214	# Wait for the migration to complete
215	migstatus=`qmp ${src_qmp} '"query-migrate"' | grep return`
216	while ! grep -q '"completed"' <<<"$migstatus" ; do
217		sleep 0.1
218		if ! migstatus=`qmp ${src_qmp} '"query-migrate"'`; then
219			echo "ERROR: Querying migration state failed." >&2
220			echo > ${dst_infifo}
221			qmp ${dst_qmp} '"quit"'> ${dst_qmpout} 2>/dev/null
222			return 2
223		fi
224		migstatus=`grep return <<<"$migstatus"`
225		if grep -q '"failed"' <<<"$migstatus"; then
226			echo "ERROR: Migration failed." >&2
227			echo > ${dst_infifo}
228			qmp ${src_qmp} '"quit"'> ${src_qmpout} 2>/dev/null
229			qmp ${dst_qmp} '"quit"'> ${dst_qmpout} 2>/dev/null
230			return 2
231		fi
232	done
233
234	qmp ${src_qmp} '"quit"'> ${src_qmpout} 2>/dev/null
235
236	# keypress to dst so getchar completes and test continues
237	echo > ${dst_infifo}
238	rm ${dst_infifo}
239
240	# Ensure the incoming socket is removed, ready for next destination
241	if [ -S ${dst_incoming} ] ; then
242		echo "ERROR: Incoming migration socket not removed after migration." >& 2
243		qmp ${dst_qmp} '"quit"'> ${dst_qmpout} 2>/dev/null
244		return 2
245	fi
246
247	wait ${live_pid}
248	ret=$?
249
250	# Now flip the variables because destination machine becomes source
251	# for the next migration.
252	live_pid=${incoming_pid}
253	tmp=${src_out}
254	src_out=${dst_out}
255	dst_out=${tmp}
256	tmp=${src_outfifo}
257	src_outfifo=${dst_outfifo}
258	dst_outfifo=${tmp}
259	tmp=${src_qmp}
260	src_qmp=${dst_qmp}
261	dst_qmp=${tmp}
262
263	return $ret
264}
265
266run_panic ()
267{
268	if ! command -v ncat >/dev/null 2>&1; then
269		echo "${FUNCNAME[0]} needs ncat (netcat)" >&2
270		return 77
271	fi
272
273	if ! command -v jq >/dev/null 2>&1; then
274		echo "${FUNCNAME[0]} needs jq" >&2
275		return 77
276	fi
277
278	trap 'trap - TERM ; kill 0 ; exit 2' INT TERM
279	trap 'rm -f ${qmp}' RETURN EXIT
280
281	qmp=$(mktemp -u -t panic-qmp.XXXXXXXXXX)
282
283	# start VM stopped so we don't miss any events
284	eval "$@" -chardev socket,id=mon,path=${qmp},server=on,wait=off \
285		-mon chardev=mon,mode=control -S &
286
287	panic_event_count=$(qmp_events ${qmp} | jq -c 'select(.event == "GUEST_PANICKED")' | wc -l)
288	if [ "$panic_event_count" -lt 1 ]; then
289		echo "FAIL: guest did not panic"
290		ret=3
291	else
292		# some QEMU versions report multiple panic events
293		echo "PASS: guest panicked"
294		ret=1
295	fi
296
297	return $ret
298}
299
300migration_cmd ()
301{
302	if [ "$MIGRATION" = "yes" ]; then
303		echo "run_migration"
304	fi
305}
306
307panic_cmd ()
308{
309	if [ "$PANIC" = "yes" ]; then
310		echo "run_panic"
311	fi
312}
313
314search_qemu_binary ()
315{
316	local save_path=$PATH
317	local qemucmd qemu
318
319	: "${QEMU_ARCH:=$ARCH_NAME}"
320
321	export PATH=$PATH:/usr/libexec
322	for qemucmd in ${QEMU:-qemu-system-$QEMU_ARCH qemu-kvm}; do
323		if $qemucmd --help 2>/dev/null | grep -q 'QEMU'; then
324			qemu="$qemucmd"
325			break
326		fi
327	done
328
329	if [ -z "$qemu" ]; then
330		echo "A QEMU binary was not found." >&2
331		echo "You can set a custom location by using the QEMU=<path> environment variable." >&2
332		return 2
333	fi
334	command -v $qemu
335	export PATH=$save_path
336}
337
338initrd_cleanup ()
339{
340	rm -f $KVM_UNIT_TESTS_ENV
341	if [ "$KVM_UNIT_TESTS_ENV_OLD" ]; then
342		export KVM_UNIT_TESTS_ENV="$KVM_UNIT_TESTS_ENV_OLD"
343	else
344		unset KVM_UNIT_TESTS_ENV
345	fi
346	unset KVM_UNIT_TESTS_ENV_OLD
347}
348
349initrd_create ()
350{
351	if [ "$ENVIRON_DEFAULT" = "yes" ]; then
352		trap_exit_push 'initrd_cleanup'
353		[ -f "$KVM_UNIT_TESTS_ENV" ] && export KVM_UNIT_TESTS_ENV_OLD="$KVM_UNIT_TESTS_ENV"
354		export KVM_UNIT_TESTS_ENV=$(mktemp)
355		env_params
356		env_file
357		env_errata || return $?
358	fi
359
360	unset INITRD
361	[ -f "$KVM_UNIT_TESTS_ENV" ] && INITRD="-initrd $KVM_UNIT_TESTS_ENV"
362
363	return 0
364}
365
366env_add_params ()
367{
368	local p
369
370	for p in "$@"; do
371		if eval test -v $p; then
372			eval export "$p"
373		else
374			eval export "$p="
375		fi
376		grep "^$p=" <(env) >>$KVM_UNIT_TESTS_ENV
377	done
378}
379
380env_params ()
381{
382	local qemu have_qemu
383	local _ rest
384
385	qemu=$(search_qemu_binary) && have_qemu=1
386
387	if [ "$have_qemu" ]; then
388		if [ -n "$ACCEL" ] || [ -n "$QEMU_ACCEL" ]; then
389			[ -n "$ACCEL" ] && QEMU_ACCEL=$ACCEL
390		fi
391		QEMU_VERSION_STRING="$($qemu -h | head -1)"
392		IFS='[ .]' read -r _ _ _ QEMU_MAJOR QEMU_MINOR QEMU_MICRO rest <<<"$QEMU_VERSION_STRING"
393	fi
394	env_add_params QEMU_ACCEL QEMU_VERSION_STRING QEMU_MAJOR QEMU_MINOR QEMU_MICRO
395
396	KERNEL_VERSION_STRING=$(uname -r)
397	IFS=. read -r KERNEL_VERSION KERNEL_PATCHLEVEL rest <<<"$KERNEL_VERSION_STRING"
398	IFS=- read -r KERNEL_SUBLEVEL KERNEL_EXTRAVERSION <<<"$rest"
399	KERNEL_SUBLEVEL=${KERNEL_SUBLEVEL%%[!0-9]*}
400	KERNEL_EXTRAVERSION=${KERNEL_EXTRAVERSION%%[!0-9]*}
401	! [[ $KERNEL_SUBLEVEL =~ ^[0-9]+$ ]] && unset $KERNEL_SUBLEVEL
402	! [[ $KERNEL_EXTRAVERSION =~ ^[0-9]+$ ]] && unset $KERNEL_EXTRAVERSION
403	env_add_params KERNEL_VERSION_STRING KERNEL_VERSION KERNEL_PATCHLEVEL KERNEL_SUBLEVEL KERNEL_EXTRAVERSION
404}
405
406env_file ()
407{
408	local line var
409
410	[ ! -f "$KVM_UNIT_TESTS_ENV_OLD" ] && return
411
412	for line in $(grep -E '^[[:blank:]]*[[:alpha:]_][[:alnum:]_]*=' "$KVM_UNIT_TESTS_ENV_OLD"); do
413		var=${line%%=*}
414		if ! grep -q "^$var=" $KVM_UNIT_TESTS_ENV; then
415			eval export "$line"
416			grep "^$var=" <(env) >>$KVM_UNIT_TESTS_ENV
417		fi
418	done
419}
420
421env_errata ()
422{
423	if [ "$ACCEL" = "tcg" ]; then
424		export "ERRATA_FORCE=y"
425	elif [ "$ERRATATXT" ] && [ ! -f "$ERRATATXT" ]; then
426		echo "$ERRATATXT not found. (ERRATATXT=$ERRATATXT)" >&2
427		return 2
428	elif [ "$ERRATATXT" ]; then
429		env_generate_errata
430	fi
431	sort <(env | grep '^ERRATA_') <(grep '^ERRATA_' $KVM_UNIT_TESTS_ENV) | uniq -u >>$KVM_UNIT_TESTS_ENV
432}
433
434env_generate_errata ()
435{
436	local line commit minver errata rest v p s x have
437
438	for line in $(grep -v '^#' "$ERRATATXT" | tr -d '[:blank:]' | cut -d: -f1,2); do
439		commit=${line%:*}
440		minver=${line#*:}
441
442		test -z "$commit" && continue
443		errata="ERRATA_$commit"
444		[ -n "${!errata}" ] && continue
445
446		IFS=. read -r v p rest <<<"$minver"
447		IFS=- read -r s x <<<"$rest"
448		s=${s%%[!0-9]*}
449		x=${x%%[!0-9]*}
450
451		if ! [[ $v =~ ^[0-9]+$ ]] || ! [[ $p =~ ^[0-9]+$ ]]; then
452			echo "Bad minimum kernel version in $ERRATATXT, $minver"
453			return 2
454		fi
455		! [[ $s =~ ^[0-9]+$ ]] && unset $s
456		! [[ $x =~ ^[0-9]+$ ]] && unset $x
457
458		if (( $KERNEL_VERSION > $v ||
459		      ($KERNEL_VERSION == $v && $KERNEL_PATCHLEVEL > $p) )); then
460			have=y
461		elif (( $KERNEL_VERSION == $v && $KERNEL_PATCHLEVEL == $p )); then
462			if [ "$KERNEL_SUBLEVEL" ] && [ "$s" ]; then
463				if (( $KERNEL_SUBLEVEL > $s )); then
464					have=y
465				elif (( $KERNEL_SUBLEVEL == $s )); then
466					if [ "$KERNEL_EXTRAVERSION" ] && [ "$x" ]; then
467						if (( $KERNEL_EXTRAVERSION >= $x )); then
468							have=y
469						else
470							have=n
471						fi
472					elif [ "$x" ] && (( $x != 0 )); then
473						have=n
474					else
475						have=y
476					fi
477				else
478					have=n
479				fi
480			elif [ "$s" ] && (( $s != 0 )); then
481				have=n
482			else
483				have=y
484			fi
485		else
486			have=n
487		fi
488		eval export "$errata=$have"
489	done
490}
491
492trap_exit_push ()
493{
494	local old_exit=$(trap -p EXIT | sed "s/^[^']*'//;s/'[^']*$//")
495	trap -- "$1; $old_exit" EXIT
496}
497
498kvm_available ()
499{
500	[ -c /dev/kvm ] ||
501		return 1
502
503	[ "$HOST" = "$ARCH_NAME" ] ||
504		( [ "$HOST" = aarch64 ] && [ "$ARCH" = arm ] ) ||
505		( [ "$HOST" = x86_64 ] && [ "$ARCH" = i386 ] )
506}
507
508hvf_available ()
509{
510	[ "$(sysctl -n kern.hv_support 2>/dev/null)" = "1" ] || return 1
511	[ "$HOST" = "$ARCH_NAME" ] ||
512		( [ "$HOST" = x86_64 ] && [ "$ARCH" = i386 ] )
513}
514
515set_qemu_accelerator ()
516{
517	ACCEL_PROPS=${ACCEL#"${ACCEL%%,*}"}
518	ACCEL=${ACCEL%%,*}
519
520	if [ "$ACCEL" = "kvm" ] && ! kvm_available; then
521		echo "KVM is needed, but not available on this host" >&2
522		return 2
523	fi
524	if [ "$ACCEL" = "hvf" ] && ! hvf_available; then
525		echo "HVF is needed, but not available on this host" >&2
526		return 2
527	fi
528
529	if [ -z "$ACCEL" ]; then
530		if kvm_available; then
531			ACCEL="kvm"
532		elif hvf_available; then
533			ACCEL="hvf"
534		else
535			ACCEL="tcg"
536		fi
537	fi
538
539	return 0
540}
541