提交 33d987cd 编写于 作者: L Lu Baolu 提交者: Zheng Zengkai

iommu/vt-d: Fix lockdep splat in sva bind()/unbind()

stable inclusion
from stable-5.10.9
commit a4c84cd83c158df7f4e9c634cabbac0b9cade8cc
bugzilla: 47457

--------------------------------

[ Upstream commit 420d42f6 ]

Lock(&iommu->lock) without disabling irq causes lockdep warnings.
Acked-by: NXie XiuQi <xiexiuqi@huawei.com>

========================================================
WARNING: possible irq lock inversion dependency detected
5.11.0-rc1+ #828 Not tainted
--------------------------------------------------------
kworker/0:1H/120 just changed the state of lock:
ffffffffad9ea1b8 (device_domain_lock){..-.}-{2:2}, at:
iommu_flush_dev_iotlb.part.0+0x32/0x120
but this lock took another, SOFTIRQ-unsafe lock in the past:
 (&iommu->lock){+.+.}-{2:2}

and interrupts could create inverse lock ordering between them.

other info that might help us debug this:
 Possible interrupt unsafe locking scenario:

       CPU0                    CPU1
       ----                    ----
  lock(&iommu->lock);
                               local_irq_disable();
                               lock(device_domain_lock);
                               lock(&iommu->lock);
  <Interrupt>
    lock(device_domain_lock);

 *** DEADLOCK ***
Signed-off-by: NLu Baolu <baolu.lu@linux.intel.com>
Link: https://lore.kernel.org/r/20201231005323.2178523-5-baolu.lu@linux.intel.comSigned-off-by: NWill Deacon <will@kernel.org>
Signed-off-by: NSasha Levin <sashal@kernel.org>
Signed-off-by: NChen Jun <chenjun102@huawei.com>
上级 fb9f4177
...@@ -281,6 +281,7 @@ int intel_svm_bind_gpasid(struct iommu_domain *domain, struct device *dev, ...@@ -281,6 +281,7 @@ int intel_svm_bind_gpasid(struct iommu_domain *domain, struct device *dev,
struct dmar_domain *dmar_domain; struct dmar_domain *dmar_domain;
struct device_domain_info *info; struct device_domain_info *info;
struct intel_svm *svm = NULL; struct intel_svm *svm = NULL;
unsigned long iflags;
int ret = 0; int ret = 0;
if (WARN_ON(!iommu) || !data) if (WARN_ON(!iommu) || !data)
...@@ -382,12 +383,12 @@ int intel_svm_bind_gpasid(struct iommu_domain *domain, struct device *dev, ...@@ -382,12 +383,12 @@ int intel_svm_bind_gpasid(struct iommu_domain *domain, struct device *dev,
* each bind of a new device even with an existing PASID, we need to * each bind of a new device even with an existing PASID, we need to
* call the nested mode setup function here. * call the nested mode setup function here.
*/ */
spin_lock(&iommu->lock); spin_lock_irqsave(&iommu->lock, iflags);
ret = intel_pasid_setup_nested(iommu, dev, ret = intel_pasid_setup_nested(iommu, dev,
(pgd_t *)(uintptr_t)data->gpgd, (pgd_t *)(uintptr_t)data->gpgd,
data->hpasid, &data->vendor.vtd, dmar_domain, data->hpasid, &data->vendor.vtd, dmar_domain,
data->addr_width); data->addr_width);
spin_unlock(&iommu->lock); spin_unlock_irqrestore(&iommu->lock, iflags);
if (ret) { if (ret) {
dev_err_ratelimited(dev, "Failed to set up PASID %llu in nested mode, Err %d\n", dev_err_ratelimited(dev, "Failed to set up PASID %llu in nested mode, Err %d\n",
data->hpasid, ret); data->hpasid, ret);
...@@ -487,6 +488,7 @@ intel_svm_bind_mm(struct device *dev, unsigned int flags, ...@@ -487,6 +488,7 @@ intel_svm_bind_mm(struct device *dev, unsigned int flags,
struct device_domain_info *info; struct device_domain_info *info;
struct intel_svm_dev *sdev; struct intel_svm_dev *sdev;
struct intel_svm *svm = NULL; struct intel_svm *svm = NULL;
unsigned long iflags;
int pasid_max; int pasid_max;
int ret; int ret;
...@@ -606,14 +608,14 @@ intel_svm_bind_mm(struct device *dev, unsigned int flags, ...@@ -606,14 +608,14 @@ intel_svm_bind_mm(struct device *dev, unsigned int flags,
} }
} }
spin_lock(&iommu->lock); spin_lock_irqsave(&iommu->lock, iflags);
ret = intel_pasid_setup_first_level(iommu, dev, ret = intel_pasid_setup_first_level(iommu, dev,
mm ? mm->pgd : init_mm.pgd, mm ? mm->pgd : init_mm.pgd,
svm->pasid, FLPT_DEFAULT_DID, svm->pasid, FLPT_DEFAULT_DID,
(mm ? 0 : PASID_FLAG_SUPERVISOR_MODE) | (mm ? 0 : PASID_FLAG_SUPERVISOR_MODE) |
(cpu_feature_enabled(X86_FEATURE_LA57) ? (cpu_feature_enabled(X86_FEATURE_LA57) ?
PASID_FLAG_FL5LP : 0)); PASID_FLAG_FL5LP : 0));
spin_unlock(&iommu->lock); spin_unlock_irqrestore(&iommu->lock, iflags);
if (ret) { if (ret) {
if (mm) if (mm)
mmu_notifier_unregister(&svm->notifier, mm); mmu_notifier_unregister(&svm->notifier, mm);
...@@ -633,14 +635,14 @@ intel_svm_bind_mm(struct device *dev, unsigned int flags, ...@@ -633,14 +635,14 @@ intel_svm_bind_mm(struct device *dev, unsigned int flags,
* Binding a new device with existing PASID, need to setup * Binding a new device with existing PASID, need to setup
* the PASID entry. * the PASID entry.
*/ */
spin_lock(&iommu->lock); spin_lock_irqsave(&iommu->lock, iflags);
ret = intel_pasid_setup_first_level(iommu, dev, ret = intel_pasid_setup_first_level(iommu, dev,
mm ? mm->pgd : init_mm.pgd, mm ? mm->pgd : init_mm.pgd,
svm->pasid, FLPT_DEFAULT_DID, svm->pasid, FLPT_DEFAULT_DID,
(mm ? 0 : PASID_FLAG_SUPERVISOR_MODE) | (mm ? 0 : PASID_FLAG_SUPERVISOR_MODE) |
(cpu_feature_enabled(X86_FEATURE_LA57) ? (cpu_feature_enabled(X86_FEATURE_LA57) ?
PASID_FLAG_FL5LP : 0)); PASID_FLAG_FL5LP : 0));
spin_unlock(&iommu->lock); spin_unlock_irqrestore(&iommu->lock, iflags);
if (ret) { if (ret) {
kfree(sdev); kfree(sdev);
goto out; goto out;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册