]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
iommu/sva: Fix crash in iommu_sva_unbind_device()
authorLizhi Hou <lizhi.hou@amd.com>
Thu, 5 Mar 2026 06:18:42 +0000 (22:18 -0800)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Wed, 25 Mar 2026 10:13:31 +0000 (11:13 +0100)
[ Upstream commit 06e14c36e20b48171df13d51b89fe67c594ed07a ]

domain->mm->iommu_mm can be freed by iommu_domain_free():
  iommu_domain_free()
    mmdrop()
      __mmdrop()
        mm_pasid_drop()
After iommu_domain_free() returns, accessing domain->mm->iommu_mm may
dereference a freed mm structure, leading to a crash.

Fix this by moving the code that accesses domain->mm->iommu_mm to before
the call to iommu_domain_free().

Fixes: e37d5a2d60a3 ("iommu/sva: invalidate stale IOTLB entries for kernel address space")
Signed-off-by: Lizhi Hou <lizhi.hou@amd.com>
Reviewed-by: Jason Gunthorpe <jgg@nvidia.com>
Reviewed-by: Yi Liu <yi.l.liu@intel.com>
Reviewed-by: Vasant Hegde <vasant.hegde@amd.com>
Reviewed-by: Lu Baolu <baolu.lu@linux.intel.com>
Signed-off-by: Joerg Roedel <joerg.roedel@amd.com>
Signed-off-by: Sasha Levin <sashal@kernel.org>
drivers/iommu/iommu-sva.c

index e1e63c2be82b2a4f37da572f066186e2b4374db3..fd735aaae9e3f0841f5ce8b1f87cd85d6f5f6d22 100644 (file)
@@ -182,13 +182,13 @@ void iommu_sva_unbind_device(struct iommu_sva *handle)
        iommu_detach_device_pasid(domain, dev, iommu_mm->pasid);
        if (--domain->users == 0) {
                list_del(&domain->next);
-               iommu_domain_free(domain);
-       }
+               if (list_empty(&iommu_mm->sva_domains)) {
+                       list_del(&iommu_mm->mm_list_elm);
+                       if (list_empty(&iommu_sva_mms))
+                               iommu_sva_present = false;
+               }
 
-       if (list_empty(&iommu_mm->sva_domains)) {
-               list_del(&iommu_mm->mm_list_elm);
-               if (list_empty(&iommu_sva_mms))
-                       iommu_sva_present = false;
+               iommu_domain_free(domain);
        }
 
        mutex_unlock(&iommu_sva_lock);