提交 a25381aa 编写于 作者: C Christoph Hellwig

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: NChristoph Hellwig <hch@lst.de>
Acked-by: NChristian König <christian.koenig@amd.com>
上级 aaf796dc
...@@ -780,22 +780,31 @@ swiotlb_alloc_coherent(struct device *hwdev, size_t size, ...@@ -780,22 +780,31 @@ swiotlb_alloc_coherent(struct device *hwdev, size_t size,
} }
EXPORT_SYMBOL(swiotlb_alloc_coherent); EXPORT_SYMBOL(swiotlb_alloc_coherent);
void static bool swiotlb_free_buffer(struct device *dev, size_t size,
swiotlb_free_coherent(struct device *hwdev, size_t size, void *vaddr, dma_addr_t dma_addr)
dma_addr_t dev_addr)
{ {
phys_addr_t paddr = dma_to_phys(hwdev, dev_addr); phys_addr_t phys_addr = dma_to_phys(dev, dma_addr);
WARN_ON_ONCE(irqs_disabled());
if (!is_swiotlb_buffer(phys_addr))
return false;
WARN_ON(irqs_disabled());
if (!is_swiotlb_buffer(paddr))
free_pages((unsigned long)vaddr, get_order(size));
else
/* /*
* DMA_TO_DEVICE to avoid memcpy in swiotlb_tbl_unmap_single. * DMA_TO_DEVICE to avoid memcpy in swiotlb_tbl_unmap_single.
* DMA_ATTR_SKIP_CPU_SYNC is optional. * DMA_ATTR_SKIP_CPU_SYNC is optional.
*/ */
swiotlb_tbl_unmap_single(hwdev, paddr, size, DMA_TO_DEVICE, swiotlb_tbl_unmap_single(dev, phys_addr, size, DMA_TO_DEVICE,
DMA_ATTR_SKIP_CPU_SYNC); DMA_ATTR_SKIP_CPU_SYNC);
return true;
}
void
swiotlb_free_coherent(struct device *hwdev, size_t size, void *vaddr,
dma_addr_t dev_addr)
{
if (!swiotlb_free_buffer(hwdev, size, dev_addr))
free_pages((unsigned long)vaddr, get_order(size));
} }
EXPORT_SYMBOL(swiotlb_free_coherent); EXPORT_SYMBOL(swiotlb_free_coherent);
...@@ -1110,9 +1119,7 @@ void *swiotlb_alloc(struct device *dev, size_t size, dma_addr_t *dma_handle, ...@@ -1110,9 +1119,7 @@ void *swiotlb_alloc(struct device *dev, size_t size, dma_addr_t *dma_handle,
void swiotlb_free(struct device *dev, size_t size, void *vaddr, void swiotlb_free(struct device *dev, size_t size, void *vaddr,
dma_addr_t dma_addr, unsigned long attrs) dma_addr_t dma_addr, unsigned long attrs)
{ {
if (is_swiotlb_buffer(dma_to_phys(dev, dma_addr))) if (!swiotlb_free_buffer(dev, size, dma_addr))
swiotlb_free_coherent(dev, size, vaddr, dma_addr);
else
dma_direct_free(dev, size, vaddr, dma_addr, attrs); dma_direct_free(dev, size, vaddr, dma_addr, attrs);
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册