‘nitro-enclave’ virtual machine (nitro-enclave)
nitro-enclave is a machine type which emulates an AWS nitro enclave
virtual machine. AWS nitro enclaves is an Amazon EC2 feature that allows
creating isolated execution environments, called enclaves, from Amazon EC2
instances which are used for processing highly sensitive data. Enclaves have
no persistent storage and no external networking. The enclave VMs are based
on Firecracker microvm with a vhost-vsock device for communication with the
parent EC2 instance that spawned it and a Nitro Secure Module (NSM) device
for cryptographic attestation. The parent instance VM always has CID 3 while
the enclave VM gets a dynamic CID. Enclaves use an EIF (Enclave Image Format)
file which contains the necessary kernel, cmdline and ramdisk(s) to boot.
In QEMU, nitro-enclave is a machine type based on microvm similar to how
AWS nitro enclaves are based on Firecracker microvm. This is useful for
local testing of EIF files using QEMU instead of running real AWS Nitro Enclaves
which can be difficult for debugging due to its roots in security. The vsock
device emulation is done using vhost-user-vsock which means another process that
can do the userspace emulation, like vhost-device-vsock from rust-vmm crate,
must be run alongside nitro-enclave for the vsock communication to work.
libcbor and gnutls are required dependencies for nitro-enclave machine
support to be added when building QEMU from source.
Using the nitro-enclave machine type
Machine-specific options
It supports the following machine-specific options:
- nitro-enclave.vsock=string (required) (Id of the chardev from ‘-chardev’ option that vhost-user-vsock device will use) 
- nitro-enclave.id=string (optional) (Set enclave identifier) 
- nitro-enclave.parent-role=string (optional) (Set parent instance IAM role ARN) 
- nitro-enclave.parent-id=string (optional) (Set parent instance identifier) 
Running a nitro-enclave VM
First, run vhost-device-vsock (or a similar tool that supports vhost-user-vsock). The forward-cid option below with value 1 forwards all connections from the enclave VM to the host machine and the forward-listen (port numbers separated by ‘+’) is used for forwarding connections from the host machine to the enclave VM.
- $ vhost-device-vsock
–vm guest-cid=4,forward-cid=1,forward-listen=9001+9002,socket=/tmp/vhost4.socket
Now run the necessary applications on the host machine so that the nitro-enclave VM applications’ vsock communication works. For example, the nitro-enclave VM’s init process connects to CID 3 and sends a single byte hello heartbeat (0xB7) to let the parent VM know that it booted expecting a heartbeat (0xB7) response. So you must run a AF_VSOCK server on the host machine that listens on port 9000 and sends the heartbeat after it receives the heartbeat for enclave VM to boot successfully. You should run all the applications on the host machine that would typically be running in the parent EC2 VM for successful communication with the enclave VM.
Then run the nitro-enclave VM using the following command where hello.eif is
an EIF file you would use to spawn a real AWS nitro enclave virtual machine:
- $ qemu-system-x86_64 -M nitro-enclave,vsock=c,id=hello-world
-kernel hello-world.eif -nographic -m 4G –enable-kvm -cpu host -chardev socket,id=c,path=/tmp/vhost4.socket
In this example, the nitro-enclave VM has CID 4. If there are applications that connect to the enclave VM, run them on the host machine after enclave VM starts. You need to modify the applications to connect to CID 1 (instead of the enclave VM’s CID) and use the forward-listen (e.g., 9001+9002) option of vhost-device-vsock to forward the ports they connect to.