Commit 81d88ce5 authored by Christoph Hellwig's avatar Christoph Hellwig

dma-mapping: remove the {alloc,free}_noncoherent methods

It turns out allowing non-contigous allocations here was a rather bad
idea, as we'll now need to define ways to get the pages for mmaping
or dma_buf sharing.  Revert this change and stick to the original
concept.  A different API for the use case of non-contigous allocations
will be added back later.
Signed-off-by: default avatarChristoph Hellwig <hch@lst.de>
Reviewed-by: default avatarTomasz Figa <tfiga@chromium.org>
Tested-by: Ricardo Ribalda <ribalda@chromium.org>:wq
parent 9dc00b25
...@@ -528,16 +528,14 @@ an I/O device, you should not be using this part of the API. ...@@ -528,16 +528,14 @@ an I/O device, you should not be using this part of the API.
:: ::
void * struct page *
dma_alloc_noncoherent(struct device *dev, size_t size, dma_alloc_pages(struct device *dev, size_t size, dma_addr_t *dma_handle,
dma_addr_t *dma_handle, enum dma_data_direction dir, enum dma_data_direction dir, gfp_t gfp)
gfp_t gfp)
This routine allocates a region of <size> bytes of consistent memory. It This routine allocates a region of <size> bytes of non-coherent memory. It
returns a pointer to the allocated region (in the processor's virtual address returns a pointer to first struct page for the region, or NULL if the
space) or NULL if the allocation failed. The returned memory may or may not allocation failed. The resulting struct page can be used for everything a
be in the kernel direct mapping. Drivers must not call virt_to_page on struct page is suitable for.
the returned memory region.
It also returns a <dma_handle> which may be cast to an unsigned integer the It also returns a <dma_handle> which may be cast to an unsigned integer the
same width as the bus and given to the device as the DMA address base of same width as the bus and given to the device as the DMA address base of
...@@ -558,51 +556,33 @@ reused. ...@@ -558,51 +556,33 @@ reused.
:: ::
void void
dma_free_noncoherent(struct device *dev, size_t size, void *cpu_addr, dma_free_pages(struct device *dev, size_t size, struct page *page,
dma_addr_t dma_handle, enum dma_data_direction dir) dma_addr_t dma_handle, enum dma_data_direction dir)
Free a region of memory previously allocated using dma_alloc_noncoherent(). Free a region of memory previously allocated using dma_alloc_pages().
dev, size and dma_handle and dir must all be the same as those passed into dev, size, dma_handle and dir must all be the same as those passed into
dma_alloc_noncoherent(). cpu_addr must be the virtual address returned by dma_alloc_pages(). page must be the pointer returned by dma_alloc_pages().
dma_alloc_noncoherent().
:: ::
struct page * void *
dma_alloc_pages(struct device *dev, size_t size, dma_addr_t *dma_handle, dma_alloc_noncoherent(struct device *dev, size_t size,
enum dma_data_direction dir, gfp_t gfp) dma_addr_t *dma_handle, enum dma_data_direction dir,
gfp_t gfp)
This routine allocates a region of <size> bytes of non-coherent memory. It
returns a pointer to first struct page for the region, or NULL if the
allocation failed. The resulting struct page can be used for everything a
struct page is suitable for.
It also returns a <dma_handle> which may be cast to an unsigned integer the
same width as the bus and given to the device as the DMA address base of
the region.
The dir parameter specified if data is read and/or written by the device,
see dma_map_single() for details.
The gfp parameter allows the caller to specify the ``GFP_`` flags (see
kmalloc()) for the allocation, but rejects flags used to specify a memory
zone such as GFP_DMA or GFP_HIGHMEM.
Before giving the memory to the device, dma_sync_single_for_device() needs This routine is a convenient wrapper around dma_alloc_pages that returns the
to be called, and before reading memory written by the device, kernel virtual address for the allocated memory instead of the page structure.
dma_sync_single_for_cpu(), just like for streaming DMA mappings that are
reused.
:: ::
void void
dma_free_pages(struct device *dev, size_t size, struct page *page, dma_free_noncoherent(struct device *dev, size_t size, void *cpu_addr,
dma_addr_t dma_handle, enum dma_data_direction dir) dma_addr_t dma_handle, enum dma_data_direction dir)
Free a region of memory previously allocated using dma_alloc_pages(). Free a region of memory previously allocated using dma_alloc_noncoherent().
dev, size and dma_handle and dir must all be the same as those passed into dev, size, dma_handle and dir must all be the same as those passed into
dma_alloc_noncoherent(). page must be the pointer returned by dma_alloc_noncoherent(). cpu_addr must be the virtual address returned by
dma_alloc_pages(). dma_alloc_noncoherent().
:: ::
......
...@@ -1197,34 +1197,6 @@ static void *iommu_dma_alloc(struct device *dev, size_t size, ...@@ -1197,34 +1197,6 @@ static void *iommu_dma_alloc(struct device *dev, size_t size,
return cpu_addr; return cpu_addr;
} }
#ifdef CONFIG_DMA_REMAP
static void *iommu_dma_alloc_noncoherent(struct device *dev, size_t size,
dma_addr_t *handle, enum dma_data_direction dir, gfp_t gfp)
{
if (!gfpflags_allow_blocking(gfp)) {
struct page *page;
page = dma_common_alloc_pages(dev, size, handle, dir, gfp);
if (!page)
return NULL;
return page_address(page);
}
return iommu_dma_alloc_remap(dev, size, handle, gfp | __GFP_ZERO,
PAGE_KERNEL, 0);
}
static void iommu_dma_free_noncoherent(struct device *dev, size_t size,
void *cpu_addr, dma_addr_t handle, enum dma_data_direction dir)
{
__iommu_dma_unmap(dev, handle, size);
__iommu_dma_free(dev, size, cpu_addr);
}
#else
#define iommu_dma_alloc_noncoherent NULL
#define iommu_dma_free_noncoherent NULL
#endif /* CONFIG_DMA_REMAP */
static int iommu_dma_mmap(struct device *dev, struct vm_area_struct *vma, static int iommu_dma_mmap(struct device *dev, struct vm_area_struct *vma,
void *cpu_addr, dma_addr_t dma_addr, size_t size, void *cpu_addr, dma_addr_t dma_addr, size_t size,
unsigned long attrs) unsigned long attrs)
...@@ -1295,8 +1267,6 @@ static const struct dma_map_ops iommu_dma_ops = { ...@@ -1295,8 +1267,6 @@ static const struct dma_map_ops iommu_dma_ops = {
.free = iommu_dma_free, .free = iommu_dma_free,
.alloc_pages = dma_common_alloc_pages, .alloc_pages = dma_common_alloc_pages,
.free_pages = dma_common_free_pages, .free_pages = dma_common_free_pages,
.alloc_noncoherent = iommu_dma_alloc_noncoherent,
.free_noncoherent = iommu_dma_free_noncoherent,
.mmap = iommu_dma_mmap, .mmap = iommu_dma_mmap,
.get_sgtable = iommu_dma_get_sgtable, .get_sgtable = iommu_dma_get_sgtable,
.map_page = iommu_dma_map_page, .map_page = iommu_dma_map_page,
......
...@@ -22,11 +22,6 @@ struct dma_map_ops { ...@@ -22,11 +22,6 @@ struct dma_map_ops {
gfp_t gfp); gfp_t gfp);
void (*free_pages)(struct device *dev, size_t size, struct page *vaddr, void (*free_pages)(struct device *dev, size_t size, struct page *vaddr,
dma_addr_t dma_handle, enum dma_data_direction dir); dma_addr_t dma_handle, enum dma_data_direction dir);
void *(*alloc_noncoherent)(struct device *dev, size_t size,
dma_addr_t *dma_handle, enum dma_data_direction dir,
gfp_t gfp);
void (*free_noncoherent)(struct device *dev, size_t size, void *vaddr,
dma_addr_t dma_handle, enum dma_data_direction dir);
int (*mmap)(struct device *, struct vm_area_struct *, int (*mmap)(struct device *, struct vm_area_struct *,
void *, dma_addr_t, size_t, unsigned long attrs); void *, dma_addr_t, size_t, unsigned long attrs);
......
...@@ -263,10 +263,19 @@ struct page *dma_alloc_pages(struct device *dev, size_t size, ...@@ -263,10 +263,19 @@ struct page *dma_alloc_pages(struct device *dev, size_t size,
dma_addr_t *dma_handle, enum dma_data_direction dir, gfp_t gfp); dma_addr_t *dma_handle, enum dma_data_direction dir, gfp_t gfp);
void dma_free_pages(struct device *dev, size_t size, struct page *page, void dma_free_pages(struct device *dev, size_t size, struct page *page,
dma_addr_t dma_handle, enum dma_data_direction dir); dma_addr_t dma_handle, enum dma_data_direction dir);
void *dma_alloc_noncoherent(struct device *dev, size_t size,
dma_addr_t *dma_handle, enum dma_data_direction dir, gfp_t gfp); static inline void *dma_alloc_noncoherent(struct device *dev, size_t size,
void dma_free_noncoherent(struct device *dev, size_t size, void *vaddr, dma_addr_t *dma_handle, enum dma_data_direction dir, gfp_t gfp)
dma_addr_t dma_handle, enum dma_data_direction dir); {
struct page *page = dma_alloc_pages(dev, size, dma_handle, dir, gfp);
return page ? page_address(page) : NULL;
}
static inline void dma_free_noncoherent(struct device *dev, size_t size,
void *vaddr, dma_addr_t dma_handle, enum dma_data_direction dir)
{
dma_free_pages(dev, size, virt_to_page(vaddr), dma_handle, dir);
}
static inline dma_addr_t dma_map_single_attrs(struct device *dev, void *ptr, static inline dma_addr_t dma_map_single_attrs(struct device *dev, void *ptr,
size_t size, enum dma_data_direction dir, unsigned long attrs) size_t size, enum dma_data_direction dir, unsigned long attrs)
......
...@@ -515,46 +515,6 @@ void dma_free_pages(struct device *dev, size_t size, struct page *page, ...@@ -515,46 +515,6 @@ void dma_free_pages(struct device *dev, size_t size, struct page *page,
} }
EXPORT_SYMBOL_GPL(dma_free_pages); EXPORT_SYMBOL_GPL(dma_free_pages);
void *dma_alloc_noncoherent(struct device *dev, size_t size,
dma_addr_t *dma_handle, enum dma_data_direction dir, gfp_t gfp)
{
const struct dma_map_ops *ops = get_dma_ops(dev);
void *vaddr;
if (!ops || !ops->alloc_noncoherent) {
struct page *page;
page = dma_alloc_pages(dev, size, dma_handle, dir, gfp);
if (!page)
return NULL;
return page_address(page);
}
size = PAGE_ALIGN(size);
vaddr = ops->alloc_noncoherent(dev, size, dma_handle, dir, gfp);
if (vaddr)
debug_dma_map_page(dev, virt_to_page(vaddr), 0, size, dir,
*dma_handle);
return vaddr;
}
EXPORT_SYMBOL_GPL(dma_alloc_noncoherent);
void dma_free_noncoherent(struct device *dev, size_t size, void *vaddr,
dma_addr_t dma_handle, enum dma_data_direction dir)
{
const struct dma_map_ops *ops = get_dma_ops(dev);
if (!ops || !ops->free_noncoherent) {
dma_free_pages(dev, size, virt_to_page(vaddr), dma_handle, dir);
return;
}
size = PAGE_ALIGN(size);
debug_dma_unmap_page(dev, dma_handle, size, dir);
ops->free_noncoherent(dev, size, vaddr, dma_handle, dir);
}
EXPORT_SYMBOL_GPL(dma_free_noncoherent);
int dma_supported(struct device *dev, u64 mask) int dma_supported(struct device *dev, u64 mask)
{ {
const struct dma_map_ops *ops = get_dma_ops(dev); const struct dma_map_ops *ops = get_dma_ops(dev);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment