diff options
Diffstat (limited to 'drivers/iommu/arm')
| -rw-r--r-- | drivers/iommu/arm/arm-smmu-v3/arm-smmu-v3-sva.c | 6 | ||||
| -rw-r--r-- | drivers/iommu/arm/arm-smmu-v3/arm-smmu-v3.c | 23 | ||||
| -rw-r--r-- | drivers/iommu/arm/arm-smmu-v3/tegra241-cmdqv.c | 32 | 
3 files changed, 29 insertions, 32 deletions
| diff --git a/drivers/iommu/arm/arm-smmu-v3/arm-smmu-v3-sva.c b/drivers/iommu/arm/arm-smmu-v3/arm-smmu-v3-sva.c index 9ba596430e7c..980cc6b33c43 100644 --- a/drivers/iommu/arm/arm-smmu-v3/arm-smmu-v3-sva.c +++ b/drivers/iommu/arm/arm-smmu-v3/arm-smmu-v3-sva.c @@ -411,6 +411,12 @@ struct iommu_domain *arm_smmu_sva_domain_alloc(struct device *dev,  		return ERR_CAST(smmu_domain);  	smmu_domain->domain.type = IOMMU_DOMAIN_SVA;  	smmu_domain->domain.ops = &arm_smmu_sva_domain_ops; + +	/* +	 * Choose page_size as the leaf page size for invalidation when +	 * ARM_SMMU_FEAT_RANGE_INV is present +	 */ +	smmu_domain->domain.pgsize_bitmap = PAGE_SIZE;  	smmu_domain->smmu = smmu;  	ret = xa_alloc(&arm_smmu_asid_xa, &asid, smmu_domain, diff --git a/drivers/iommu/arm/arm-smmu-v3/arm-smmu-v3.c b/drivers/iommu/arm/arm-smmu-v3/arm-smmu-v3.c index b4c21aaed126..48d910399a1b 100644 --- a/drivers/iommu/arm/arm-smmu-v3/arm-smmu-v3.c +++ b/drivers/iommu/arm/arm-smmu-v3/arm-smmu-v3.c @@ -3388,6 +3388,7 @@ static int arm_smmu_insert_master(struct arm_smmu_device *smmu,  	mutex_lock(&smmu->streams_mutex);  	for (i = 0; i < fwspec->num_ids; i++) {  		struct arm_smmu_stream *new_stream = &master->streams[i]; +		struct rb_node *existing;  		u32 sid = fwspec->ids[i];  		new_stream->id = sid; @@ -3398,11 +3399,21 @@ static int arm_smmu_insert_master(struct arm_smmu_device *smmu,  			break;  		/* Insert into SID tree */ -		if (rb_find_add(&new_stream->node, &smmu->streams, -				arm_smmu_streams_cmp_node)) { -			dev_warn(master->dev, "stream %u already in tree\n", -				 sid); -			ret = -EINVAL; +		existing = rb_find_add(&new_stream->node, &smmu->streams, +				       arm_smmu_streams_cmp_node); +		if (existing) { +			struct arm_smmu_master *existing_master = +				rb_entry(existing, struct arm_smmu_stream, node) +					->master; + +			/* Bridged PCI devices may end up with duplicated IDs */ +			if (existing_master == master) +				continue; + +			dev_warn(master->dev, +				 "Aliasing StreamID 0x%x (from %s) unsupported, expect DMA to be broken\n", +				 sid, dev_name(existing_master->dev)); +			ret = -ENODEV;  			break;  		}  	} @@ -4429,6 +4440,8 @@ static int arm_smmu_device_hw_probe(struct arm_smmu_device *smmu)  	reg = readl_relaxed(smmu->base + ARM_SMMU_IDR3);  	if (FIELD_GET(IDR3_RIL, reg))  		smmu->features |= ARM_SMMU_FEAT_RANGE_INV; +	if (FIELD_GET(IDR3_FWB, reg)) +		smmu->features |= ARM_SMMU_FEAT_S2FWB;  	/* IDR5 */  	reg = readl_relaxed(smmu->base + ARM_SMMU_IDR5); diff --git a/drivers/iommu/arm/arm-smmu-v3/tegra241-cmdqv.c b/drivers/iommu/arm/arm-smmu-v3/tegra241-cmdqv.c index d525ab43a4ae..dd7d030d2e89 100644 --- a/drivers/iommu/arm/arm-smmu-v3/tegra241-cmdqv.c +++ b/drivers/iommu/arm/arm-smmu-v3/tegra241-cmdqv.c @@ -487,17 +487,6 @@ static int tegra241_cmdqv_hw_reset(struct arm_smmu_device *smmu)  /* VCMDQ Resource Helpers */ -static void tegra241_vcmdq_free_smmu_cmdq(struct tegra241_vcmdq *vcmdq) -{ -	struct arm_smmu_queue *q = &vcmdq->cmdq.q; -	size_t nents = 1 << q->llq.max_n_shift; -	size_t qsz = nents << CMDQ_ENT_SZ_SHIFT; - -	if (!q->base) -		return; -	dmam_free_coherent(vcmdq->cmdqv->smmu.dev, qsz, q->base, q->base_dma); -} -  static int tegra241_vcmdq_alloc_smmu_cmdq(struct tegra241_vcmdq *vcmdq)  {  	struct arm_smmu_device *smmu = &vcmdq->cmdqv->smmu; @@ -560,7 +549,8 @@ static void tegra241_vintf_free_lvcmdq(struct tegra241_vintf *vintf, u16 lidx)  	struct tegra241_vcmdq *vcmdq = vintf->lvcmdqs[lidx];  	char header[64]; -	tegra241_vcmdq_free_smmu_cmdq(vcmdq); +	/* Note that the lvcmdq queue memory space is managed by devres */ +  	tegra241_vintf_deinit_lvcmdq(vintf, lidx);  	dev_dbg(vintf->cmdqv->dev, @@ -768,13 +758,13 @@ static int tegra241_cmdqv_init_structures(struct arm_smmu_device *smmu)  	vintf = kzalloc(sizeof(*vintf), GFP_KERNEL);  	if (!vintf) -		goto out_fallback; +		return -ENOMEM;  	/* Init VINTF0 for in-kernel use */  	ret = tegra241_cmdqv_init_vintf(cmdqv, 0, vintf);  	if (ret) {  		dev_err(cmdqv->dev, "failed to init vintf0: %d\n", ret); -		goto free_vintf; +		return ret;  	}  	/* Preallocate logical VCMDQs to VINTF0 */ @@ -783,24 +773,12 @@ static int tegra241_cmdqv_init_structures(struct arm_smmu_device *smmu)  		vcmdq = tegra241_vintf_alloc_lvcmdq(vintf, lidx);  		if (IS_ERR(vcmdq)) -			goto free_lvcmdq; +			return PTR_ERR(vcmdq);  	}  	/* Now, we are ready to run all the impl ops */  	smmu->impl_ops = &tegra241_cmdqv_impl_ops;  	return 0; - -free_lvcmdq: -	for (lidx--; lidx >= 0; lidx--) -		tegra241_vintf_free_lvcmdq(vintf, lidx); -	tegra241_cmdqv_deinit_vintf(cmdqv, vintf->idx); -free_vintf: -	kfree(vintf); -out_fallback: -	dev_info(smmu->impl_dev, "Falling back to standard SMMU CMDQ\n"); -	smmu->options &= ~ARM_SMMU_OPT_TEGRA241_CMDQV; -	tegra241_cmdqv_remove(smmu); -	return 0;  }  #ifdef CONFIG_IOMMU_DEBUGFS | 
