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
 }