Guest Automation with QEMU/KVM
Recently, working on some project, I had to do some benchmarking on qemu. This is nothing but a two step process:
1. run a guest VM with qemu,
2. run some benchmarks inside the guest and collect the results.
Naturally, I looked for options to automate this process. I tried out a bunch of things and it took me couple of days to get to the right tools.
By default, qemu uses SDL to display the VGA output. So, the first step is make this interaction with qemu through stdio. Qemu provides an option for this.
From qemu docs:
-nographic Normally, QEMU uses SDL to display the VGA output. With this option, you can totally disable graphical output so that QEMU is a simple command line application. The emulated serial port is redirected on the console. Therefore, you can still use QEMU to debug a Linux kernel with a serial console.
So, all you have to do is invoke qemu with -nographic.
qemu -nographic -hda guest.disk
Now that you can interact with your guest (or qemu process) through command line, you have to automate this interaction. The obvious way to do this in python is start the qemu process (with -nographic) with subprocess module and then communicate with that process. But to my surprise, this just didn’t work out for me. So, I looked for some other way.
Later, I found out that the most awesome tool for this kind of jobs is Expect. It is an automation tool for interactive applications written in tcl.
This guide should help you in getting started with Expect. Here is the script to run a guest with qemu using Expect.
#starts guest vm, run benchmarks, poweroff
set timeout -1
#Assign a variable to the log file
set log [lindex $argv 0]
#Start the guest VM
spawn qemu -nographic -hda guest.disk
expect “login: “
expect “Password: “
#Do whatever you want to do with in the guest VM. ( Run a process and write result to log )
#poweroff the Guest VM
expect “# “
send “shutdown -h now\r”
Save this as automate_qemu.exp, mark this file as executable and then “./automate_qemu.exp qemu.log”