提交 03edb226 编写于 作者: W Will Deacon

iommu/arm-smmu: handle multi-alias IOMMU groups for PCI devices

IOMMU groups for PCI devices can correspond to multiple DMA aliases due
to things like ACS and PCI quirks.

This patch extends the ARM SMMU ->add_device callback so that we
consider all of the DMA aliases for a PCI IOMMU group, rather than
creating a separate group for each Requester ID.
Signed-off-by: NWill Deacon <will.deacon@arm.com>
上级 f1d84548
...@@ -1330,61 +1330,83 @@ static void __arm_smmu_release_pci_iommudata(void *data) ...@@ -1330,61 +1330,83 @@ static void __arm_smmu_release_pci_iommudata(void *data)
kfree(data); kfree(data);
} }
static int arm_smmu_add_device(struct device *dev) static int arm_smmu_add_pci_device(struct pci_dev *pdev)
{ {
struct arm_smmu_device *smmu; int i, ret;
struct arm_smmu_master_cfg *cfg; u16 sid;
struct iommu_group *group; struct iommu_group *group;
void (*releasefn)(void *) = NULL; struct arm_smmu_master_cfg *cfg;
int ret;
smmu = find_smmu_for_device(dev);
if (!smmu)
return -ENODEV;
group = iommu_group_alloc(); group = iommu_group_get_for_dev(&pdev->dev);
if (IS_ERR(group)) { if (IS_ERR(group))
dev_err(dev, "Failed to allocate IOMMU group\n");
return PTR_ERR(group); return PTR_ERR(group);
}
if (dev_is_pci(dev)) {
struct pci_dev *pdev = to_pci_dev(dev);
cfg = iommu_group_get_iommudata(group);
if (!cfg) {
cfg = kzalloc(sizeof(*cfg), GFP_KERNEL); cfg = kzalloc(sizeof(*cfg), GFP_KERNEL);
if (!cfg) { if (!cfg) {
ret = -ENOMEM; ret = -ENOMEM;
goto out_put_group; goto out_put_group;
} }
cfg->num_streamids = 1; iommu_group_set_iommudata(group, cfg,
/* __arm_smmu_release_pci_iommudata);
* Assume Stream ID == Requester ID for now. }
* We need a way to describe the ID mappings in FDT.
*/
pci_for_each_dma_alias(pdev, __arm_smmu_get_pci_sid,
&cfg->streamids[0]);
releasefn = __arm_smmu_release_pci_iommudata;
} else {
struct arm_smmu_master *master;
master = find_smmu_master(smmu, dev->of_node);
if (!master) {
ret = -ENODEV;
goto out_put_group;
}
cfg = &master->cfg; if (cfg->num_streamids >= MAX_MASTER_STREAMIDS) {
ret = -ENOSPC;
goto out_put_group;
} }
iommu_group_set_iommudata(group, cfg, releasefn); /*
ret = iommu_group_add_device(group, dev); * Assume Stream ID == Requester ID for now.
* We need a way to describe the ID mappings in FDT.
*/
pci_for_each_dma_alias(pdev, __arm_smmu_get_pci_sid, &sid);
for (i = 0; i < cfg->num_streamids; ++i)
if (cfg->streamids[i] == sid)
break;
/* Avoid duplicate SIDs, as this can lead to SMR conflicts */
if (i == cfg->num_streamids)
cfg->streamids[cfg->num_streamids++] = sid;
return 0;
out_put_group: out_put_group:
iommu_group_put(group); iommu_group_put(group);
return ret; return ret;
} }
static int arm_smmu_add_platform_device(struct device *dev)
{
struct iommu_group *group;
struct arm_smmu_master *master;
struct arm_smmu_device *smmu = find_smmu_for_device(dev);
if (!smmu)
return -ENODEV;
master = find_smmu_master(smmu, dev->of_node);
if (!master)
return -ENODEV;
/* No automatic group creation for platform devices */
group = iommu_group_alloc();
if (IS_ERR(group))
return PTR_ERR(group);
iommu_group_set_iommudata(group, &master->cfg, NULL);
return iommu_group_add_device(group, dev);
}
static int arm_smmu_add_device(struct device *dev)
{
if (dev_is_pci(dev))
return arm_smmu_add_pci_device(to_pci_dev(dev));
return arm_smmu_add_platform_device(dev);
}
static void arm_smmu_remove_device(struct device *dev) static void arm_smmu_remove_device(struct device *dev)
{ {
iommu_group_remove_device(dev); iommu_group_remove_device(dev);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册