2014-09-17 22:56:52 +04:00
#
# Support for parsing libvirt's domcapabilities XML
#
# Copyright 2014 Red Hat, Inc.
#
2018-04-04 16:35:41 +03:00
# This work is licensed under the GNU GPLv2 or later.
2018-03-20 22:00:02 +03:00
# See the COPYING file in the top-level directory.
2014-09-17 22:56:52 +04:00
2015-02-18 23:16:48 +03:00
import re
2019-03-15 11:49:56 +03:00
import xml . etree . ElementTree as ET
2015-02-18 23:16:48 +03:00
2019-03-15 11:49:56 +03:00
import libvirt
from . domain import DomainCpu
2019-06-17 04:12:39 +03:00
from . logger import log
2015-04-22 21:44:52 +03:00
from . xmlbuilder import XMLBuilder , XMLChildProperty , XMLProperty
2014-09-17 22:56:52 +04:00
2018-04-03 18:03:32 +03:00
########################################
# Genering <enum> and <value> handling #
########################################
2014-09-17 22:56:52 +04:00
class _Value ( XMLBuilder ) :
2018-03-21 17:53:34 +03:00
XML_NAME = " value "
2014-09-17 22:56:52 +04:00
value = XMLProperty ( " . " )
class _HasValues ( XMLBuilder ) :
values = XMLChildProperty ( _Value )
def get_values ( self ) :
return [ v . value for v in self . values ]
2022-02-17 22:00:22 +03:00
def has_value ( self , val ) :
return val in self . get_values ( )
2014-09-17 22:56:52 +04:00
class _Enum ( _HasValues ) :
2018-03-21 17:53:34 +03:00
XML_NAME = " enum "
2014-09-17 22:56:52 +04:00
name = XMLProperty ( " ./@name " )
class _CapsBlock ( _HasValues ) :
supported = XMLProperty ( " ./@supported " , is_yesno = True )
2022-02-17 22:51:04 +03:00
_supported_present = XMLProperty ( " ./@supported " )
2014-09-17 22:56:52 +04:00
enums = XMLChildProperty ( _Enum )
2022-02-17 22:51:04 +03:00
@property
def present ( self ) :
return self . _supported_present is not None
2014-09-17 22:56:52 +04:00
def enum_names ( self ) :
return [ e . name for e in self . enums ]
2022-02-03 21:11:20 +03:00
def has_enum ( self , name ) :
return name in self . enum_names ( )
2014-09-17 22:56:52 +04:00
def get_enum ( self , name ) :
2020-09-28 20:44:34 +03:00
for enum in self . enums :
if enum . name == name :
return enum
# Didn't find a match. Could be talking to older libvirt, or
# driver with incomplete info. Return a stub enum
return _Enum ( self . conn )
2014-09-17 22:56:52 +04:00
def _make_capsblock ( xml_root_name ) :
2018-04-03 18:03:32 +03:00
"""
Build a class object representing a list of < enum > in the XML . For
example , domcapabilities may have a block like :
< graphics supported = ' yes ' >
< enum name = ' type ' >
< value > sdl < / value >
< value > vnc < / value >
< value > spice < / value >
< / enum >
< / graphics >
To build a class that tracks that whole < graphics > block , call this
like _make_capsblock ( " graphics " )
"""
2014-09-17 22:56:52 +04:00
class TmpClass ( _CapsBlock ) :
pass
2018-03-21 17:53:34 +03:00
setattr ( TmpClass , " XML_NAME " , xml_root_name )
2014-09-17 22:56:52 +04:00
return TmpClass
2019-06-11 18:41:59 +03:00
################################
# SEV launch security handling #
################################
class _SEV ( XMLBuilder ) :
XML_NAME = " sev "
2019-06-11 18:42:00 +03:00
supported = XMLProperty ( " ./@supported " , is_yesno = True )
2022-08-03 15:47:02 +03:00
maxESGuests = XMLProperty ( " ./maxESGuests " )
2019-06-11 18:41:59 +03:00
2018-04-03 18:03:32 +03:00
#############################
# Misc toplevel XML classes #
#############################
2014-09-17 22:56:52 +04:00
class _OS ( _CapsBlock ) :
2018-03-21 17:53:34 +03:00
XML_NAME = " os "
2014-09-17 22:56:52 +04:00
loader = XMLChildProperty ( _make_capsblock ( " loader " ) , is_single = True )
class _Devices ( _CapsBlock ) :
2018-03-21 17:53:34 +03:00
XML_NAME = " devices "
2014-09-17 22:56:52 +04:00
hostdev = XMLChildProperty ( _make_capsblock ( " hostdev " ) , is_single = True )
disk = XMLChildProperty ( _make_capsblock ( " disk " ) , is_single = True )
2019-10-03 11:44:52 +03:00
video = XMLChildProperty ( _make_capsblock ( " video " ) , is_single = True )
2021-06-08 17:30:30 +03:00
graphics = XMLChildProperty ( _make_capsblock ( " graphics " ) , is_single = True )
2021-11-24 19:05:16 +03:00
tpm = XMLChildProperty ( _make_capsblock ( " tpm " ) , is_single = True )
2021-08-01 15:36:36 +03:00
filesystem = XMLChildProperty ( _make_capsblock ( " filesystem " ) , is_single = True )
2022-11-10 10:57:08 +03:00
redirdev = XMLChildProperty ( _make_capsblock ( " redirdev " ) , is_single = True )
2022-11-10 10:57:24 +03:00
channel = XMLChildProperty ( _make_capsblock ( " channel " ) , is_single = True )
2024-12-03 08:32:15 +03:00
panic = XMLChildProperty ( _make_capsblock ( " panic " ) , is_single = True )
2014-09-17 22:56:52 +04:00
2016-06-10 03:22:25 +03:00
class _Features ( _CapsBlock ) :
2018-03-21 17:53:34 +03:00
XML_NAME = " features "
2016-06-10 03:22:25 +03:00
gic = XMLChildProperty ( _make_capsblock ( " gic " ) , is_single = True )
2019-06-11 18:41:59 +03:00
sev = XMLChildProperty ( _SEV , is_single = True )
2024-06-11 23:45:35 +03:00
hyperv = XMLChildProperty ( _make_capsblock ( " hyperv " ) , is_single = True )
2016-06-10 03:22:25 +03:00
2021-08-01 15:36:38 +03:00
class _MemoryBacking ( _CapsBlock ) :
XML_NAME = " memoryBacking "
2018-04-03 18:03:32 +03:00
###############
# CPU classes #
###############
2018-03-28 22:45:29 +03:00
class _CPUModel ( XMLBuilder ) :
XML_NAME = " model "
model = XMLProperty ( " . " )
2019-03-14 16:10:26 +03:00
usable = XMLProperty ( " ./@usable " )
2018-10-14 00:47:31 +03:00
fallback = XMLProperty ( " ./@fallback " )
2022-02-16 20:13:16 +03:00
class _CPUMode ( _CapsBlock ) :
2018-03-28 22:45:29 +03:00
XML_NAME = " mode "
name = XMLProperty ( " ./@name " )
2018-10-14 00:47:31 +03:00
models = XMLChildProperty ( _CPUModel )
2018-03-28 22:45:29 +03:00
def get_model ( self , name ) :
for model in self . models :
if model . model == name :
return model
2018-04-03 18:03:32 +03:00
2018-03-28 22:45:29 +03:00
class _CPU ( XMLBuilder ) :
XML_NAME = " cpu "
modes = XMLChildProperty ( _CPUMode )
def get_mode ( self , name ) :
for mode in self . modes :
if mode . name == name :
return mode
2022-02-17 22:15:46 +03:00
#############################
# CPU flags/baseline helpers#
#############################
def _convert_mode_to_cpu ( xml , arch ) :
root = ET . fromstring ( xml )
root . tag = " cpu "
root . attrib = { }
aelement = ET . SubElement ( root , " arch " )
aelement . text = arch
return ET . tostring ( root , encoding = " unicode " )
def _get_expanded_cpu ( domcaps , mode ) :
cpuXML = _convert_mode_to_cpu ( mode . get_xml ( ) , domcaps . arch )
log . debug ( " Generated CPU XML for security flag baseline: \n %s " , cpuXML )
try :
expandedXML = domcaps . conn . baselineHypervisorCPU (
domcaps . path , domcaps . arch ,
domcaps . machine , domcaps . domain , [ cpuXML ] ,
libvirt . VIR_CONNECT_BASELINE_CPU_EXPAND_FEATURES )
except ( libvirt . libvirtError , AttributeError ) :
expandedXML = domcaps . conn . baselineCPU ( [ cpuXML ] ,
libvirt . VIR_CONNECT_BASELINE_CPU_EXPAND_FEATURES )
return DomainCpu ( domcaps . conn , expandedXML )
def _lookup_cpu_security_features ( domcaps ) :
ret = [ ]
sec_features = [
' spec-ctrl ' ,
' ssbd ' ,
' ibpb ' ,
' virt-ssbd ' ,
' md-clear ' ]
for m in domcaps . cpu . modes :
if m . name != " host-model " or not m . supported :
continue # pragma: no cover
try :
cpu = _get_expanded_cpu ( domcaps , m )
except libvirt . libvirtError as e : # pragma: no cover
log . warning ( _ ( " Failed to get expanded CPU XML: %s " ) , e )
break
for feature in cpu . features :
if feature . name in sec_features :
ret . append ( feature . name )
log . debug ( " Found host-model security features: %s " , ret )
return ret
2018-04-03 18:03:32 +03:00
#################################
# DomainCapabilities main class #
#################################
2014-09-17 22:56:52 +04:00
class DomainCapabilities ( XMLBuilder ) :
2022-02-17 22:15:46 +03:00
XML_NAME = " domainCapabilities "
os = XMLChildProperty ( _OS , is_single = True )
cpu = XMLChildProperty ( _CPU , is_single = True )
devices = XMLChildProperty ( _Devices , is_single = True )
features = XMLChildProperty ( _Features , is_single = True )
memorybacking = XMLChildProperty ( _MemoryBacking , is_single = True )
arch = XMLProperty ( " ./arch " )
domain = XMLProperty ( " ./domain " )
machine = XMLProperty ( " ./machine " )
path = XMLProperty ( " ./path " )
################
# Init helpers #
################
2015-02-18 23:16:48 +03:00
@staticmethod
2015-02-22 18:01:02 +03:00
def build_from_params ( conn , emulator , arch , machine , hvtype ) :
2015-04-08 16:53:30 +03:00
xml = None
2019-06-07 23:06:52 +03:00
if conn . support . conn_domain_capabilities ( ) :
2015-04-08 16:53:30 +03:00
try :
xml = conn . getDomainCapabilities ( emulator , arch ,
machine , hvtype )
2022-06-28 23:56:09 +03:00
log . debug ( " Fetched domain capabilities for ( %s , %s , %s , %s ): %s " ,
emulator , arch , machine , hvtype , xml )
2020-01-27 18:06:18 +03:00
except Exception : # pragma: no cover
2019-06-17 04:12:39 +03:00
log . debug ( " Error fetching domcapabilities XML " ,
2015-04-08 16:53:30 +03:00
exc_info = True )
if not xml :
2015-02-18 23:16:48 +03:00
# If not supported, just use a stub object
2015-02-22 18:01:02 +03:00
return DomainCapabilities ( conn )
return DomainCapabilities ( conn , parsexml = xml )
2015-02-18 23:16:48 +03:00
2015-02-22 18:01:02 +03:00
@staticmethod
def build_from_guest ( guest ) :
return DomainCapabilities . build_from_params ( guest . conn ,
2015-02-18 23:16:48 +03:00
guest . emulator , guest . os . arch , guest . os . machine , guest . type )
2022-02-17 22:15:46 +03:00
#########################
# UEFI/firmware methods #
#########################
2015-02-18 23:16:48 +03:00
# Mapping of UEFI binary names to their associated architectures. We
# only use this info to do things automagically for the user, it shouldn't
# validate anything the user explicitly enters.
_uefi_arch_patterns = {
2018-08-08 18:55:29 +03:00
" i686 " : [
2019-12-14 14:48:32 +03:00
r " .*edk2-i386-.* \ .fd " , # upstream qemu
2018-09-29 20:59:19 +03:00
r " .*ovmf-ia32.* " , # fedora, gerd's firmware repo
2018-08-08 18:55:29 +03:00
] ,
2015-02-18 23:16:48 +03:00
" x86_64 " : [
2019-12-14 14:48:32 +03:00
r " .*edk2-x86_64-.* \ .fd " , # upstream qemu
2018-09-29 20:59:19 +03:00
r " .*OVMF_CODE \ .fd " , # RHEL
r " .*ovmf-x64/OVMF.* \ .fd " , # gerd's firmware repo
r " .*ovmf-x86_64-.* " , # SUSE
r " .*ovmf.* " , " .*OVMF.* " , # generic attempt at a catchall
2015-02-18 23:16:48 +03:00
] ,
" aarch64 " : [
2018-09-29 20:59:19 +03:00
r " .*AAVMF_CODE \ .fd " , # RHEL
2024-09-09 15:52:39 +03:00
r " .*aarch64/QEMU_EFI.* " , # fedora, gerd's firmware repo
2018-09-29 20:59:19 +03:00
r " .*aarch64.* " , # generic attempt at a catchall
2019-12-14 14:48:32 +03:00
r " .*edk2-aarch64-code \ .fd " , # upstream qemu
2015-02-18 23:16:48 +03:00
] ,
2018-08-08 18:55:29 +03:00
" armv7l " : [
2020-11-03 20:48:12 +03:00
r " .*AAVMF32_CODE \ .fd " , # Debian qemu-efi-arm package
2018-09-29 20:59:19 +03:00
r " .*arm/QEMU_EFI.* " , # fedora, gerd's firmware repo
2019-12-14 14:48:32 +03:00
r " .*edk2-arm-code \ .fd " # upstream qemu
2018-08-08 18:55:29 +03:00
] ,
2024-09-09 18:49:28 +03:00
" riscv64 " : [
r " .*RISCV_VIRT_CODE \ ..* " , # Fedora, Debian
r " .*riscv64.* " , # generic attempt at a catchall
] ,
2024-09-09 18:49:38 +03:00
" loongarch64 " : [
r " .*loongarch64/QEMU_CODE \ ..* " , # Fedora
r " .*loongarch64.* " , # generic attempt at a catchall
] ,
2015-02-18 23:16:48 +03:00
}
2015-02-22 19:02:55 +03:00
def find_uefi_path_for_arch ( self ) :
2015-02-18 23:16:48 +03:00
"""
Search the loader paths for one that matches the passed arch
"""
2015-02-22 19:02:55 +03:00
if not self . arch_can_uefi ( ) :
2020-01-27 18:06:18 +03:00
return # pragma: no cover
2015-02-18 23:16:48 +03:00
2021-02-15 21:36:40 +03:00
firmware_files = [ f . value for f in self . os . loader . values ]
2021-02-11 18:41:12 +03:00
if self . conn . is_bhyve ( ) :
for firmware_file in firmware_files :
if ' BHYVE_UEFI.fd ' in firmware_file :
return firmware_file
2021-02-15 21:36:40 +03:00
return ( firmware_files and
firmware_files [ 0 ] or None ) # pragma: no cover
2021-02-11 18:41:12 +03:00
2015-02-22 19:02:55 +03:00
patterns = self . _uefi_arch_patterns . get ( self . arch )
2015-02-18 23:16:48 +03:00
for pattern in patterns :
2021-02-15 21:36:40 +03:00
for path in firmware_files :
2015-02-18 23:16:48 +03:00
if re . match ( pattern , path ) :
return path
2015-02-22 19:13:21 +03:00
def label_for_firmware_path ( self , path ) :
"""
Return a pretty label for passed path , based on if we know
about it or not
"""
if not path :
if self . arch in [ " i686 " , " x86_64 " ] :
return _ ( " BIOS " )
2022-01-26 21:08:55 +03:00
return _ ( " Default " )
2015-02-22 19:13:21 +03:00
for arch , patterns in self . _uefi_arch_patterns . items ( ) :
for pattern in patterns :
if re . match ( pattern , path ) :
return ( _ ( " UEFI %(arch)s : %(path)s " ) %
{ " arch " : arch , " path " : path } )
return _ ( " Custom: %(path)s " % { " path " : path } )
2015-02-22 19:02:55 +03:00
def arch_can_uefi ( self ) :
2015-02-18 23:16:48 +03:00
"""
Return True if we know how to setup UEFI for the passed arch
"""
2022-12-13 18:51:14 +03:00
return self . arch in self . _uefi_arch_patterns
2015-02-18 23:16:48 +03:00
2022-01-26 19:59:51 +03:00
def supports_uefi_loader ( self ) :
2015-02-18 23:16:48 +03:00
"""
2022-01-26 19:59:51 +03:00
Return True if libvirt advertises support for UEFI loader
2015-02-18 23:16:48 +03:00
"""
2022-02-17 22:00:22 +03:00
return self . os . loader . get_enum ( " readonly " ) . has_value ( " yes " )
2015-02-18 23:16:48 +03:00
2022-01-26 19:59:51 +03:00
def supports_firmware_efi ( self ) :
2022-02-17 22:00:22 +03:00
return self . os . get_enum ( " firmware " ) . has_value ( " efi " )
2022-01-26 19:59:51 +03:00
2022-02-17 22:15:46 +03:00
#######################
# CPU support methods #
#######################
2018-10-04 19:23:32 +03:00
def supports_safe_host_model ( self ) :
"""
Return True if domcaps reports support for cpu mode = host - model .
2020-06-03 16:14:31 +03:00
host - model in fact predates this support , however it wasn ' t
2019-03-13 16:18:14 +03:00
general purpose safe prior to domcaps advertisement .
2018-10-04 19:23:32 +03:00
"""
2022-02-16 20:13:16 +03:00
m = self . cpu . get_mode ( " host-model " )
return ( m and m . supported and
m . models [ 0 ] . fallback == " forbid " )
def supports_safe_host_passthrough ( self ) :
"""
Return True if host - passthrough is safe enough to use by default .
We limit this to domcaps new enough to report whether host - passthrough
is migratable or not , which also means libvirt is about new enough
to not taint the VM for using host - passthrough
"""
m = self . cpu . get_mode ( " host-passthrough " )
return ( m and m . supported and
" on " in m . get_enum ( " hostPassthroughMigratable " ) . get_values ( ) )
2018-10-04 19:23:32 +03:00
2024-12-07 01:02:29 +03:00
def supports_maximum_cpu_mode ( self ) :
m = self . cpu . get_mode ( " maximum " )
return ( m and m . supported )
2019-03-15 11:49:29 +03:00
def get_cpu_models ( self ) :
models = [ ]
for m in self . cpu . modes :
if m . name == " custom " and m . supported :
for model in m . models :
if model . usable != " no " :
models . append ( model . model )
return models
2020-01-27 18:06:18 +03:00
_features = None
def get_cpu_security_features ( self ) :
if self . _features is None :
2022-02-17 22:15:46 +03:00
self . _features = _lookup_cpu_security_features ( self ) or [ ]
2019-04-03 16:17:08 +03:00
return self . _features
2019-03-15 11:49:56 +03:00
2020-01-27 18:06:18 +03:00
2022-02-17 22:15:46 +03:00
########################
# Misc support methods #
########################
2022-08-03 15:47:02 +03:00
def supports_sev_launch_security ( self , check_es = False ) :
2019-06-11 18:42:00 +03:00
"""
Returns False if either libvirt doesn ' t advertise support for SEV at
all ( < libvirt - 4.5 .0 ) or if it explicitly advertises it as unsupported
on the platform
"""
2022-08-03 15:47:02 +03:00
if check_es :
return bool ( self . features . sev . supported and
self . features . sev . maxESGuests )
2019-06-11 18:42:00 +03:00
return bool ( self . features . sev . supported )
2015-02-18 23:16:48 +03:00
2019-10-03 11:47:15 +03:00
def supports_video_bochs ( self ) :
"""
Returns False if either libvirt or qemu do not have support to bochs
video type .
"""
2022-02-17 22:00:22 +03:00
return self . devices . video . get_enum ( " modelType " ) . has_value ( " bochs " )
2019-10-03 11:47:15 +03:00
2022-02-03 21:11:20 +03:00
def supports_video_qxl ( self ) :
if not self . devices . video . has_enum ( " modelType " ) :
# qxl long predates modelType in domcaps, so if it is missing,
# use spice support as a rough value
return self . supports_graphics_spice ( )
2022-02-17 22:00:22 +03:00
return self . devices . video . get_enum ( " modelType " ) . has_value ( " qxl " )
2022-02-03 21:11:20 +03:00
def supports_video_virtio ( self ) :
2022-02-17 22:00:22 +03:00
return self . devices . video . get_enum ( " modelType " ) . has_value ( " virtio " )
2022-02-03 21:11:20 +03:00
2021-11-24 19:05:16 +03:00
def supports_tpm_emulator ( self ) :
"""
Returns False if either libvirt or qemu do not have support for
emulating a TPM .
"""
models = self . devices . tpm . get_enum ( " model " ) . get_values ( )
backends = self . devices . tpm . get_enum ( " backendModel " ) . get_values ( )
2022-06-17 20:04:05 +03:00
if self . arch == " armv7l " and models == [ " tpm-tis " ] :
# libvirt as of 8.4.0 can advertise armv7l tpm-tis support,
# but then explicitly rejects that config. If we see it,
# assume TPM is not supported
# https://gitlab.com/libvirt/libvirt/-/issues/329
return False
2021-11-24 19:05:16 +03:00
return len ( models ) > 0 and bool ( " emulator " in backends )
2021-06-08 17:30:30 +03:00
def supports_graphics_spice ( self ) :
if not self . devices . graphics . supported :
# domcaps is too old, or the driver doesn't advertise graphics
# support. Use our pre-existing logic
if not self . conn . is_qemu ( ) and not self . conn . is_test ( ) :
return False
return self . conn . caps . host . cpu . arch in [ " i686 " , " x86_64 " ]
2022-02-17 22:00:22 +03:00
return self . devices . graphics . get_enum ( " type " ) . has_value ( " spice " )
2021-06-08 17:30:30 +03:00
2022-11-10 10:57:24 +03:00
def supports_channel_spicevmc ( self ) :
"""
Return False if libvirt explicitly advertises no support for
spice channel
"""
if self . devices . channel . supported is None :
# Follow the original behavior in case of talking to older
# libvirt.
return True
return self . devices . channel . get_enum ( " type " ) . has_value ( " spicevmc " )
2022-11-10 10:57:08 +03:00
def supports_redirdev_usb ( self ) :
"""
Return False if libvirt explicitly advertises no support for
USB redirect
"""
if self . devices . redirdev . supported is None :
# Follow the original behavior in case of talking to older
# libvirt.
return True
return self . devices . redirdev . get_enum ( " bus " ) . has_value ( " usb " )
2021-08-01 15:36:37 +03:00
def supports_filesystem_virtiofs ( self ) :
"""
Return True if libvirt advertises support for virtiofs
"""
2022-02-17 22:00:22 +03:00
return self . devices . filesystem . get_enum (
" driverType " ) . has_value ( " virtiofs " )
2021-08-01 15:36:37 +03:00
2021-08-01 15:36:38 +03:00
def supports_memorybacking_memfd ( self ) :
"""
Return True if libvirt advertises support for memfd memory backend
"""
2022-02-17 22:00:22 +03:00
return self . memorybacking . get_enum ( " sourceType " ) . has_value ( " memfd " )
2024-06-11 23:45:35 +03:00
def supported_hyperv_features ( self ) :
"""
Return list of supported Hyper - V features .
"""
if not self . features . hyperv . supported :
return [ ]
return self . features . hyperv . get_enum ( " features " ) . get_values ( )
2024-12-03 08:32:15 +03:00
def supported_panic_models ( self ) :
"""
Return list of supported panic device models .
"""
return self . devices . panic . get_enum ( " model " ) . get_values ( )