[ macOS/ARM64 | Linux/AMD64 ]
Previous: Introduction
Let's learn how to launch virtual machines.
In a real-world scenario, virtual machines would be managed by a high-level tool like Vagrant. However, we want to do things the hard way in this guide, in order to see the guts of all the layers involved. Because of that, we will use perhaps the most raw tool out there for running VMs: QEMU.
This chapter is an introduction to virtualization in general, exploration of QEMU, plus an overview of the cloud-init
tool. None of the actions in this chapter are necessary to progress with the deployment, they are intended purely
for learning. You may jump straight into the next chapter if you wish.
Table of Contents generated with DocToc
- Prerequisites
- Introduction to QEMU and virtualization
- Running a Ubuntu Live CD
- Running a cloud image
- Summary
- Resources
Make sure you have all the necessary packages installed.
QEMU stands for Quick Emulator. It's an open source, command-line tool originally written by Fabrice Bellard in 2003. As its name suggests, it is primarily an emulator. It can emulate a physical computer with wide variety of physical devices, including CPUs of many architectures, network cards from many vendors, disks, flash drives, etc. Crafting a QEMU command is like building an actual computer from real hardware components.
This is very powerful - it allows us to test software written for all sorts of CPU architectures and devices without actual physical access to these devices. However, emulation means "translation on the fly" and is done in userspace, which makes it slow. Besides, testing device drivers is not our use case anyway. We don't care about simulating some specific vendor's hardware or CPU architecture, we only want to run a bunch of Linux VMs, with lowest possible overhead. In other words, we want virtualization rather than emulation.
Fortunately, virtualization isn't that far from emulation. In fact, we only need to virtualize performance-critical hardware like CPU while all the non-performance-critical devices can be emulated. Actual virtualization is done by the hypervisor, a piece of software typically implemented by the host operating system. The standard kernel-level hypervisor for Linux is KVM and for macOS, it is HVF. However, in order to make virtualization fast, a hypervisor is not enough, virtualization must also be supported by the hardware itself.
Hence the hardware-assisted virtualization.
Hardware-assisted virtualization is implemented by the host CPU. The details differ between Intel/AMD, Apple Silicon and other architectures, but usually it consists of a set of special CPU instructions and data structures that allow the hypervisor to make the CPU aware that it is currently running a virtual machine. This way the CPU knows that, at certain moments (instructions), it needs to return the control back to the hypervisor. For example, when the VM wants to perform an IO operation, the CPU cannot simply execute it on host hardware. It must give the control back to the hypervisor so that this operation can be emulated on virtual hardware.
A key part of hardware-assisted virtualization is also a second layer of memory address translation which translates between VM's virtual memory and host's virtual memory. Without it, the hypervisor would have to translate or intercept every single memory access instruction.
As a result of this hardware-level support, most of the VM code can run directly on the underlying CPU, without the need for intermediate translation or inspection of every instruction by the hypervisor. Naturally, this requires that the guest system uses the same CPU architecture as the host CPU.
QEMU, despite being originally an emulator, integrates with hypervisors and therefore supports hardware-accelerated virtualization. As we'll see soon, enabling it is just a matter of passing a certain command line option. That's great! It means we can run our Linux VMs with QEMU almost as fast as natively.
There's also another nuance worth mentioning: a technique called paravirtualization. It is a "weaker" form of virtualization where the guest system is aware that it is being virtualized. This allows simpler communication between the guest machine and the hypervisor, e.g. when the VM is doing IO operations. In practice, this means that QEMU no longer needs to emulate some specific vendor's hardware for, say, a network card. Instead, it can expose a "virtio" network card. You can think of it as a network card that's being "honest" about being virtualized instead of pretending to be some physically existing hardware. This simplifies things at the cost of the guest system needing additional drivers for these virtio devices. Fortunately, Linux supports them well so we can take full advantage of paravirtualization.
Try running QEMU with this bare-bones command:
qemu-system-aarch64 \
-machine virt,accel=hvf \
-cpu host
-machine virt
specifies a type of a machine - we have no interest in emulating a specific hardware so we just use the special typevirt
- the
accel=hvf
part is the important one: it enables hardware acceleration using macOS Hypervisor Framework -cpu host
specifies that the guest machine will see exactly the same CPU model as the host machine (required for acceleration)
Note that we haven't specified any drives yet, and as a result, there's no operating system to boot. What we're currently simulating is like a bare-bones computer without any hard drive or CDROM plugged in.
You should see a window with the QEMU monitor console:
There isn't anything particularly interesting about it, but it's a good opportunity to learn some basic QEMU controls. The console itself gives a bunch of commands for things like stopping or resuming a running VM.
What might not be apparent, though, is that this window has other "tabs" (terminals). Try hitting Ctrl
+Opt
+2
or Ctrl
+Opt
+3
and you should
see the output of serial and parallel ports. This is where we're going to see our operating system running. You can always go back to
the monitor console with Ctrl
+Opt
+1
.
Note
If you click on the QEMU window, it will capture your mouse. Hit Ctrl
+Opt
+G
to release it.
In order to kill the machine, type quit
into the console, close the window or kill the process from the terminal.
Wait, what? Even harder?
Yes! Even a raw tool like QEMU provides some nice defaults. In the previous run, QEMU automatically created at least three devices:
- the monitor console (pseudo-device)
- a serial port
- a parallel port
Just to see what's really going on and have full control, we will turn off these defaults and create these devices manually (monitor console and serial port, we'll ignore the parallel port). We achieve this with the following command:
qemu-system-aarch64 \
-nodefaults \
-machine virt,accel=hvf \
-cpu host \
-chardev vc,id=monitor \
-mon monitor \
-serial vc
-nodefaults
makes sure we have to do everything by hand- the
-chardev
option creates a host-side "backend" for character devices vc
indicates the QEMU text console (e.g. the one in the QEMU window) and probably stands for "virtual console"monitor
is an arbitrary id assigned to this character device-mon monitor
creates a monitor console that usesmonitor
as its backend-serial vc
creates a virtual serial port device that also uses QEMU text console
In order to start an operating system, we are first going to need something to boot it up, i.e. a BIOS or UEFI, on an emulated, read-only flash drive. QEMU uses BIOS by default. However, it is obsolete and won't work on an Apple Silicon CPU. We'll need to plug in an UEFI flash drive instead.
QEMU comes with bundled, open source impelementation of UEFI firmware called OVMF (Open Virtual Machine Firmware), which is a port of proprietary Intel UEFI implementation (TianoCore). If you installed QEMU with Homebrew, you should be able to find it in this location:
/opt/homebrew/share/qemu/edk2-aarch64-code.fd
The simplest QEMU option to mount OVMF file as a BIOS/UEFI is:
-bios /opt/homebrew/share/qemu/edk2-aarch64-code.fd
However, we can be a little more explicit:
-drive if=pflash,readonly=on,format=raw,file=/opt/homebrew/share/qemu/edk2-aarch64-code.fd
This is a common situation in QEMU - we can use very raw and detailed options and wire every device manually, or we can use some convenient shorthands. Since we're doing things the hard way, I try to show the most low-level version possible, but as our command grows new options, we don't want it to become too bloated, so let's go back and use the shorthand:
qemu-system-aarch64 \
-nodefaults \
-machine virt,accel=hvf \
-cpu host \
-chardev vc,id=monitor \
-mon monitor \
-serial vc \
-bios /opt/homebrew/share/qemu/edk2-aarch64-code.fd
Now if you go to the serial console (using Ctrl
+Opt
+2
in the QEMU window), you should see the UEFI running:
We haven't provided any drive with an actual operating system though, so nothing boots up and UEFI defaults to a console.
Note
Usually, along with the UEFI firmware itself, another, writable flash drive is mounted for UEFI variables, i.e. user-editable UEFI settings. Since we don't plan to modify these settings, we'll omit this drive.
So far we have a VM with a monitor console, serial console and a UEFI flash drive. Let's add a CDROM drive with a Live CD Ubuntu distribution to finally have a working operating system!
Let's download a Live CD image for Ubuntu Jammy:
Warning
Unfortunately, as of Jan 2025 Ubuntu no longer publishes LiveCD images for ARM64 architecture that this chapter refers to, and at this moment it is not possible to replicate the hands-on instructions. Despite that, just reading through it may still be valuable for learning.
wget https://cdimage.ubuntu.com/jammy/daily-live/current/jammy-desktop-arm64.iso
The shortest option to mount it as a CD-ROM is:
-cdrom jammy-desktop-arm64.iso
which has a longer version:
-drive file=jammy-desktop-arm64.iso,index=2,media=cdrom
...which can be further split into a separate "backend" (-blockdev
) and "frontend" (-device
):
-blockdev node-name=cdrom,driver=file,read-only=on,filename=jammy-desktop-arm64.iso \
-device virtio-blk-pci,drive=cdrom
Again, we will use the simplest version.
But before we run the machine, we also need to give it some resources. By default, QEMU will allocate a very small amount of RAM (e.g. 128MB) which is certainly not enough to run an everyday Linux distribution.
Let's give it 2GB for a start. An option for that is:
-m 2G
We will also add a highmem=on
property for the -machine
option so that we can increase the amount of physical memory
to over 4GB in the future.
This is also a good moment to explicitly assign the number of virtual CPUs to our VM. Let's give it 2 CPUs:
-smp 2
Note
smp
stands for symmetric multiprocessing
Note
These CPUs do not necessarily map directly to host CPUs. There is a scheduling algorithm involved between the host and guest CPUs. It is even possible for a VM to have more CPUs than there are present on the host machine.
Ultimately we end up with this command:
qemu-system-aarch64 \
-nodefaults \
-machine virt,accel=hvf,highmem=on \
-cpu host \
-smp 2 \
-m 2G \
-chardev vc,id=monitor \
-mon monitor \
-serial vc \
-bios /opt/homebrew/share/qemu/edk2-aarch64-code.fd \
-cdrom jammy-desktop-arm64.iso
If you now go to the serial console (using Ctrl
+Opt
+2
in the QEMU window), you'll see that UEFI has picked up the
new drive and detected a system on it:
Hit enter and launch the system. Soon you should see a login screen:
Log in as ubuntu
and you're in.
Congratulations! You've successfully run a Linux distribution using raw QEMU.
The machine we have built has no network access. Let's change that.
There are several ways QEMU can emulate a network interface. They can roughly be split into two categories:
- userspace backends - usually slow and limited but available on all host OSes
- native backends - implemented by host OSes and their hypervisors, usually fast but OS-dependent and may require escalated privileges
Some of the most commonly used network backends are:
-
user
is an userspace implemented network (SLIRP) between the host and guest OS. Allows the guest to access the guest machine to communicate with the host, and provides internet access to the guest. Unfortunately, guest machine is not addressable from the host machine in this mode. It also has poor performance, being implemented in userspace. For these reasons we will not be using it. It is however worth mentioning because it is the default network backend that QEMU sets up if we don't configure one (and provided that we don't use-nodefaults
). -
tap
creates a virtual layer 2 network interface on the host machine connected to the guest machine. This is versatile and has good performance, but generally requires root privileges. Unfortunately, macOS does not currently support it so we won't be able to use it in this tutorial. -
bridge
connects the VM to a network bridge that needs to be previously set up on the host machine and connected to one of host's native interfaces. This effectively makes the VM appear in the same network that the host machine lives in, making it visible to the external world. The VM will then typically get configured by the same DHCP server as the host machine (e.g. your home router). Unfortunately, this exact mode is not available on macOS either, but it has a macOS specific equivalent calledvmnet-bridged
.
There are many more modes which we will not cover here. We are on macOS, which provides three additional modes implemented
by its vmnet
framework:
vmnet-host
- a host-only network that allows the guest to communicate with the host but without internet accessvmnet-shared
- allows the guest to communicate with the host and provides it with internet access via NATvmnet-bridged
- just likebridge
mode, connects the VM to a layer 2 bridge, but the bridge itself is set up automatically
When using these modes, macOS also automatically configures its built-in DHCP server and starts a DNS server so that the VM can have a properly configured network.
In this guide we'll always use the vmnet-shared
network backend. Let's enable it with the following option:
-nic vmnet-shared
A longer version (with separate "backend" and "frontend") would be:
-netdev vmnet-shared,id=net0 \
-device virtio-net-pci,netdev=net0
These options can be refined with additional properties, e.g. we can choose the IP address range for shared network and decide to isolate it from other VMs. We can also manually assign a MAC address to the guest VM. We will use some of these options later, when setting up an actual Kubernetes machine.
Unfortunately, usage of vmnet
requires escalated privileges on macOS, so from now on we must run QEMU using sudo
:
sudo qemu-system-aarch64 \
-nodefaults \
-machine virt,accel=hvf,highmem=on \
-cpu host \
-smp 2 \
-m 2G \
-chardev vc,id=monitor \
-mon monitor \
-serial vc \
-bios /opt/homebrew/share/qemu/edk2-aarch64-code.fd \
-cdrom jammy-desktop-arm64.iso \
-nic vmnet-shared
Let's log into Ubuntu and run the ip addr
command. We can see a new virtual ethernet interface with a nicely
assigned IP address:
You can also run ping google.com
to confirm internet access.
Ubuntu on Live CD is a graphical distribution. It would be nice to see a proper graphical output, hear some sounds and be able to work with mouse and keyboard in a graphical shell. We're not going to need any of this for our Kubernetes deployment so I will not explain all the QEMU options in detail, but for the sake of completeness, this is the full command that runs our VM with support for all these devices:
sudo qemu-system-aarch64 \
-nodefaults \
-machine virt,accel=hvf,highmem=on \
-cpu host \
-smp 2 \
-m 2G \
-chardev vc,id=monitor \
-mon monitor \
-serial vc \
-bios /opt/homebrew/share/qemu/edk2-aarch64-code.fd \
-cdrom jammy-desktop-arm64.iso \
-nic vmnet-shared \
-device virtio-gpu-pci \
-display cocoa,show-cursor=on \
-device nec-usb-xhci \
-device usb-mouse \
-device usb-tablet \
-device usb-kbd \
-audiodev coreaudio,id=audio0 \
-device ich9-intel-hda \
-device hda-output,audiodev=audio0
And here it is running:
The final missing piece for our system to be fully functional is a disk drive. We could use it to install Ubuntu from Live CD distribution, just like on a physical machine.
A disk drive will be backed by a file on a host machine. There are many formats for disk images. One of the most commonly used
in QEMU is QCOW2, which stands for QEMU Copy On Write version 2. We'll explain what that means later. For now let's
just create an image file with maximum size of 128GB. This can be done with qemu-img
utility:
qemu-img create -f qcow2 ubuntu.img 128G
Note
The image won't immediately take 128GB of your disk. It grows dynamically, as more space is requested by the VM.
Now, mounting this file as a drive to a VM is as simple as:
-hda ubuntu.img
Unsurprisingly, this is a shorthand for something more verbose:
-drive file=ubuntu.img,index=0,media=disk
...or even longer:
-blockdev node-name=disk,driver=qcow2,file.driver=file,file.filename=ubuntu.img \
-device virtio-blk-pci,drive=disk
And the full command:
sudo qemu-system-aarch64 \
-nodefaults \
-machine virt,accel=hvf,highmem=on \
-cpu host \
-smp 2 \
-m 2G \
-chardev vc,id=monitor \
-mon monitor \
-serial vc \
-bios /opt/homebrew/share/qemu/edk2-aarch64-code.fd \
-cdrom jammy-desktop-arm64.iso \
-nic vmnet-shared \
-hda ubuntu.img \
-device virtio-gpu-pci \
-display cocoa,show-cursor=on \
-device nec-usb-xhci \
-device usb-mouse \
-device usb-tablet \
-device usb-kbd \
-audiodev coreaudio,id=audio0 \
-device ich9-intel-hda \
-device hda-output,audiodev=audio0
Now you can install your Ubuntu on this drive. After you do that, you can remove the CDROM device and image and work with the system on a disk. This is what you would typically do with a desktop-like virtual machine. However, for our ultimate goal - a working Kubernetes deployment - we will take a different route.
We're done with the Live CD distribution. It was nice for playing with QEMU, but now we're taking a step back in order to prepare a more server-like distribution. Here's what's going to change:
- instead of using a Live CD image, we will use a cloud image
- the server will run headless, so no graphics or peripherals
- we will run the VM entirely in terminal (no QEMU window)
Cloud image is a disk image with a preinstalled Ubuntu distribution. It is optimized for server usage (headless) and requires some additional, automated preconfiguration (e.g. to set up remote SSH access).
Let's download a Jammy cloud image for AArch64:
wget https://cloud-images.ubuntu.com/jammy/current/jammy-server-cloudimg-arm64.img
This file is in QCOW2 format.
As already mentioned, QCOW2 stands for QEMU Copy On Write Version 2. The important part of this name is copy on write, which describes a key capability of this format. It allows us to create a QCOW2 image that is backed by another QCOW2 image. This is what it means:
- the new image effectively represents a diff over the backing image
- when data present only in the backing image is modified by the VM, it is copied into the new image and modified there, while the backing image stays unchanged
- reading data from the new image that was never written "falls through" to the backing image
This is very useful. Since the backing image never changes, it can be used as a backing image for multiple other images (e.g. for multiple VMs). This allows significant space savings on the host machine if it is running multiple similar machines. It also allows us to quickly reset a VM to its original state, with the backing image serving as a snapshot of VM's state from the past.
Let's create an image backed by the Ubuntu cloud image that we have just downloaded:
qemu-img create -F qcow2 -b jammy-server-cloudimg-arm64.img -f qcow2 ubuntu0.img 128G
Note
The -F
specifies the format of the backing image while -f
specifies the format of the image being created.
Let's run the server in QEMU. This time we'll just boot the newly created image as a disk, there will be no CDROM. We will also strip down the VM from all graphics and peripherals and make QEMU run fully in terminal instead of a window. Ultimately, our goal is to make the VM accessible only via (properly secured) SSH and (as a last resort) via the serial port.
The QEMU option we want is -nographic
, which removes the window and connects VM's monitor and serial port to
QEMU's standard input and output. Or at least that's what would happen if we weren't overly pedantic in this tutorial
and didn't use -nodefaults
... Ok, let's stop being pedantic and drop it.
We end up with a much simplified command:
sudo qemu-system-aarch64 \
-nographic \
-machine virt,accel=hvf,highmem=on \
-cpu host \
-smp 2 \
-m 2G \
-bios /opt/homebrew/share/qemu/edk2-aarch64-code.fd \
-nic vmnet-shared \
-hda ubuntu0.img
Run it, and soon you should see logs of the Linux kernel starting in the terminal, followed by a login prompt:
You can switch between serial port output and QEMU monitor console using Ctrl
+A
followed by C
.
You can also kill the VM with Ctrl
+A
followed by X
. For a help on these keyboard shortcuts, use Ctrl
+A
followed by H
.
Yay, our system runs! Unfortunately, we can't log in. Any credentials we try to use will be rejected. We need some additional initialization, and we'll deal with that in the next section.
A VM running a bare cloud image is inaccessible as it has no password or SSH key configured for the default user (ubuntu
).
They need to be injected using more "low-level" means. This is done with the cloud-init
project.
cloud-init
is a piece of software built into the system preinstalled on the cloud image. It is able to detect various sources of configuration in its "environment"
(usually the cloud platform it's running in) and let itself be provisioned with this configuration. This sounds vague. The exact details of how this happens
heavily depend on the cloud platform. Usually it involves consulting some magic IP address, reading something from kernel boot parameters or
SMBIOS, or reading the configuration from a special drive mounted to the VM.
The cloud-init
project refers to these methods as datasources.
Since we're running on a laptop, the datasource for us is called... nocloud
.
Using this datasource, the guest OS can pull configuration from its local filesystem, a specially mounted drive, or an external web server. The drive method seems to
be the simplest for our needs as it does not require passing any kernel or SMBIOS parameters or running a web server.
First, we must prepare the configuration data. It is organized in a set of YAML files which must be called
user-data
, meta-data
, vendor-data
, and network-config
(and possibly others which I am not aware of).
Out of these, user-data
and meta-data
are required while the rest is optional. Let's put them into a directory:
mkdir cloud-init
touch cloud-init/{user-data,meta-data}
These files can configure a multitude of things, including passwords, SSH keys, CA certificates, auto-installed packages, system configuration files, etc.
Various settings are provided by different modules - refer to the documentation
fore more details. In this section we are only interested in a minimal configuration that will allow us to log into our machine. We'll set up an initial
password for the ubuntu
user. We can do it by putting this setting into user-data
:
#cloud-config
password: ubuntu
Note
The #cloud-config
is a magic comment that must be present at the beginning of user-data
file to be picked up by cloud-init
.
Now we need to format a special ISO drive with these files. The drive must be labeled as cidata
in order for cloud-init
to recognize it.
On macOS, the command to do this is mkisofs
from cdrtools
package. Let's build the ISO:
mkisofs -output cidata.iso -volid cidata -joliet -rock cloud-init/{user-data,meta-data}
Now we can plug it into the VM with this QEMU option:
-drive file=cidata.iso,driver=raw,if=virtio
But before we launch the VM, there's one more important thing to remember: cloud-init
picks up configuration
only during the first boot of the VM. If we have already launched the machine at least once (which we have), it won't work
and the machine will remain inaccessible for eternity.
We can deal with this in two ways:
-
Reset the VM to its initial state. We can do that simply by reformatting its image file, using the same command that was used to create it, i.e.
qemu-img create -F qcow2 -b jammy-server-cloudimg-arm64.img -f qcow2 ubuntu0.img 128G
This is where the QCOW2 format comes in handy - we effectively removed only the "diff" over the original cloud image.
-
Provide or change
instance-id
in themeta-data
file:instance-id: ubuntu0
This ID us used by
cloud-init
to determine whether it is running during a "first boot". Changinginstance-id
is like tellingcloud-init
"you are a different machine now" which makes it initialize the VM from scratch.
Now we can finally launch the VM and log in as ubuntu
with ubuntu
password:
sudo qemu-system-aarch64 \
-nographic \
-machine virt,accel=hvf,highmem=on \
-cpu host \
-smp 2 \
-m 2G \
-bios /opt/homebrew/share/qemu/edk2-aarch64-code.fd \
-nic vmnet-shared \
-hda ubuntu0.img \
-drive file=cidata.iso,driver=raw,if=virtio
After logging in, the system immediately asks for a password change:
Ubuntu 22.04.3 LTS ubuntu ttyAMA0
ubuntu login: ubuntu
Password:
You are required to change your password immediately (administrator enforced).
Changing password for ubuntu.
Current password:
We do what it wants, and we're finally in. Yay!
In this chapter, we have:
- learnt the differences between emulation, virtualization and paravirtualization
- learnt the basic principles of a hypervisor and hardware assisted virtualization
- learnt how to launch QEMU and how to craft a virtual machine from its vast set of options
- learnt how to work with VM images
- learnt how to succesfully launch a headless cloud image and initialize it using
cloud-init
In the next chapter, we will proceed to build a multi-machine cluster in preparation for installing Kubernetes on it.
- QEMU main page
- QEMU ELI5 by Tuna Cici
qemu-system-aarch64
manpagecloud-init