| =============== |
| QEMU TPM Device |
| =============== |
| |
| Guest-side hardware interface |
| ============================= |
| |
| TIS interface |
| ------------- |
| |
| The QEMU TPM emulation implements a TPM TIS hardware interface |
| following the Trusted Computing Group's specification "TCG PC Client |
| Specific TPM Interface Specification (TIS)", Specification Version |
| 1.3, 21 March 2013. (see the `TIS specification`_, or a later version |
| of it). |
| |
| The TIS interface makes a memory mapped IO region in the area |
| 0xfed40000-0xfed44fff available to the guest operating system. |
| |
| QEMU files related to TPM TIS interface: |
| - ``hw/tpm/tpm_tis_common.c`` |
| - ``hw/tpm/tpm_tis_isa.c`` |
| - ``hw/tpm/tpm_tis_sysbus.c`` |
| - ``hw/tpm/tpm_tis_i2c.c`` |
| - ``hw/tpm/tpm_tis.h`` |
| |
| Both an ISA device and a sysbus device are available. The former is |
| used with pc/q35 machine while the latter can be instantiated in the |
| Arm virt machine. |
| |
| An I2C device support is also provided which can be instantiated in the Arm |
| based emulation machines. This device only supports the TPM 2 protocol. |
| |
| CRB interface |
| ------------- |
| |
| QEMU also implements a TPM CRB interface following the Trusted |
| Computing Group's specification "TCG PC Client Platform TPM Profile |
| (PTP) Specification", Family "2.0", Level 00 Revision 01.03 v22, May |
| 22, 2017. (see the `CRB specification`_, or a later version of it) |
| |
| The CRB interface makes a memory mapped IO region in the area |
| 0xfed40000-0xfed40fff (1 locality) available to the guest |
| operating system. |
| |
| QEMU files related to TPM CRB interface: |
| - ``hw/tpm/tpm_crb.c`` |
| |
| SPAPR interface |
| --------------- |
| |
| pSeries (ppc64) machines offer a tpm-spapr device model. |
| |
| QEMU files related to the SPAPR interface: |
| - ``hw/tpm/tpm_spapr.c`` |
| |
| fw_cfg interface |
| ================ |
| |
| The bios/firmware may read the ``"etc/tpm/config"`` fw_cfg entry for |
| configuring the guest appropriately. |
| |
| The entry of 6 bytes has the following content, in little-endian: |
| |
| .. code-block:: c |
| |
| #define TPM_VERSION_UNSPEC 0 |
| #define TPM_VERSION_1_2 1 |
| #define TPM_VERSION_2_0 2 |
| |
| #define TPM_PPI_VERSION_NONE 0 |
| #define TPM_PPI_VERSION_1_30 1 |
| |
| struct FwCfgTPMConfig { |
| uint32_t tpmppi_address; /* PPI memory location */ |
| uint8_t tpm_version; /* TPM version */ |
| uint8_t tpmppi_version; /* PPI version */ |
| }; |
| |
| ACPI interface |
| ============== |
| |
| The TPM device is defined with ACPI ID "PNP0C31". QEMU builds a SSDT |
| and passes it into the guest through the fw_cfg device. The device |
| description contains the base address of the TIS interface 0xfed40000 |
| and the size of the MMIO area (0x5000). In case a TPM2 is used by |
| QEMU, a TPM2 ACPI table is also provided. The device is described to |
| be used in polling mode rather than interrupt mode primarily because |
| no unused IRQ could be found. |
| |
| To support measurement logs to be written by the firmware, |
| e.g. SeaBIOS, a TCPA table is implemented. This table provides a 64kb |
| buffer where the firmware can write its log into. For TPM 2 only a |
| more recent version of the TPM2 table provides support for |
| measurements logs and a TCPA table does not need to be created. |
| |
| The TCPA and TPM2 ACPI tables follow the Trusted Computing Group |
| specification "TCG ACPI Specification" Family "1.2" and "2.0", Level |
| 00 Revision 00.37. (see the `ACPI specification`_, or a later version |
| of it) |
| |
| ACPI PPI Interface |
| ------------------ |
| |
| QEMU supports the Physical Presence Interface (PPI) for TPM 1.2 and |
| TPM 2. This interface requires ACPI and firmware support. (see the |
| `PPI specification`_) |
| |
| PPI enables a system administrator (root) to request a modification to |
| the TPM upon reboot. The PPI specification defines the operation |
| requests and the actions the firmware has to take. The system |
| administrator passes the operation request number to the firmware |
| through an ACPI interface which writes this number to a memory |
| location that the firmware knows. Upon reboot, the firmware finds the |
| number and sends commands to the TPM. The firmware writes the TPM |
| result code and the operation request number to a memory location that |
| ACPI can read from and pass the result on to the administrator. |
| |
| The PPI specification defines a set of mandatory and optional |
| operations for the firmware to implement. The ACPI interface also |
| allows an administrator to list the supported operations. In QEMU the |
| ACPI code is generated by QEMU, yet the firmware needs to implement |
| support on a per-operations basis, and different firmwares may support |
| a different subset. Therefore, QEMU introduces the virtual memory |
| device for PPI where the firmware can indicate which operations it |
| supports and ACPI can enable the ones that are supported and disable |
| all others. This interface lies in main memory and has the following |
| layout: |
| |
| +-------------+--------+--------+-------------------------------------------+ |
| | Field | Length | Offset | Description | |
| +=============+========+========+===========================================+ |
| | ``func`` | 0x100 | 0x000 | Firmware sets values for each supported | |
| | | | | operation. See defined values below. | |
| +-------------+--------+--------+-------------------------------------------+ |
| | ``ppin`` | 0x1 | 0x100 | SMI interrupt to use. Set by firmware. | |
| | | | | Not supported. | |
| +-------------+--------+--------+-------------------------------------------+ |
| | ``ppip`` | 0x4 | 0x101 | ACPI function index to pass to SMM code. | |
| | | | | Set by ACPI. Not supported. | |
| +-------------+--------+--------+-------------------------------------------+ |
| | ``pprp`` | 0x4 | 0x105 | Result of last executed operation. Set by | |
| | | | | firmware. See function index 5 for values.| |
| +-------------+--------+--------+-------------------------------------------+ |
| | ``pprq`` | 0x4 | 0x109 | Operation request number to execute. See | |
| | | | | 'Physical Presence Interface Operation | |
| | | | | Summary' tables in specs. Set by ACPI. | |
| +-------------+--------+--------+-------------------------------------------+ |
| | ``pprm`` | 0x4 | 0x10d | Operation request optional parameter. | |
| | | | | Values depend on operation. Set by ACPI. | |
| +-------------+--------+--------+-------------------------------------------+ |
| | ``lppr`` | 0x4 | 0x111 | Last executed operation request number. | |
| | | | | Copied from pprq field by firmware. | |
| +-------------+--------+--------+-------------------------------------------+ |
| | ``fret`` | 0x4 | 0x115 | Result code from SMM function. | |
| | | | | Not supported. | |
| +-------------+--------+--------+-------------------------------------------+ |
| | ``res1`` | 0x40 | 0x119 | Reserved for future use | |
| +-------------+--------+--------+-------------------------------------------+ |
| |``next_step``| 0x1 | 0x159 | Operation to execute after reboot by | |
| | | | | firmware. Used by firmware. | |
| +-------------+--------+--------+-------------------------------------------+ |
| | ``movv`` | 0x1 | 0x15a | Memory overwrite variable | |
| +-------------+--------+--------+-------------------------------------------+ |
| |
| The following values are supported for the ``func`` field. They |
| correspond to the values used by ACPI function index 8. |
| |
| +----------+-------------------------------------------------------------+ |
| | Value | Description | |
| +==========+=============================================================+ |
| | 0 | Operation is not implemented. | |
| +----------+-------------------------------------------------------------+ |
| | 1 | Operation is only accessible through firmware. | |
| +----------+-------------------------------------------------------------+ |
| | 2 | Operation is blocked for OS by firmware configuration. | |
| +----------+-------------------------------------------------------------+ |
| | 3 | Operation is allowed and physically present user required. | |
| +----------+-------------------------------------------------------------+ |
| | 4 | Operation is allowed and physically present user is not | |
| | | required. | |
| +----------+-------------------------------------------------------------+ |
| |
| The location of the table is given by the fw_cfg ``tpmppi_address`` |
| field. The PPI memory region size is 0x400 (``TPM_PPI_ADDR_SIZE``) to |
| leave enough room for future updates. |
| |
| QEMU files related to TPM ACPI tables: |
| - ``hw/i386/acpi-build.c`` |
| - ``include/hw/acpi/tpm.h`` |
| |
| TPM backend devices |
| =================== |
| |
| The TPM implementation is split into two parts, frontend and |
| backend. The frontend part is the hardware interface, such as the TPM |
| TIS interface described earlier, and the other part is the TPM backend |
| interface. The backend interfaces implement the interaction with a TPM |
| device, which may be a physical or an emulated device. The split |
| between the front- and backend devices allows a frontend to be |
| connected with any available backend. This enables the TIS interface |
| to be used with the passthrough backend or the swtpm backend. |
| |
| QEMU files related to TPM backends: |
| - ``backends/tpm.c`` |
| - ``include/sysemu/tpm.h`` |
| - ``include/sysemu/tpm_backend.h`` |
| |
| The QEMU TPM passthrough device |
| ------------------------------- |
| |
| In case QEMU is run on Linux as the host operating system it is |
| possible to make the hardware TPM device available to a single QEMU |
| guest. In this case the user must make sure that no other program is |
| using the device, e.g., /dev/tpm0, before trying to start QEMU with |
| it. |
| |
| The passthrough driver uses the host's TPM device for sending TPM |
| commands and receiving responses from. Besides that it accesses the |
| TPM device's sysfs entry for support of command cancellation. Since |
| none of the state of a hardware TPM can be migrated between hosts, |
| virtual machine migration is disabled when the TPM passthrough driver |
| is used. |
| |
| Since the host's TPM device will already be initialized by the host's |
| firmware, certain commands, e.g. ``TPM_Startup()``, sent by the |
| virtual firmware for device initialization, will fail. In this case |
| the firmware should not use the TPM. |
| |
| Sharing the device with the host is generally not a recommended usage |
| scenario for a TPM device. The primary reason for this is that two |
| operating systems can then access the device's single set of |
| resources, such as platform configuration registers |
| (PCRs). Applications or kernel security subsystems, such as the Linux |
| Integrity Measurement Architecture (IMA), are not expecting to share |
| PCRs. |
| |
| QEMU files related to the TPM passthrough device: |
| - ``backends/tpm/tpm_passthrough.c`` |
| - ``backends/tpm/tpm_util.c`` |
| - ``include/sysemu/tpm_util.h`` |
| |
| |
| Command line to start QEMU with the TPM passthrough device using the host's |
| hardware TPM ``/dev/tpm0``: |
| |
| .. code-block:: console |
| |
| qemu-system-x86_64 -display sdl -accel kvm \ |
| -m 1024 -boot d -bios bios-256k.bin -boot menu=on \ |
| -tpmdev passthrough,id=tpm0,path=/dev/tpm0 \ |
| -device tpm-tis,tpmdev=tpm0 test.img |
| |
| |
| The following commands should result in similar output inside the VM |
| with a Linux kernel that either has the TPM TIS driver built-in or |
| available as a module (assuming a TPM 2 is passed through): |
| |
| .. code-block:: console |
| |
| # dmesg | grep -i tpm |
| [ 0.012560] ACPI: TPM2 0x000000000BFFD1900 00004C (v04 BOCHS \ |
| BXPC 0000001 BXPC 00000001) |
| |
| # ls -l /dev/tpm* |
| crw-rw----. 1 tss root 10, 224 Sep 6 12:36 /dev/tpm0 |
| crw-rw----. 1 tss rss 253, 65536 Sep 6 12:36 /dev/tpmrm0 |
| |
| Starting with Linux 5.12 there are PCR entries for TPM 2 in sysfs: |
| # find /sys/devices/ -type f | grep pcr-sha |
| ... |
| /sys/devices/LNXSYSTEM:00/LNXSYBUS:00/MSFT0101:00/tpm/tpm0/pcr-sha256/1 |
| ... |
| /sys/devices/LNXSYSTEM:00/LNXSYBUS:00/MSFT0101:00/tpm/tpm0/pcr-sha256/9 |
| ... |
| |
| The QEMU TPM emulator device |
| ---------------------------- |
| |
| The TPM emulator device uses an external TPM emulator called 'swtpm' |
| for sending TPM commands to and receiving responses from. The swtpm |
| program must have been started before trying to access it through the |
| TPM emulator with QEMU. |
| |
| The TPM emulator implements a command channel for transferring TPM |
| commands and responses as well as a control channel over which control |
| commands can be sent. (see the `SWTPM protocol`_ specification) |
| |
| The control channel serves the purpose of resetting, initializing, and |
| migrating the TPM state, among other things. |
| |
| The swtpm program behaves like a hardware TPM and therefore needs to |
| be initialized by the firmware running inside the QEMU virtual |
| machine. One necessary step for initializing the device is to send |
| the TPM_Startup command to it. SeaBIOS, for example, has been |
| instrumented to initialize a TPM 1.2 or TPM 2 device using this |
| command. |
| |
| QEMU files related to the TPM emulator device: |
| - ``backends/tpm/tpm_emulator.c`` |
| - ``backends/tpm/tpm_util.c`` |
| - ``include/sysemu/tpm_util.h`` |
| |
| The following commands start the swtpm with a UnixIO control channel over |
| a socket interface. They do not need to be run as root. |
| |
| .. code-block:: console |
| |
| mkdir /tmp/mytpm1 |
| swtpm socket --tpmstate dir=/tmp/mytpm1 \ |
| --ctrl type=unixio,path=/tmp/mytpm1/swtpm-sock \ |
| --tpm2 \ |
| --log level=20 |
| |
| Command line to start QEMU with the TPM emulator device communicating |
| with the swtpm (x86): |
| |
| .. code-block:: console |
| |
| qemu-system-x86_64 -display sdl -accel kvm \ |
| -m 1024 -boot d -bios bios-256k.bin -boot menu=on \ |
| -chardev socket,id=chrtpm,path=/tmp/mytpm1/swtpm-sock \ |
| -tpmdev emulator,id=tpm0,chardev=chrtpm \ |
| -device tpm-tis,tpmdev=tpm0 test.img |
| |
| In case a pSeries machine is emulated, use the following command line: |
| |
| .. code-block:: console |
| |
| qemu-system-ppc64 -display sdl -machine pseries,accel=kvm \ |
| -m 1024 -bios slof.bin -boot menu=on \ |
| -nodefaults -device VGA -device pci-ohci -device usb-kbd \ |
| -chardev socket,id=chrtpm,path=/tmp/mytpm1/swtpm-sock \ |
| -tpmdev emulator,id=tpm0,chardev=chrtpm \ |
| -device tpm-spapr,tpmdev=tpm0 \ |
| -device spapr-vscsi,id=scsi0,reg=0x00002000 \ |
| -device virtio-blk-pci,scsi=off,bus=pci.0,addr=0x3,drive=drive-virtio-disk0,id=virtio-disk0 \ |
| -drive file=test.img,format=raw,if=none,id=drive-virtio-disk0 |
| |
| In case an Arm virt machine is emulated, use the following command line: |
| |
| .. code-block:: console |
| |
| qemu-system-aarch64 -machine virt,gic-version=3,accel=kvm \ |
| -cpu host -m 4G \ |
| -nographic -no-acpi \ |
| -chardev socket,id=chrtpm,path=/tmp/mytpm1/swtpm-sock \ |
| -tpmdev emulator,id=tpm0,chardev=chrtpm \ |
| -device tpm-tis-device,tpmdev=tpm0 \ |
| -device virtio-blk-pci,drive=drv0 \ |
| -drive format=qcow2,file=hda.qcow2,if=none,id=drv0 \ |
| -drive if=pflash,format=raw,file=flash0.img,readonly=on \ |
| -drive if=pflash,format=raw,file=flash1.img |
| |
| In case a ast2600-evb bmc machine is emulated and you want to use a TPM device |
| attached to I2C bus, use the following command line: |
| |
| .. code-block:: console |
| |
| qemu-system-arm -M ast2600-evb -nographic \ |
| -kernel arch/arm/boot/zImage \ |
| -dtb arch/arm/boot/dts/aspeed-ast2600-evb.dtb \ |
| -initrd rootfs.cpio \ |
| -chardev socket,id=chrtpm,path=/tmp/mytpm1/swtpm-sock \ |
| -tpmdev emulator,id=tpm0,chardev=chrtpm \ |
| -device tpm-tis-i2c,tpmdev=tpm0,bus=aspeed.i2c.bus.12,address=0x2e |
| |
| For testing, use this command to load the driver to the correct address |
| |
| echo tpm_tis_i2c 0x2e > /sys/bus/i2c/devices/i2c-12/new_device |
| |
| In case SeaBIOS is used as firmware, it should show the TPM menu item |
| after entering the menu with 'ESC'. |
| |
| .. code-block:: console |
| |
| Select boot device: |
| 1. DVD/CD [ata1-0: QEMU DVD-ROM ATAPI-4 DVD/CD] |
| [...] |
| 5. Legacy option rom |
| |
| t. TPM Configuration |
| |
| The following commands should result in similar output inside the VM |
| with a Linux kernel that either has the TPM TIS driver built-in or |
| available as a module: |
| |
| .. code-block:: console |
| |
| # dmesg | grep -i tpm |
| [ 0.012560] ACPI: TPM2 0x000000000BFFD1900 00004C (v04 BOCHS \ |
| BXPC 0000001 BXPC 00000001) |
| |
| # ls -l /dev/tpm* |
| crw-rw----. 1 tss root 10, 224 Sep 6 12:36 /dev/tpm0 |
| crw-rw----. 1 tss rss 253, 65536 Sep 6 12:36 /dev/tpmrm0 |
| |
| Starting with Linux 5.12 there are PCR entries for TPM 2 in sysfs: |
| # find /sys/devices/ -type f | grep pcr-sha |
| ... |
| /sys/devices/LNXSYSTEM:00/LNXSYBUS:00/MSFT0101:00/tpm/tpm0/pcr-sha256/1 |
| ... |
| /sys/devices/LNXSYSTEM:00/LNXSYBUS:00/MSFT0101:00/tpm/tpm0/pcr-sha256/9 |
| ... |
| |
| Migration with the TPM emulator |
| =============================== |
| |
| The TPM emulator supports the following types of virtual machine |
| migration: |
| |
| - VM save / restore (migration into a file) |
| - Network migration |
| - Snapshotting (migration into storage like QoW2 or QED) |
| |
| The following command sequences can be used to test VM save / restore. |
| |
| In a 1st terminal start an instance of a swtpm using the following command: |
| |
| .. code-block:: console |
| |
| mkdir /tmp/mytpm1 |
| swtpm socket --tpmstate dir=/tmp/mytpm1 \ |
| --ctrl type=unixio,path=/tmp/mytpm1/swtpm-sock \ |
| --tpm2 \ |
| --log level=20 |
| |
| In a 2nd terminal start the VM: |
| |
| .. code-block:: console |
| |
| qemu-system-x86_64 -display sdl -accel kvm \ |
| -m 1024 -boot d -bios bios-256k.bin -boot menu=on \ |
| -chardev socket,id=chrtpm,path=/tmp/mytpm1/swtpm-sock \ |
| -tpmdev emulator,id=tpm0,chardev=chrtpm \ |
| -device tpm-tis,tpmdev=tpm0 \ |
| -monitor stdio \ |
| test.img |
| |
| Verify that the attached TPM is working as expected using applications |
| inside the VM. |
| |
| To store the state of the VM use the following command in the QEMU |
| monitor in the 2nd terminal: |
| |
| .. code-block:: console |
| |
| (qemu) migrate "exec:cat > testvm.bin" |
| (qemu) quit |
| |
| At this point a file called ``testvm.bin`` should exists and the swtpm |
| and QEMU processes should have ended. |
| |
| To test 'VM restore' you have to start the swtpm with the same |
| parameters as before. If previously a TPM 2 [--tpm2] was saved, --tpm2 |
| must now be passed again on the command line. |
| |
| In the 1st terminal restart the swtpm with the same command line as |
| before: |
| |
| .. code-block:: console |
| |
| swtpm socket --tpmstate dir=/tmp/mytpm1 \ |
| --ctrl type=unixio,path=/tmp/mytpm1/swtpm-sock \ |
| --log level=20 --tpm2 |
| |
| In the 2nd terminal restore the state of the VM using the additional |
| '-incoming' option. |
| |
| .. code-block:: console |
| |
| qemu-system-x86_64 -display sdl -accel kvm \ |
| -m 1024 -boot d -bios bios-256k.bin -boot menu=on \ |
| -chardev socket,id=chrtpm,path=/tmp/mytpm1/swtpm-sock \ |
| -tpmdev emulator,id=tpm0,chardev=chrtpm \ |
| -device tpm-tis,tpmdev=tpm0 \ |
| -incoming "exec:cat < testvm.bin" \ |
| test.img |
| |
| Troubleshooting migration |
| ------------------------- |
| |
| There are several reasons why migration may fail. In case of problems, |
| please ensure that the command lines adhere to the following rules |
| and, if possible, that identical versions of QEMU and swtpm are used |
| at all times. |
| |
| VM save and restore: |
| |
| - QEMU command line parameters should be identical apart from the |
| '-incoming' option on VM restore |
| |
| - swtpm command line parameters should be identical |
| |
| VM migration to 'localhost': |
| |
| - QEMU command line parameters should be identical apart from the |
| '-incoming' option on the destination side |
| |
| - swtpm command line parameters should point to two different |
| directories on the source and destination swtpm (--tpmstate dir=...) |
| (especially if different versions of libtpms were to be used on the |
| same machine). |
| |
| VM migration across the network: |
| |
| - QEMU command line parameters should be identical apart from the |
| '-incoming' option on the destination side |
| |
| - swtpm command line parameters should be identical |
| |
| VM Snapshotting: |
| - QEMU command line parameters should be identical |
| |
| - swtpm command line parameters should be identical |
| |
| |
| Besides that, migration failure reasons on the swtpm level may include |
| the following: |
| |
| - the versions of the swtpm on the source and destination sides are |
| incompatible |
| |
| - downgrading of TPM state may not be supported |
| |
| - the source and destination libtpms were compiled with different |
| compile-time options and the destination side refuses to accept the |
| state |
| |
| - different migration keys are used on the source and destination side |
| and the destination side cannot decrypt the migrated state |
| (swtpm ... --migration-key ... ) |
| |
| |
| .. _TIS specification: |
| https://trustedcomputinggroup.org/pc-client-work-group-pc-client-specific-tpm-interface-specification-tis/ |
| |
| .. _CRB specification: |
| https://trustedcomputinggroup.org/resource/pc-client-platform-tpm-profile-ptp-specification/ |
| |
| |
| .. _ACPI specification: |
| https://trustedcomputinggroup.org/tcg-acpi-specification/ |
| |
| .. _PPI specification: |
| https://trustedcomputinggroup.org/resource/tcg-physical-presence-interface-specification/ |
| |
| .. _SWTPM protocol: |
| https://github.com/stefanberger/swtpm/blob/master/man/man3/swtpm_ioctls.pod |