diff options
Diffstat (limited to 'drivers/iommu/iommu.c')
-rw-r--r-- | drivers/iommu/iommu.c | 131 |
1 files changed, 117 insertions, 14 deletions
diff --git a/drivers/iommu/iommu.c b/drivers/iommu/iommu.c index 7a2953d8f12e..b278458d5816 100644 --- a/drivers/iommu/iommu.c +++ b/drivers/iommu/iommu.c | |||
@@ -16,6 +16,8 @@ | |||
16 | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | 16 | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
17 | */ | 17 | */ |
18 | 18 | ||
19 | #define pr_fmt(fmt) "%s: " fmt, __func__ | ||
20 | |||
19 | #include <linux/device.h> | 21 | #include <linux/device.h> |
20 | #include <linux/kernel.h> | 22 | #include <linux/kernel.h> |
21 | #include <linux/bug.h> | 23 | #include <linux/bug.h> |
@@ -47,6 +49,16 @@ int bus_set_iommu(struct bus_type *bus, struct iommu_ops *ops) | |||
47 | if (bus->iommu_ops != NULL) | 49 | if (bus->iommu_ops != NULL) |
48 | return -EBUSY; | 50 | return -EBUSY; |
49 | 51 | ||
52 | /* | ||
53 | * Set the default pgsize values, which retain the existing | ||
54 | * IOMMU API behavior: drivers will be called to map | ||
55 | * regions that are sized/aligned to order of 4KiB pages. | ||
56 | * | ||
57 | * This will be removed once all drivers are migrated. | ||
58 | */ | ||
59 | if (!ops->pgsize_bitmap) | ||
60 | ops->pgsize_bitmap = ~0xFFFUL; | ||
61 | |||
50 | bus->iommu_ops = ops; | 62 | bus->iommu_ops = ops; |
51 | 63 | ||
52 | /* Do IOMMU specific setup for this bus-type */ | 64 | /* Do IOMMU specific setup for this bus-type */ |
@@ -157,34 +169,125 @@ int iommu_domain_has_cap(struct iommu_domain *domain, | |||
157 | EXPORT_SYMBOL_GPL(iommu_domain_has_cap); | 169 | EXPORT_SYMBOL_GPL(iommu_domain_has_cap); |
158 | 170 | ||
159 | int iommu_map(struct iommu_domain *domain, unsigned long iova, | 171 | int iommu_map(struct iommu_domain *domain, unsigned long iova, |
160 | phys_addr_t paddr, int gfp_order, int prot) | 172 | phys_addr_t paddr, size_t size, int prot) |
161 | { | 173 | { |
162 | size_t size; | 174 | unsigned long orig_iova = iova; |
175 | unsigned int min_pagesz; | ||
176 | size_t orig_size = size; | ||
177 | int ret = 0; | ||
163 | 178 | ||
164 | if (unlikely(domain->ops->map == NULL)) | 179 | if (unlikely(domain->ops->map == NULL)) |
165 | return -ENODEV; | 180 | return -ENODEV; |
166 | 181 | ||
167 | size = PAGE_SIZE << gfp_order; | 182 | /* find out the minimum page size supported */ |
183 | min_pagesz = 1 << __ffs(domain->ops->pgsize_bitmap); | ||
184 | |||
185 | /* | ||
186 | * both the virtual address and the physical one, as well as | ||
187 | * the size of the mapping, must be aligned (at least) to the | ||
188 | * size of the smallest page supported by the hardware | ||
189 | */ | ||
190 | if (!IS_ALIGNED(iova | paddr | size, min_pagesz)) { | ||
191 | pr_err("unaligned: iova 0x%lx pa 0x%lx size 0x%lx min_pagesz " | ||
192 | "0x%x\n", iova, (unsigned long)paddr, | ||
193 | (unsigned long)size, min_pagesz); | ||
194 | return -EINVAL; | ||
195 | } | ||
196 | |||
197 | pr_debug("map: iova 0x%lx pa 0x%lx size 0x%lx\n", iova, | ||
198 | (unsigned long)paddr, (unsigned long)size); | ||
199 | |||
200 | while (size) { | ||
201 | unsigned long pgsize, addr_merge = iova | paddr; | ||
202 | unsigned int pgsize_idx; | ||
203 | |||
204 | /* Max page size that still fits into 'size' */ | ||
205 | pgsize_idx = __fls(size); | ||
206 | |||
207 | /* need to consider alignment requirements ? */ | ||
208 | if (likely(addr_merge)) { | ||
209 | /* Max page size allowed by both iova and paddr */ | ||
210 | unsigned int align_pgsize_idx = __ffs(addr_merge); | ||
211 | |||
212 | pgsize_idx = min(pgsize_idx, align_pgsize_idx); | ||
213 | } | ||
214 | |||
215 | /* build a mask of acceptable page sizes */ | ||
216 | pgsize = (1UL << (pgsize_idx + 1)) - 1; | ||
217 | |||
218 | /* throw away page sizes not supported by the hardware */ | ||
219 | pgsize &= domain->ops->pgsize_bitmap; | ||
168 | 220 | ||
169 | BUG_ON(!IS_ALIGNED(iova | paddr, size)); | 221 | /* make sure we're still sane */ |
222 | BUG_ON(!pgsize); | ||
170 | 223 | ||
171 | return domain->ops->map(domain, iova, paddr, size, prot); | 224 | /* pick the biggest page */ |
225 | pgsize_idx = __fls(pgsize); | ||
226 | pgsize = 1UL << pgsize_idx; | ||
227 | |||
228 | pr_debug("mapping: iova 0x%lx pa 0x%lx pgsize %lu\n", iova, | ||
229 | (unsigned long)paddr, pgsize); | ||
230 | |||
231 | ret = domain->ops->map(domain, iova, paddr, pgsize, prot); | ||
232 | if (ret) | ||
233 | break; | ||
234 | |||
235 | iova += pgsize; | ||
236 | paddr += pgsize; | ||
237 | size -= pgsize; | ||
238 | } | ||
239 | |||
240 | /* unroll mapping in case something went wrong */ | ||
241 | if (ret) | ||
242 | iommu_unmap(domain, orig_iova, orig_size - size); | ||
243 | |||
244 | return ret; | ||
172 | } | 245 | } |
173 | EXPORT_SYMBOL_GPL(iommu_map); | 246 | EXPORT_SYMBOL_GPL(iommu_map); |
174 | 247 | ||
175 | int iommu_unmap(struct iommu_domain *domain, unsigned long iova, int gfp_order) | 248 | size_t iommu_unmap(struct iommu_domain *domain, unsigned long iova, size_t size) |
176 | { | 249 | { |
177 | size_t size, unmapped; | 250 | size_t unmapped_page, unmapped = 0; |
251 | unsigned int min_pagesz; | ||
178 | 252 | ||
179 | if (unlikely(domain->ops->unmap == NULL)) | 253 | if (unlikely(domain->ops->unmap == NULL)) |
180 | return -ENODEV; | 254 | return -ENODEV; |
181 | 255 | ||
182 | size = PAGE_SIZE << gfp_order; | 256 | /* find out the minimum page size supported */ |
183 | 257 | min_pagesz = 1 << __ffs(domain->ops->pgsize_bitmap); | |
184 | BUG_ON(!IS_ALIGNED(iova, size)); | 258 | |
185 | 259 | /* | |
186 | unmapped = domain->ops->unmap(domain, iova, size); | 260 | * The virtual address, as well as the size of the mapping, must be |
187 | 261 | * aligned (at least) to the size of the smallest page supported | |
188 | return get_order(unmapped); | 262 | * by the hardware |
263 | */ | ||
264 | if (!IS_ALIGNED(iova | size, min_pagesz)) { | ||
265 | pr_err("unaligned: iova 0x%lx size 0x%lx min_pagesz 0x%x\n", | ||
266 | iova, (unsigned long)size, min_pagesz); | ||
267 | return -EINVAL; | ||
268 | } | ||
269 | |||
270 | pr_debug("unmap this: iova 0x%lx size 0x%lx\n", iova, | ||
271 | (unsigned long)size); | ||
272 | |||
273 | /* | ||
274 | * Keep iterating until we either unmap 'size' bytes (or more) | ||
275 | * or we hit an area that isn't mapped. | ||
276 | */ | ||
277 | while (unmapped < size) { | ||
278 | size_t left = size - unmapped; | ||
279 | |||
280 | unmapped_page = domain->ops->unmap(domain, iova, left); | ||
281 | if (!unmapped_page) | ||
282 | break; | ||
283 | |||
284 | pr_debug("unmapped: iova 0x%lx size %lx\n", iova, | ||
285 | (unsigned long)unmapped_page); | ||
286 | |||
287 | iova += unmapped_page; | ||
288 | unmapped += unmapped_page; | ||
289 | } | ||
290 | |||
291 | return unmapped; | ||
189 | } | 292 | } |
190 | EXPORT_SYMBOL_GPL(iommu_unmap); | 293 | EXPORT_SYMBOL_GPL(iommu_unmap); |