diff options
author | FUJITA Tomonori <fujita.tomonori@lab.ntt.co.jp> | 2008-09-12 06:42:35 -0400 |
---|---|---|
committer | Ingo Molnar <mingo@elte.hu> | 2008-09-14 10:42:37 -0400 |
commit | bee44f294efd8417f5e68553778a6cc957af1547 (patch) | |
tree | 765ab42985836390cb0e83403deb9ff59d1f59a8 | |
parent | 589fc9a6e2102b498978f6350581ec7fa5aeb032 (diff) |
x86: make GART to respect device's dma_mask about virtual mappings
Currently, GART IOMMU ingores device's dma_mask when it does virtual
mappings. So it could give a device a virtual address that the device
can't access to.
This patch fixes the above problem.
Signed-off-by: FUJITA Tomonori <fujita.tomonori@lab.ntt.co.jp>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
-rw-r--r-- | arch/x86/kernel/pci-gart_64.c | 39 |
1 files changed, 28 insertions, 11 deletions
diff --git a/arch/x86/kernel/pci-gart_64.c b/arch/x86/kernel/pci-gart_64.c index 1b0c412566e5..9972c42ac925 100644 --- a/arch/x86/kernel/pci-gart_64.c +++ b/arch/x86/kernel/pci-gart_64.c | |||
@@ -83,23 +83,34 @@ static unsigned long next_bit; /* protected by iommu_bitmap_lock */ | |||
83 | static int need_flush; /* global flush state. set for each gart wrap */ | 83 | static int need_flush; /* global flush state. set for each gart wrap */ |
84 | 84 | ||
85 | static unsigned long alloc_iommu(struct device *dev, int size, | 85 | static unsigned long alloc_iommu(struct device *dev, int size, |
86 | unsigned long align_mask) | 86 | unsigned long align_mask, u64 dma_mask) |
87 | { | 87 | { |
88 | unsigned long offset, flags; | 88 | unsigned long offset, flags; |
89 | unsigned long boundary_size; | 89 | unsigned long boundary_size; |
90 | unsigned long base_index; | 90 | unsigned long base_index; |
91 | unsigned long limit; | ||
91 | 92 | ||
92 | base_index = ALIGN(iommu_bus_base & dma_get_seg_boundary(dev), | 93 | base_index = ALIGN(iommu_bus_base & dma_get_seg_boundary(dev), |
93 | PAGE_SIZE) >> PAGE_SHIFT; | 94 | PAGE_SIZE) >> PAGE_SHIFT; |
94 | boundary_size = ALIGN((unsigned long long)dma_get_seg_boundary(dev) + 1, | 95 | boundary_size = ALIGN((unsigned long long)dma_get_seg_boundary(dev) + 1, |
95 | PAGE_SIZE) >> PAGE_SHIFT; | 96 | PAGE_SIZE) >> PAGE_SHIFT; |
96 | 97 | ||
98 | limit = iommu_device_max_index(iommu_pages, | ||
99 | DIV_ROUND_UP(iommu_bus_base, PAGE_SIZE), | ||
100 | dma_mask >> PAGE_SHIFT); | ||
101 | |||
97 | spin_lock_irqsave(&iommu_bitmap_lock, flags); | 102 | spin_lock_irqsave(&iommu_bitmap_lock, flags); |
98 | offset = iommu_area_alloc(iommu_gart_bitmap, iommu_pages, next_bit, | 103 | |
104 | if (limit <= next_bit) { | ||
105 | need_flush = 1; | ||
106 | next_bit = 0; | ||
107 | } | ||
108 | |||
109 | offset = iommu_area_alloc(iommu_gart_bitmap, limit, next_bit, | ||
99 | size, base_index, boundary_size, align_mask); | 110 | size, base_index, boundary_size, align_mask); |
100 | if (offset == -1) { | 111 | if (offset == -1 && next_bit) { |
101 | need_flush = 1; | 112 | need_flush = 1; |
102 | offset = iommu_area_alloc(iommu_gart_bitmap, iommu_pages, 0, | 113 | offset = iommu_area_alloc(iommu_gart_bitmap, limit, 0, |
103 | size, base_index, boundary_size, | 114 | size, base_index, boundary_size, |
104 | align_mask); | 115 | align_mask); |
105 | } | 116 | } |
@@ -228,12 +239,14 @@ nonforced_iommu(struct device *dev, unsigned long addr, size_t size) | |||
228 | * Caller needs to check if the iommu is needed and flush. | 239 | * Caller needs to check if the iommu is needed and flush. |
229 | */ | 240 | */ |
230 | static dma_addr_t dma_map_area(struct device *dev, dma_addr_t phys_mem, | 241 | static dma_addr_t dma_map_area(struct device *dev, dma_addr_t phys_mem, |
231 | size_t size, int dir, unsigned long align_mask) | 242 | size_t size, int dir, unsigned long align_mask, |
243 | u64 dma_mask) | ||
232 | { | 244 | { |
233 | unsigned long npages = iommu_num_pages(phys_mem, size); | 245 | unsigned long npages = iommu_num_pages(phys_mem, size); |
234 | unsigned long iommu_page = alloc_iommu(dev, npages, align_mask); | 246 | unsigned long iommu_page; |
235 | int i; | 247 | int i; |
236 | 248 | ||
249 | iommu_page = alloc_iommu(dev, npages, align_mask, dma_mask); | ||
237 | if (iommu_page == -1) { | 250 | if (iommu_page == -1) { |
238 | if (!nonforced_iommu(dev, phys_mem, size)) | 251 | if (!nonforced_iommu(dev, phys_mem, size)) |
239 | return phys_mem; | 252 | return phys_mem; |
@@ -263,7 +276,7 @@ gart_map_single(struct device *dev, phys_addr_t paddr, size_t size, int dir) | |||
263 | if (!need_iommu(dev, paddr, size)) | 276 | if (!need_iommu(dev, paddr, size)) |
264 | return paddr; | 277 | return paddr; |
265 | 278 | ||
266 | bus = dma_map_area(dev, paddr, size, dir, 0); | 279 | bus = dma_map_area(dev, paddr, size, dir, 0, dma_get_mask(dev)); |
267 | flush_gart(); | 280 | flush_gart(); |
268 | 281 | ||
269 | return bus; | 282 | return bus; |
@@ -314,6 +327,7 @@ static int dma_map_sg_nonforce(struct device *dev, struct scatterlist *sg, | |||
314 | { | 327 | { |
315 | struct scatterlist *s; | 328 | struct scatterlist *s; |
316 | int i; | 329 | int i; |
330 | u64 dma_mask = dma_get_mask(dev); | ||
317 | 331 | ||
318 | #ifdef CONFIG_IOMMU_DEBUG | 332 | #ifdef CONFIG_IOMMU_DEBUG |
319 | printk(KERN_DEBUG "dma_map_sg overflow\n"); | 333 | printk(KERN_DEBUG "dma_map_sg overflow\n"); |
@@ -323,7 +337,8 @@ static int dma_map_sg_nonforce(struct device *dev, struct scatterlist *sg, | |||
323 | unsigned long addr = sg_phys(s); | 337 | unsigned long addr = sg_phys(s); |
324 | 338 | ||
325 | if (nonforced_iommu(dev, addr, s->length)) { | 339 | if (nonforced_iommu(dev, addr, s->length)) { |
326 | addr = dma_map_area(dev, addr, s->length, dir, 0); | 340 | addr = dma_map_area(dev, addr, s->length, dir, 0, |
341 | dma_mask); | ||
327 | if (addr == bad_dma_address) { | 342 | if (addr == bad_dma_address) { |
328 | if (i > 0) | 343 | if (i > 0) |
329 | gart_unmap_sg(dev, sg, i, dir); | 344 | gart_unmap_sg(dev, sg, i, dir); |
@@ -345,14 +360,16 @@ static int __dma_map_cont(struct device *dev, struct scatterlist *start, | |||
345 | int nelems, struct scatterlist *sout, | 360 | int nelems, struct scatterlist *sout, |
346 | unsigned long pages) | 361 | unsigned long pages) |
347 | { | 362 | { |
348 | unsigned long iommu_start = alloc_iommu(dev, pages, 0); | 363 | unsigned long iommu_start; |
349 | unsigned long iommu_page = iommu_start; | 364 | unsigned long iommu_page; |
350 | struct scatterlist *s; | 365 | struct scatterlist *s; |
351 | int i; | 366 | int i; |
352 | 367 | ||
368 | iommu_start = alloc_iommu(dev, pages, 0, dma_get_mask(dev)); | ||
353 | if (iommu_start == -1) | 369 | if (iommu_start == -1) |
354 | return -1; | 370 | return -1; |
355 | 371 | ||
372 | iommu_page = iommu_start; | ||
356 | for_each_sg(start, s, nelems, i) { | 373 | for_each_sg(start, s, nelems, i) { |
357 | unsigned long pages, addr; | 374 | unsigned long pages, addr; |
358 | unsigned long phys_addr = s->dma_address; | 375 | unsigned long phys_addr = s->dma_address; |
@@ -497,7 +514,7 @@ gart_alloc_coherent(struct device *dev, size_t size, dma_addr_t *dma_addr, | |||
497 | align_mask = (1UL << get_order(size)) - 1; | 514 | align_mask = (1UL << get_order(size)) - 1; |
498 | 515 | ||
499 | *dma_addr = dma_map_area(dev, __pa(vaddr), size, DMA_BIDIRECTIONAL, | 516 | *dma_addr = dma_map_area(dev, __pa(vaddr), size, DMA_BIDIRECTIONAL, |
500 | align_mask); | 517 | align_mask, dma_mask); |
501 | flush_gart(); | 518 | flush_gart(); |
502 | 519 | ||
503 | if (*dma_addr != bad_dma_address) | 520 | if (*dma_addr != bad_dma_address) |