Commit 5d95f40e authored by Robin Murphy's avatar Robin Murphy Committed by Joerg Roedel

iommu: Do physical merging in iommu_map_sg()

The original motivation for iommu_map_sg() was to give IOMMU drivers the
chance to map an IOVA-contiguous scatterlist as efficiently as they
could. It turns out that there isn't really much driver-specific
business involved there, so now that the default implementation is
mandatory let's just improve that - the main thing we're after is to use
larger pages wherever possible, and as long as domain->pgsize_bitmap
reflects reality, iommu_map() can already do that in a generic way. All
we need to do is detect physically-contiguous segments and batch them
into a single map operation, since whatever we do here is transparent to
our caller and not bound by any segment-length restrictions on the list
itself.

Speaking of efficiency, there's really very little point in duplicating
the checks that iommu_map() is going to do anyway, so those get cleared
up in the process.
Signed-off-by: default avatarRobin Murphy <robin.murphy@arm.com>
Signed-off-by: default avatarJoerg Roedel <jroedel@suse.de>
parent 65102238
...@@ -1712,33 +1712,32 @@ EXPORT_SYMBOL_GPL(iommu_unmap_fast); ...@@ -1712,33 +1712,32 @@ EXPORT_SYMBOL_GPL(iommu_unmap_fast);
size_t iommu_map_sg(struct iommu_domain *domain, unsigned long iova, size_t iommu_map_sg(struct iommu_domain *domain, unsigned long iova,
struct scatterlist *sg, unsigned int nents, int prot) struct scatterlist *sg, unsigned int nents, int prot)
{ {
struct scatterlist *s; size_t len = 0, mapped = 0;
size_t mapped = 0; phys_addr_t start;
unsigned int i, min_pagesz; unsigned int i = 0;
int ret; int ret;
if (unlikely(domain->pgsize_bitmap == 0UL)) while (i <= nents) {
return 0; phys_addr_t s_phys = sg_phys(sg);
min_pagesz = 1 << __ffs(domain->pgsize_bitmap);
for_each_sg(sg, s, nents, i) { if (len && s_phys != start + len) {
phys_addr_t phys = page_to_phys(sg_page(s)) + s->offset; ret = iommu_map(domain, iova + mapped, start, len, prot);
if (ret)
goto out_err;
/* mapped += len;
* We are mapping on IOMMU page boundaries, so offset within len = 0;
* the page must be 0. However, the IOMMU may support pages }
* smaller than PAGE_SIZE, so s->offset may still represent
* an offset of that boundary within the CPU page.
*/
if (!IS_ALIGNED(s->offset, min_pagesz))
goto out_err;
ret = iommu_map(domain, iova + mapped, phys, s->length, prot); if (len) {
if (ret) len += sg->length;
goto out_err; } else {
len = sg->length;
start = s_phys;
}
mapped += s->length; if (++i < nents)
sg = sg_next(sg);
} }
return mapped; return mapped;
......
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