diff options
author | Jan Beulich <jbeulich@novell.com> | 2007-02-05 21:46:40 -0500 |
---|---|---|
committer | Tony Luck <tony.luck@intel.com> | 2007-02-05 21:46:40 -0500 |
commit | cde14bbfb3aa79b479db35bd29e6c083513d8614 (patch) | |
tree | 68b2d66d1eee3067051f4a6e4df8ace461bf440f /lib | |
parent | 86afa9eb88af2248bcc91d5b3568c63fdea65d6c (diff) |
[IA64] swiotlb bug fixes
This patch fixes
- marking I-cache clean of pages DMAed to now only done for IA64
- broken multiple inclusion in include/asm-x86_64/swiotlb.h
- missing call to mark_clean in swiotlb_sync_sg()
- a (perhaps only theoretical) issue in swiotlb_dma_supported() when
io_tlb_end is exactly at the end of memory
Signed-off-by: Jan Beulich <jbeulich@novell.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Tony Luck <tony.luck@intel.com>
Diffstat (limited to 'lib')
-rw-r--r-- | lib/swiotlb.c | 33 |
1 files changed, 8 insertions, 25 deletions
diff --git a/lib/swiotlb.c b/lib/swiotlb.c index 10625785eefd..34278338aad0 100644 --- a/lib/swiotlb.c +++ b/lib/swiotlb.c | |||
@@ -558,25 +558,6 @@ swiotlb_map_single(struct device *hwdev, void *ptr, size_t size, int dir) | |||
558 | } | 558 | } |
559 | 559 | ||
560 | /* | 560 | /* |
561 | * Since DMA is i-cache coherent, any (complete) pages that were written via | ||
562 | * DMA can be marked as "clean" so that lazy_mmu_prot_update() doesn't have to | ||
563 | * flush them when they get mapped into an executable vm-area. | ||
564 | */ | ||
565 | static void | ||
566 | mark_clean(void *addr, size_t size) | ||
567 | { | ||
568 | unsigned long pg_addr, end; | ||
569 | |||
570 | pg_addr = PAGE_ALIGN((unsigned long) addr); | ||
571 | end = (unsigned long) addr + size; | ||
572 | while (pg_addr + PAGE_SIZE <= end) { | ||
573 | struct page *page = virt_to_page(pg_addr); | ||
574 | set_bit(PG_arch_1, &page->flags); | ||
575 | pg_addr += PAGE_SIZE; | ||
576 | } | ||
577 | } | ||
578 | |||
579 | /* | ||
580 | * Unmap a single streaming mode DMA translation. The dma_addr and size must | 561 | * Unmap a single streaming mode DMA translation. The dma_addr and size must |
581 | * match what was provided for in a previous swiotlb_map_single call. All | 562 | * match what was provided for in a previous swiotlb_map_single call. All |
582 | * other usages are undefined. | 563 | * other usages are undefined. |
@@ -594,7 +575,7 @@ swiotlb_unmap_single(struct device *hwdev, dma_addr_t dev_addr, size_t size, | |||
594 | if (dma_addr >= io_tlb_start && dma_addr < io_tlb_end) | 575 | if (dma_addr >= io_tlb_start && dma_addr < io_tlb_end) |
595 | unmap_single(hwdev, dma_addr, size, dir); | 576 | unmap_single(hwdev, dma_addr, size, dir); |
596 | else if (dir == DMA_FROM_DEVICE) | 577 | else if (dir == DMA_FROM_DEVICE) |
597 | mark_clean(dma_addr, size); | 578 | dma_mark_clean(dma_addr, size); |
598 | } | 579 | } |
599 | 580 | ||
600 | /* | 581 | /* |
@@ -617,7 +598,7 @@ swiotlb_sync_single(struct device *hwdev, dma_addr_t dev_addr, | |||
617 | if (dma_addr >= io_tlb_start && dma_addr < io_tlb_end) | 598 | if (dma_addr >= io_tlb_start && dma_addr < io_tlb_end) |
618 | sync_single(hwdev, dma_addr, size, dir, target); | 599 | sync_single(hwdev, dma_addr, size, dir, target); |
619 | else if (dir == DMA_FROM_DEVICE) | 600 | else if (dir == DMA_FROM_DEVICE) |
620 | mark_clean(dma_addr, size); | 601 | dma_mark_clean(dma_addr, size); |
621 | } | 602 | } |
622 | 603 | ||
623 | void | 604 | void |
@@ -648,7 +629,7 @@ swiotlb_sync_single_range(struct device *hwdev, dma_addr_t dev_addr, | |||
648 | if (dma_addr >= io_tlb_start && dma_addr < io_tlb_end) | 629 | if (dma_addr >= io_tlb_start && dma_addr < io_tlb_end) |
649 | sync_single(hwdev, dma_addr, size, dir, target); | 630 | sync_single(hwdev, dma_addr, size, dir, target); |
650 | else if (dir == DMA_FROM_DEVICE) | 631 | else if (dir == DMA_FROM_DEVICE) |
651 | mark_clean(dma_addr, size); | 632 | dma_mark_clean(dma_addr, size); |
652 | } | 633 | } |
653 | 634 | ||
654 | void | 635 | void |
@@ -698,7 +679,6 @@ swiotlb_map_sg(struct device *hwdev, struct scatterlist *sg, int nelems, | |||
698 | dev_addr = virt_to_phys(addr); | 679 | dev_addr = virt_to_phys(addr); |
699 | if (swiotlb_force || address_needs_mapping(hwdev, dev_addr)) { | 680 | if (swiotlb_force || address_needs_mapping(hwdev, dev_addr)) { |
700 | void *map = map_single(hwdev, addr, sg->length, dir); | 681 | void *map = map_single(hwdev, addr, sg->length, dir); |
701 | sg->dma_address = virt_to_bus(map); | ||
702 | if (!map) { | 682 | if (!map) { |
703 | /* Don't panic here, we expect map_sg users | 683 | /* Don't panic here, we expect map_sg users |
704 | to do proper error handling. */ | 684 | to do proper error handling. */ |
@@ -707,6 +687,7 @@ swiotlb_map_sg(struct device *hwdev, struct scatterlist *sg, int nelems, | |||
707 | sg[0].dma_length = 0; | 687 | sg[0].dma_length = 0; |
708 | return 0; | 688 | return 0; |
709 | } | 689 | } |
690 | sg->dma_address = virt_to_bus(map); | ||
710 | } else | 691 | } else |
711 | sg->dma_address = dev_addr; | 692 | sg->dma_address = dev_addr; |
712 | sg->dma_length = sg->length; | 693 | sg->dma_length = sg->length; |
@@ -730,7 +711,7 @@ swiotlb_unmap_sg(struct device *hwdev, struct scatterlist *sg, int nelems, | |||
730 | if (sg->dma_address != SG_ENT_PHYS_ADDRESS(sg)) | 711 | if (sg->dma_address != SG_ENT_PHYS_ADDRESS(sg)) |
731 | unmap_single(hwdev, (void *) phys_to_virt(sg->dma_address), sg->dma_length, dir); | 712 | unmap_single(hwdev, (void *) phys_to_virt(sg->dma_address), sg->dma_length, dir); |
732 | else if (dir == DMA_FROM_DEVICE) | 713 | else if (dir == DMA_FROM_DEVICE) |
733 | mark_clean(SG_ENT_VIRT_ADDRESS(sg), sg->dma_length); | 714 | dma_mark_clean(SG_ENT_VIRT_ADDRESS(sg), sg->dma_length); |
734 | } | 715 | } |
735 | 716 | ||
736 | /* | 717 | /* |
@@ -752,6 +733,8 @@ swiotlb_sync_sg(struct device *hwdev, struct scatterlist *sg, | |||
752 | if (sg->dma_address != SG_ENT_PHYS_ADDRESS(sg)) | 733 | if (sg->dma_address != SG_ENT_PHYS_ADDRESS(sg)) |
753 | sync_single(hwdev, (void *) sg->dma_address, | 734 | sync_single(hwdev, (void *) sg->dma_address, |
754 | sg->dma_length, dir, target); | 735 | sg->dma_length, dir, target); |
736 | else if (dir == DMA_FROM_DEVICE) | ||
737 | dma_mark_clean(SG_ENT_VIRT_ADDRESS(sg), sg->dma_length); | ||
755 | } | 738 | } |
756 | 739 | ||
757 | void | 740 | void |
@@ -783,7 +766,7 @@ swiotlb_dma_mapping_error(dma_addr_t dma_addr) | |||
783 | int | 766 | int |
784 | swiotlb_dma_supported (struct device *hwdev, u64 mask) | 767 | swiotlb_dma_supported (struct device *hwdev, u64 mask) |
785 | { | 768 | { |
786 | return (virt_to_phys (io_tlb_end) - 1) <= mask; | 769 | return virt_to_phys(io_tlb_end - 1) <= mask; |
787 | } | 770 | } |
788 | 771 | ||
789 | EXPORT_SYMBOL(swiotlb_init); | 772 | EXPORT_SYMBOL(swiotlb_init); |