Commit c2b00cbd authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'iommu-fixes-v5.7-rc6' of git://git.kernel.org/pub/scm/linux/kernel/git/joro/iommu

Pull iommu fixes from Joerg Roedel:
 "All related to the AMD IOMMU driver:

   - ACPI table parser fix to correctly read the UID of ACPI devices

   - ACPI UID device matching fix

   - Fix deferred device attachment to a domain in kdump kernels when
     the IOMMU driver uses the dma-iommu DMA-API implementation"

* tag 'iommu-fixes-v5.7-rc6' of git://git.kernel.org/pub/scm/linux/kernel/git/joro/iommu:
  iommu: Fix deferred domain attachment
  iommu/amd: Fix get_acpihid_device_id()
  iommu/amd: Fix over-read of ACPI UID from IVRS table
parents 7bd57fbc bd421264
...@@ -127,7 +127,8 @@ static inline int get_acpihid_device_id(struct device *dev, ...@@ -127,7 +127,8 @@ static inline int get_acpihid_device_id(struct device *dev,
return -ENODEV; return -ENODEV;
list_for_each_entry(p, &acpihid_map, list) { list_for_each_entry(p, &acpihid_map, list) {
if (acpi_dev_hid_uid_match(adev, p->hid, p->uid)) { if (acpi_dev_hid_uid_match(adev, p->hid,
p->uid[0] ? p->uid : NULL)) {
if (entry) if (entry)
*entry = p; *entry = p;
return p->devid; return p->devid;
......
...@@ -1329,8 +1329,8 @@ static int __init init_iommu_from_acpi(struct amd_iommu *iommu, ...@@ -1329,8 +1329,8 @@ static int __init init_iommu_from_acpi(struct amd_iommu *iommu,
} }
case IVHD_DEV_ACPI_HID: { case IVHD_DEV_ACPI_HID: {
u16 devid; u16 devid;
u8 hid[ACPIHID_HID_LEN] = {0}; u8 hid[ACPIHID_HID_LEN];
u8 uid[ACPIHID_UID_LEN] = {0}; u8 uid[ACPIHID_UID_LEN];
int ret; int ret;
if (h->type != 0x40) { if (h->type != 0x40) {
...@@ -1347,6 +1347,7 @@ static int __init init_iommu_from_acpi(struct amd_iommu *iommu, ...@@ -1347,6 +1347,7 @@ static int __init init_iommu_from_acpi(struct amd_iommu *iommu,
break; break;
} }
uid[0] = '\0';
switch (e->uidf) { switch (e->uidf) {
case UID_NOT_PRESENT: case UID_NOT_PRESENT:
...@@ -1361,8 +1362,8 @@ static int __init init_iommu_from_acpi(struct amd_iommu *iommu, ...@@ -1361,8 +1362,8 @@ static int __init init_iommu_from_acpi(struct amd_iommu *iommu,
break; break;
case UID_IS_CHARACTER: case UID_IS_CHARACTER:
memcpy(uid, (u8 *)(&e->uid), ACPIHID_UID_LEN - 1); memcpy(uid, &e->uid, e->uidl);
uid[ACPIHID_UID_LEN - 1] = '\0'; uid[e->uidl] = '\0';
break; break;
default: default:
......
...@@ -693,6 +693,15 @@ static int iommu_group_create_direct_mappings(struct iommu_group *group, ...@@ -693,6 +693,15 @@ static int iommu_group_create_direct_mappings(struct iommu_group *group,
return ret; return ret;
} }
static bool iommu_is_attach_deferred(struct iommu_domain *domain,
struct device *dev)
{
if (domain->ops->is_attach_deferred)
return domain->ops->is_attach_deferred(domain, dev);
return false;
}
/** /**
* iommu_group_add_device - add a device to an iommu group * iommu_group_add_device - add a device to an iommu group
* @group: the group into which to add the device (reference should be held) * @group: the group into which to add the device (reference should be held)
...@@ -747,7 +756,7 @@ int iommu_group_add_device(struct iommu_group *group, struct device *dev) ...@@ -747,7 +756,7 @@ int iommu_group_add_device(struct iommu_group *group, struct device *dev)
mutex_lock(&group->mutex); mutex_lock(&group->mutex);
list_add_tail(&device->list, &group->devices); list_add_tail(&device->list, &group->devices);
if (group->domain) if (group->domain && !iommu_is_attach_deferred(group->domain, dev))
ret = __iommu_attach_device(group->domain, dev); ret = __iommu_attach_device(group->domain, dev);
mutex_unlock(&group->mutex); mutex_unlock(&group->mutex);
if (ret) if (ret)
...@@ -1653,9 +1662,6 @@ static int __iommu_attach_device(struct iommu_domain *domain, ...@@ -1653,9 +1662,6 @@ static int __iommu_attach_device(struct iommu_domain *domain,
struct device *dev) struct device *dev)
{ {
int ret; int ret;
if ((domain->ops->is_attach_deferred != NULL) &&
domain->ops->is_attach_deferred(domain, dev))
return 0;
if (unlikely(domain->ops->attach_dev == NULL)) if (unlikely(domain->ops->attach_dev == NULL))
return -ENODEV; return -ENODEV;
...@@ -1727,8 +1733,7 @@ EXPORT_SYMBOL_GPL(iommu_sva_unbind_gpasid); ...@@ -1727,8 +1733,7 @@ EXPORT_SYMBOL_GPL(iommu_sva_unbind_gpasid);
static void __iommu_detach_device(struct iommu_domain *domain, static void __iommu_detach_device(struct iommu_domain *domain,
struct device *dev) struct device *dev)
{ {
if ((domain->ops->is_attach_deferred != NULL) && if (iommu_is_attach_deferred(domain, dev))
domain->ops->is_attach_deferred(domain, dev))
return; return;
if (unlikely(domain->ops->detach_dev == NULL)) if (unlikely(domain->ops->detach_dev == NULL))
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment