diff options
Diffstat (limited to 'drivers/iommu/iommu.c')
| -rw-r--r-- | drivers/iommu/iommu.c | 81 | 
1 files changed, 71 insertions, 10 deletions
diff --git a/drivers/iommu/iommu.c b/drivers/iommu/iommu.c index d658c7c6a2ab..fdd40756dbc1 100644 --- a/drivers/iommu/iommu.c +++ b/drivers/iommu/iommu.c @@ -312,8 +312,8 @@ int iommu_insert_resv_region(struct iommu_resv_region *new,  	list_for_each_entry_safe(iter, tmp, regions, list) {  		phys_addr_t top_end, iter_end = iter->start + iter->length - 1; -		/* no merge needed on elements of different types than @nr */ -		if (iter->type != nr->type) { +		/* no merge needed on elements of different types than @new */ +		if (iter->type != new->type) {  			list_move_tail(&iter->list, &stack);  			continue;  		} @@ -1665,6 +1665,36 @@ out_unlock:  }  EXPORT_SYMBOL_GPL(iommu_attach_device); +int iommu_cache_invalidate(struct iommu_domain *domain, struct device *dev, +			   struct iommu_cache_invalidate_info *inv_info) +{ +	if (unlikely(!domain->ops->cache_invalidate)) +		return -ENODEV; + +	return domain->ops->cache_invalidate(domain, dev, inv_info); +} +EXPORT_SYMBOL_GPL(iommu_cache_invalidate); + +int iommu_sva_bind_gpasid(struct iommu_domain *domain, +			   struct device *dev, struct iommu_gpasid_bind_data *data) +{ +	if (unlikely(!domain->ops->sva_bind_gpasid)) +		return -ENODEV; + +	return domain->ops->sva_bind_gpasid(domain, dev, data); +} +EXPORT_SYMBOL_GPL(iommu_sva_bind_gpasid); + +int iommu_sva_unbind_gpasid(struct iommu_domain *domain, struct device *dev, +			     ioasid_t pasid) +{ +	if (unlikely(!domain->ops->sva_unbind_gpasid)) +		return -ENODEV; + +	return domain->ops->sva_unbind_gpasid(dev, pasid); +} +EXPORT_SYMBOL_GPL(iommu_sva_unbind_gpasid); +  static void __iommu_detach_device(struct iommu_domain *domain,  				  struct device *dev)  { @@ -1854,8 +1884,8 @@ static size_t iommu_pgsize(struct iommu_domain *domain,  	return pgsize;  } -int iommu_map(struct iommu_domain *domain, unsigned long iova, -	      phys_addr_t paddr, size_t size, int prot) +int __iommu_map(struct iommu_domain *domain, unsigned long iova, +	      phys_addr_t paddr, size_t size, int prot, gfp_t gfp)  {  	const struct iommu_ops *ops = domain->ops;  	unsigned long orig_iova = iova; @@ -1892,8 +1922,8 @@ int iommu_map(struct iommu_domain *domain, unsigned long iova,  		pr_debug("mapping: iova 0x%lx pa %pa pgsize 0x%zx\n",  			 iova, &paddr, pgsize); +		ret = ops->map(domain, iova, paddr, pgsize, prot, gfp); -		ret = ops->map(domain, iova, paddr, pgsize, prot);  		if (ret)  			break; @@ -1913,8 +1943,22 @@ int iommu_map(struct iommu_domain *domain, unsigned long iova,  	return ret;  } + +int iommu_map(struct iommu_domain *domain, unsigned long iova, +	      phys_addr_t paddr, size_t size, int prot) +{ +	might_sleep(); +	return __iommu_map(domain, iova, paddr, size, prot, GFP_KERNEL); +}  EXPORT_SYMBOL_GPL(iommu_map); +int iommu_map_atomic(struct iommu_domain *domain, unsigned long iova, +	      phys_addr_t paddr, size_t size, int prot) +{ +	return __iommu_map(domain, iova, paddr, size, prot, GFP_ATOMIC); +} +EXPORT_SYMBOL_GPL(iommu_map_atomic); +  static size_t __iommu_unmap(struct iommu_domain *domain,  			    unsigned long iova, size_t size,  			    struct iommu_iotlb_gather *iotlb_gather) @@ -1991,8 +2035,9 @@ size_t iommu_unmap_fast(struct iommu_domain *domain,  }  EXPORT_SYMBOL_GPL(iommu_unmap_fast); -size_t iommu_map_sg(struct iommu_domain *domain, unsigned long iova, -		    struct scatterlist *sg, unsigned int nents, int prot) +size_t __iommu_map_sg(struct iommu_domain *domain, unsigned long iova, +		    struct scatterlist *sg, unsigned int nents, int prot, +		    gfp_t gfp)  {  	size_t len = 0, mapped = 0;  	phys_addr_t start; @@ -2003,7 +2048,9 @@ size_t iommu_map_sg(struct iommu_domain *domain, unsigned long iova,  		phys_addr_t s_phys = sg_phys(sg);  		if (len && s_phys != start + len) { -			ret = iommu_map(domain, iova + mapped, start, len, prot); +			ret = __iommu_map(domain, iova + mapped, start, +					len, prot, gfp); +  			if (ret)  				goto out_err; @@ -2031,8 +2078,22 @@ out_err:  	return 0;  } + +size_t iommu_map_sg(struct iommu_domain *domain, unsigned long iova, +		    struct scatterlist *sg, unsigned int nents, int prot) +{ +	might_sleep(); +	return __iommu_map_sg(domain, iova, sg, nents, prot, GFP_KERNEL); +}  EXPORT_SYMBOL_GPL(iommu_map_sg); +size_t iommu_map_sg_atomic(struct iommu_domain *domain, unsigned long iova, +		    struct scatterlist *sg, unsigned int nents, int prot) +{ +	return __iommu_map_sg(domain, iova, sg, nents, prot, GFP_ATOMIC); +} +EXPORT_SYMBOL_GPL(iommu_map_sg_atomic); +  int iommu_domain_window_enable(struct iommu_domain *domain, u32 wnd_nr,  			       phys_addr_t paddr, u64 size, int prot)  { @@ -2221,13 +2282,13 @@ request_default_domain_for_dev(struct device *dev, unsigned long type)  		goto out;  	} -	iommu_group_create_direct_mappings(group, dev); -  	/* Make the domain the default for this group */  	if (group->default_domain)  		iommu_domain_free(group->default_domain);  	group->default_domain = domain; +	iommu_group_create_direct_mappings(group, dev); +  	dev_info(dev, "Using iommu %s mapping\n",  		 type == IOMMU_DOMAIN_DMA ? "dma" : "direct");  |