Commit 0d8227b6 authored by Jason Gunthorpe's avatar Jason Gunthorpe

vfio: Use IOMMU_CAP_ENFORCE_CACHE_COHERENCY for vfio_file_enforced_coherent()

iommufd doesn't establish the iommu_domains until after the device FD is
opened, even if the container has been set. This design is part of moving
away from the group centric iommu APIs.

This is fine, except that the normal sequence of establishing the kvm
wbinvd won't work:

   group = open("/dev/vfio/XX")
   ioctl(group, VFIO_GROUP_SET_CONTAINER)
   ioctl(kvm, KVM_DEV_VFIO_GROUP_ADD)
   ioctl(group, VFIO_GROUP_GET_DEVICE_FD)

As the domains don't start existing until GET_DEVICE_FD. Further,
GET_DEVICE_FD requires that KVM_DEV_VFIO_GROUP_ADD already be done as that
is what sets the group->kvm and thus device->kvm for the driver to use
during open.

Now that we have device centric cap ops and the new
IOMMU_CAP_ENFORCE_CACHE_COHERENCY we know what the iommu_domain will be
capable of without having to create it. Use this to compute
vfio_file_enforced_coherent() and resolve the ordering problems.

VFIO always tries to upgrade domains to enforce cache coherency, it never
attaches a device that supports enforce cache coherency to a less capable
domain, so the cap test is a sufficient proxy for the ultimate
outcome. iommufd also ensures that devices that set the cap will be
connected to enforcing domains.

Link: https://lore.kernel.org/r/4-v4-42cd2eb0e3eb+335a-vfio_iommufd_jgg@nvidia.comReviewed-by: default avatarKevin Tian <kevin.tian@intel.com>
Reviewed-by: default avatarAlex Williamson <alex.williamson@redhat.com>
Tested-by: default avatarAlex Williamson <alex.williamson@redhat.com>
Tested-by: default avatarNicolin Chen <nicolinc@nvidia.com>
Tested-by: default avatarYi Liu <yi.l.liu@intel.com>
Tested-by: default avatarLixiao Yang <lixiao.yang@intel.com>
Tested-by: default avatarMatthew Rosato <mjrosato@linux.ibm.com>
Tested-by: default avatarYu He <yu.he@intel.com>
Signed-off-by: default avatarJason Gunthorpe <jgg@nvidia.com>
parent 04f930c3
...@@ -188,7 +188,8 @@ void vfio_device_container_unregister(struct vfio_device *device) ...@@ -188,7 +188,8 @@ void vfio_device_container_unregister(struct vfio_device *device)
device->group->container->iommu_data, device); device->group->container->iommu_data, device);
} }
long vfio_container_ioctl_check_extension(struct vfio_container *container, static long
vfio_container_ioctl_check_extension(struct vfio_container *container,
unsigned long arg) unsigned long arg)
{ {
struct vfio_iommu_driver *driver; struct vfio_iommu_driver *driver;
......
...@@ -119,8 +119,6 @@ int vfio_container_attach_group(struct vfio_container *container, ...@@ -119,8 +119,6 @@ int vfio_container_attach_group(struct vfio_container *container,
void vfio_group_detach_container(struct vfio_group *group); void vfio_group_detach_container(struct vfio_group *group);
void vfio_device_container_register(struct vfio_device *device); void vfio_device_container_register(struct vfio_device *device);
void vfio_device_container_unregister(struct vfio_device *device); void vfio_device_container_unregister(struct vfio_device *device);
long vfio_container_ioctl_check_extension(struct vfio_container *container,
unsigned long arg);
int __init vfio_container_init(void); int __init vfio_container_init(void);
void vfio_container_cleanup(void); void vfio_container_cleanup(void);
......
...@@ -1622,24 +1622,27 @@ EXPORT_SYMBOL_GPL(vfio_file_is_group); ...@@ -1622,24 +1622,27 @@ EXPORT_SYMBOL_GPL(vfio_file_is_group);
bool vfio_file_enforced_coherent(struct file *file) bool vfio_file_enforced_coherent(struct file *file)
{ {
struct vfio_group *group = file->private_data; struct vfio_group *group = file->private_data;
bool ret; struct vfio_device *device;
bool ret = true;
if (!vfio_file_is_group(file)) if (!vfio_file_is_group(file))
return true; return true;
mutex_lock(&group->group_lock);
if (group->container) {
ret = vfio_container_ioctl_check_extension(group->container,
VFIO_DMA_CC_IOMMU);
} else {
/* /*
* Since the coherency state is determined only once a container * If the device does not have IOMMU_CAP_ENFORCE_CACHE_COHERENCY then
* is attached the user must do so before they can prove they * any domain later attached to it will also not support it. If the cap
* have permission. * is set then the iommu_domain eventually attached to the device/group
* must use a domain with enforce_cache_coherency().
*/ */
ret = true; mutex_lock(&group->device_lock);
list_for_each_entry(device, &group->device_list, group_next) {
if (!device_iommu_capable(device->dev,
IOMMU_CAP_ENFORCE_CACHE_COHERENCY)) {
ret = false;
break;
} }
mutex_unlock(&group->group_lock); }
mutex_unlock(&group->device_lock);
return ret; return ret;
} }
EXPORT_SYMBOL_GPL(vfio_file_enforced_coherent); EXPORT_SYMBOL_GPL(vfio_file_enforced_coherent);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment