|
Joerg Roedel |
573e89 |
From: Liu Yi L <yi.l.liu@intel.com>
|
|
Joerg Roedel |
573e89 |
Date: Thu, 7 Jan 2021 00:03:57 +0800
|
|
Joerg Roedel |
573e89 |
Subject: iommu/vt-d: Fix ineffective devTLB invalidation for subdevices
|
|
Joerg Roedel |
573e89 |
Git-commit: 7c29ada5e70083805bc3a68daa23441df421fbee
|
|
Joerg Roedel |
573e89 |
Patch-mainline: v5.11-rc3
|
|
Joerg Roedel |
573e89 |
References: bsc#1183284
|
|
Joerg Roedel |
573e89 |
|
|
Joerg Roedel |
573e89 |
iommu_flush_dev_iotlb() is called to invalidate caches on a device but
|
|
Joerg Roedel |
573e89 |
only loops over the devices which are fully-attached to the domain. For
|
|
Joerg Roedel |
573e89 |
sub-devices, this is ineffective and can result in invalid caching
|
|
Joerg Roedel |
573e89 |
entries left on the device.
|
|
Joerg Roedel |
573e89 |
|
|
Joerg Roedel |
573e89 |
Fix the missing invalidation by adding a loop over the subdevices and
|
|
Joerg Roedel |
573e89 |
ensuring that 'domain->has_iotlb_device' is updated when attaching to
|
|
Joerg Roedel |
573e89 |
subdevices.
|
|
Joerg Roedel |
573e89 |
|
|
Joerg Roedel |
573e89 |
Fixes: 67b8e02b5e76 ("iommu/vt-d: Aux-domain specific domain attach/detach")
|
|
Joerg Roedel |
573e89 |
Signed-off-by: Liu Yi L <yi.l.liu@intel.com>
|
|
Joerg Roedel |
573e89 |
Acked-by: Lu Baolu <baolu.lu@linux.intel.com>
|
|
Joerg Roedel |
573e89 |
Link: https://lore.kernel.org/r/1609949037-25291-4-git-send-email-yi.l.liu@intel.com
|
|
Joerg Roedel |
573e89 |
Signed-off-by: Will Deacon <will@kernel.org>
|
|
Joerg Roedel |
573e89 |
Acked-by: Joerg Roedel <jroedel@suse.de>
|
|
Joerg Roedel |
573e89 |
---
|
|
Joerg Roedel |
573e89 |
drivers/iommu/intel-iommu.c | 53 ++++++++++++++++++++++++++++++--------------
|
|
Joerg Roedel |
573e89 |
1 file changed, 37 insertions(+), 16 deletions(-)
|
|
Joerg Roedel |
573e89 |
|
|
Joerg Roedel |
573e89 |
--- a/drivers/iommu/intel-iommu.c
|
|
Joerg Roedel |
573e89 |
+++ b/drivers/iommu/intel-iommu.c
|
|
Joerg Roedel |
6ad821 |
@@ -723,6 +723,8 @@ static int domain_update_device_node(str
|
|
Joerg Roedel |
6ad821 |
return nid;
|
|
Joerg Roedel |
573e89 |
}
|
|
Joerg Roedel |
573e89 |
|
|
Joerg Roedel |
573e89 |
+static void domain_update_iotlb(struct dmar_domain *domain);
|
|
Joerg Roedel |
573e89 |
+
|
|
Joerg Roedel |
573e89 |
/* Some capabilities may be different across iommus */
|
|
Joerg Roedel |
573e89 |
static void domain_update_iommu_cap(struct dmar_domain *domain)
|
|
Joerg Roedel |
573e89 |
{
|
|
Joerg Roedel |
6ad821 |
@@ -736,6 +738,8 @@ static void domain_update_iommu_cap(stru
|
|
Joerg Roedel |
6ad821 |
*/
|
|
Joerg Roedel |
6ad821 |
if (domain->nid == NUMA_NO_NODE)
|
|
Joerg Roedel |
6ad821 |
domain->nid = domain_update_device_node(domain);
|
|
Joerg Roedel |
573e89 |
+
|
|
Joerg Roedel |
573e89 |
+ domain_update_iotlb(domain);
|
|
Joerg Roedel |
573e89 |
}
|
|
Joerg Roedel |
573e89 |
|
|
Joerg Roedel |
573e89 |
struct context_entry *iommu_context_addr(struct intel_iommu *iommu, u8 bus,
|
|
Joerg Roedel |
6ad821 |
@@ -1418,17 +1422,22 @@ static void domain_update_iotlb(struct d
|
|
Joerg Roedel |
573e89 |
|
|
Joerg Roedel |
573e89 |
assert_spin_locked(&device_domain_lock);
|
|
Joerg Roedel |
573e89 |
|
|
Joerg Roedel |
573e89 |
- list_for_each_entry(info, &domain->devices, link) {
|
|
Joerg Roedel |
573e89 |
- struct pci_dev *pdev;
|
|
Joerg Roedel |
573e89 |
-
|
|
Joerg Roedel |
573e89 |
- if (!info->dev || !dev_is_pci(info->dev))
|
|
Joerg Roedel |
573e89 |
- continue;
|
|
Joerg Roedel |
573e89 |
-
|
|
Joerg Roedel |
573e89 |
- pdev = to_pci_dev(info->dev);
|
|
Joerg Roedel |
573e89 |
- if (pdev->ats_enabled) {
|
|
Joerg Roedel |
573e89 |
+ list_for_each_entry(info, &domain->devices, link)
|
|
Joerg Roedel |
573e89 |
+ if (info->ats_enabled) {
|
|
Joerg Roedel |
573e89 |
has_iotlb_device = true;
|
|
Joerg Roedel |
573e89 |
break;
|
|
Joerg Roedel |
573e89 |
}
|
|
Joerg Roedel |
573e89 |
+
|
|
Joerg Roedel |
573e89 |
+ if (!has_iotlb_device) {
|
|
Joerg Roedel |
573e89 |
+ struct subdev_domain_info *sinfo;
|
|
Joerg Roedel |
573e89 |
+
|
|
Joerg Roedel |
573e89 |
+ list_for_each_entry(sinfo, &domain->subdevices, link_domain) {
|
|
Joerg Roedel |
573e89 |
+ info = get_domain_info(sinfo->pdev);
|
|
Joerg Roedel |
573e89 |
+ if (info && info->ats_enabled) {
|
|
Joerg Roedel |
573e89 |
+ has_iotlb_device = true;
|
|
Joerg Roedel |
573e89 |
+ break;
|
|
Joerg Roedel |
573e89 |
+ }
|
|
Joerg Roedel |
573e89 |
+ }
|
|
Joerg Roedel |
573e89 |
}
|
|
Joerg Roedel |
573e89 |
|
|
Joerg Roedel |
573e89 |
domain->has_iotlb_device = has_iotlb_device;
|
|
Joerg Roedel |
6ad821 |
@@ -1510,25 +1519,37 @@ static void iommu_disable_dev_iotlb(stru
|
|
Joerg Roedel |
573e89 |
#endif
|
|
Joerg Roedel |
573e89 |
}
|
|
Joerg Roedel |
573e89 |
|
|
Joerg Roedel |
573e89 |
+static void __iommu_flush_dev_iotlb(struct device_domain_info *info,
|
|
Joerg Roedel |
573e89 |
+ u64 addr, unsigned int mask)
|
|
Joerg Roedel |
573e89 |
+{
|
|
Joerg Roedel |
573e89 |
+ u16 sid, qdep;
|
|
Joerg Roedel |
573e89 |
+
|
|
Joerg Roedel |
573e89 |
+ if (!info || !info->ats_enabled)
|
|
Joerg Roedel |
573e89 |
+ return;
|
|
Joerg Roedel |
573e89 |
+
|
|
Joerg Roedel |
573e89 |
+ sid = info->bus << 8 | info->devfn;
|
|
Joerg Roedel |
573e89 |
+ qdep = info->ats_qdep;
|
|
Joerg Roedel |
573e89 |
+ qi_flush_dev_iotlb(info->iommu, sid, info->pfsid,
|
|
Joerg Roedel |
573e89 |
+ qdep, addr, mask);
|
|
Joerg Roedel |
573e89 |
+}
|
|
Joerg Roedel |
573e89 |
+
|
|
Joerg Roedel |
573e89 |
static void iommu_flush_dev_iotlb(struct dmar_domain *domain,
|
|
Joerg Roedel |
573e89 |
u64 addr, unsigned mask)
|
|
Joerg Roedel |
573e89 |
{
|
|
Joerg Roedel |
573e89 |
- u16 sid, qdep;
|
|
Joerg Roedel |
573e89 |
unsigned long flags;
|
|
Joerg Roedel |
573e89 |
struct device_domain_info *info;
|
|
Joerg Roedel |
573e89 |
+ struct subdev_domain_info *sinfo;
|
|
Joerg Roedel |
573e89 |
|
|
Joerg Roedel |
573e89 |
if (!domain->has_iotlb_device)
|
|
Joerg Roedel |
573e89 |
return;
|
|
Joerg Roedel |
573e89 |
|
|
Joerg Roedel |
573e89 |
spin_lock_irqsave(&device_domain_lock, flags);
|
|
Joerg Roedel |
573e89 |
- list_for_each_entry(info, &domain->devices, link) {
|
|
Joerg Roedel |
573e89 |
- if (!info->ats_enabled)
|
|
Joerg Roedel |
573e89 |
- continue;
|
|
Joerg Roedel |
573e89 |
+ list_for_each_entry(info, &domain->devices, link)
|
|
Joerg Roedel |
573e89 |
+ __iommu_flush_dev_iotlb(info, addr, mask);
|
|
Joerg Roedel |
573e89 |
|
|
Joerg Roedel |
573e89 |
- sid = info->bus << 8 | info->devfn;
|
|
Joerg Roedel |
573e89 |
- qdep = info->ats_qdep;
|
|
Joerg Roedel |
573e89 |
- qi_flush_dev_iotlb(info->iommu, sid, info->pfsid,
|
|
Joerg Roedel |
573e89 |
- qdep, addr, mask);
|
|
Joerg Roedel |
573e89 |
+ list_for_each_entry(sinfo, &domain->subdevices, link_domain) {
|
|
Joerg Roedel |
573e89 |
+ info = get_domain_info(sinfo->pdev);
|
|
Joerg Roedel |
573e89 |
+ __iommu_flush_dev_iotlb(info, addr, mask);
|
|
Joerg Roedel |
573e89 |
}
|
|
Joerg Roedel |
573e89 |
spin_unlock_irqrestore(&device_domain_lock, flags);
|
|
Joerg Roedel |
573e89 |
}
|