aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorChristoph Hellwig <hch@lst.de>2017-12-23 14:56:02 -0500
committerChristoph Hellwig <hch@lst.de>2018-01-15 03:35:48 -0500
commita25381aa3ae60a2e028c95f1318649b13cbbce30 (patch)
tree33eb75af3f080f6f1e6a8210c59731cd11243273
parentaaf796dc6e84e809e4c791b6517326b26312c972 (diff)
swiotlb: refactor coherent buffer freeing
Factor out a new swiotlb_free_buffer helper that checks if an address is allocated from the swiotlb bounce buffer, and if yes frees it. This allows to simplify the swiotlb_free implemenation that uses dma_direct_free to free the non-bounce buffer allocations. Signed-off-by: Christoph Hellwig <hch@lst.de> Acked-by: Christian König <christian.koenig@amd.com>
-rw-r--r--lib/swiotlb.c35
1 files changed, 21 insertions, 14 deletions
diff --git a/lib/swiotlb.c b/lib/swiotlb.c
index 539fd1099ba9..1a147f1354a1 100644
--- a/lib/swiotlb.c
+++ b/lib/swiotlb.c
@@ -780,22 +780,31 @@ err_warn:
780} 780}
781EXPORT_SYMBOL(swiotlb_alloc_coherent); 781EXPORT_SYMBOL(swiotlb_alloc_coherent);
782 782
783static bool swiotlb_free_buffer(struct device *dev, size_t size,
784 dma_addr_t dma_addr)
785{
786 phys_addr_t phys_addr = dma_to_phys(dev, dma_addr);
787
788 WARN_ON_ONCE(irqs_disabled());
789
790 if (!is_swiotlb_buffer(phys_addr))
791 return false;
792
793 /*
794 * DMA_TO_DEVICE to avoid memcpy in swiotlb_tbl_unmap_single.
795 * DMA_ATTR_SKIP_CPU_SYNC is optional.
796 */
797 swiotlb_tbl_unmap_single(dev, phys_addr, size, DMA_TO_DEVICE,
798 DMA_ATTR_SKIP_CPU_SYNC);
799 return true;
800}
801
783void 802void
784swiotlb_free_coherent(struct device *hwdev, size_t size, void *vaddr, 803swiotlb_free_coherent(struct device *hwdev, size_t size, void *vaddr,
785 dma_addr_t dev_addr) 804 dma_addr_t dev_addr)
786{ 805{
787 phys_addr_t paddr = dma_to_phys(hwdev, dev_addr); 806 if (!swiotlb_free_buffer(hwdev, size, dev_addr))
788
789 WARN_ON(irqs_disabled());
790 if (!is_swiotlb_buffer(paddr))
791 free_pages((unsigned long)vaddr, get_order(size)); 807 free_pages((unsigned long)vaddr, get_order(size));
792 else
793 /*
794 * DMA_TO_DEVICE to avoid memcpy in swiotlb_tbl_unmap_single.
795 * DMA_ATTR_SKIP_CPU_SYNC is optional.
796 */
797 swiotlb_tbl_unmap_single(hwdev, paddr, size, DMA_TO_DEVICE,
798 DMA_ATTR_SKIP_CPU_SYNC);
799} 808}
800EXPORT_SYMBOL(swiotlb_free_coherent); 809EXPORT_SYMBOL(swiotlb_free_coherent);
801 810
@@ -1110,9 +1119,7 @@ void *swiotlb_alloc(struct device *dev, size_t size, dma_addr_t *dma_handle,
1110void swiotlb_free(struct device *dev, size_t size, void *vaddr, 1119void swiotlb_free(struct device *dev, size_t size, void *vaddr,
1111 dma_addr_t dma_addr, unsigned long attrs) 1120 dma_addr_t dma_addr, unsigned long attrs)
1112{ 1121{
1113 if (is_swiotlb_buffer(dma_to_phys(dev, dma_addr))) 1122 if (!swiotlb_free_buffer(dev, size, dma_addr))
1114 swiotlb_free_coherent(dev, size, vaddr, dma_addr);
1115 else
1116 dma_direct_free(dev, size, vaddr, dma_addr, attrs); 1123 dma_direct_free(dev, size, vaddr, dma_addr, attrs);
1117} 1124}
1118 1125