iommu/tegra: gart: Fix spinlock recursion
Fix spinlock recursion bug that happens on IOMMU domain destruction if any of the allocated domains have devices attached to them. Signed-off-by: Dmitry Osipenko <digetx@gmail.com> Acked-by: Thierry Reding <treding@nvidia.com> Signed-off-by: Joerg Roedel <jroedel@suse.de>
This commit is contained in:
parent
f2dcded1be
commit
c3086fad27
@ -197,25 +197,33 @@ fail:
|
|||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void gart_iommu_detach_dev(struct iommu_domain *domain,
|
static void __gart_iommu_detach_dev(struct iommu_domain *domain,
|
||||||
struct device *dev)
|
struct device *dev)
|
||||||
{
|
{
|
||||||
struct gart_domain *gart_domain = to_gart_domain(domain);
|
struct gart_domain *gart_domain = to_gart_domain(domain);
|
||||||
struct gart_device *gart = gart_domain->gart;
|
struct gart_device *gart = gart_domain->gart;
|
||||||
struct gart_client *c;
|
struct gart_client *c;
|
||||||
|
|
||||||
spin_lock(&gart->client_lock);
|
|
||||||
|
|
||||||
list_for_each_entry(c, &gart->client, list) {
|
list_for_each_entry(c, &gart->client, list) {
|
||||||
if (c->dev == dev) {
|
if (c->dev == dev) {
|
||||||
list_del(&c->list);
|
list_del(&c->list);
|
||||||
devm_kfree(gart->dev, c);
|
devm_kfree(gart->dev, c);
|
||||||
dev_dbg(gart->dev, "Detached %s\n", dev_name(dev));
|
dev_dbg(gart->dev, "Detached %s\n", dev_name(dev));
|
||||||
goto out;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
dev_err(gart->dev, "Couldn't find\n");
|
|
||||||
out:
|
dev_err(gart->dev, "Couldn't find %s to detach\n", dev_name(dev));
|
||||||
|
}
|
||||||
|
|
||||||
|
static void gart_iommu_detach_dev(struct iommu_domain *domain,
|
||||||
|
struct device *dev)
|
||||||
|
{
|
||||||
|
struct gart_domain *gart_domain = to_gart_domain(domain);
|
||||||
|
struct gart_device *gart = gart_domain->gart;
|
||||||
|
|
||||||
|
spin_lock(&gart->client_lock);
|
||||||
|
__gart_iommu_detach_dev(domain, dev);
|
||||||
spin_unlock(&gart->client_lock);
|
spin_unlock(&gart->client_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -255,7 +263,7 @@ static void gart_iommu_domain_free(struct iommu_domain *domain)
|
|||||||
struct gart_client *c;
|
struct gart_client *c;
|
||||||
|
|
||||||
list_for_each_entry(c, &gart->client, list)
|
list_for_each_entry(c, &gart->client, list)
|
||||||
gart_iommu_detach_dev(domain, c->dev);
|
__gart_iommu_detach_dev(domain, c->dev);
|
||||||
}
|
}
|
||||||
spin_unlock(&gart->client_lock);
|
spin_unlock(&gart->client_lock);
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user