summaryrefslogtreecommitdiff
path: root/arch/arm/common
diff options
context:
space:
mode:
Diffstat (limited to 'arch/arm/common')
-rw-r--r--arch/arm/common/dmabounce.c109
1 files changed, 73 insertions, 36 deletions
diff --git a/arch/arm/common/dmabounce.c b/arch/arm/common/dmabounce.c
index bd8348017da1..3b184d3cbcc3 100644
--- a/arch/arm/common/dmabounce.c
+++ b/arch/arm/common/dmabounce.c
@@ -31,6 +31,7 @@
#include <linux/dmapool.h>
#include <linux/list.h>
#include <linux/scatterlist.h>
+#include <linux/highmem.h>
#include <asm/cacheflush.h>
@@ -49,6 +50,8 @@ struct safe_buffer {
/* original request */
void *ptr;
+ struct page *page;
+ unsigned long offset;
size_t size;
int direction;
@@ -103,7 +106,8 @@ static DEVICE_ATTR(dmabounce_stats, 0400, dmabounce_show, NULL);
/* allocate a 'safe' buffer and keep track of it */
static inline struct safe_buffer *
alloc_safe_buffer(struct dmabounce_device_info *device_info, void *ptr,
- size_t size, enum dma_data_direction dir)
+ struct page *page, unsigned long offset, size_t size,
+ enum dma_data_direction dir)
{
struct safe_buffer *buf;
struct dmabounce_pool *pool;
@@ -128,6 +132,8 @@ alloc_safe_buffer(struct dmabounce_device_info *device_info, void *ptr,
}
buf->ptr = ptr;
+ buf->page = page;
+ buf->offset = offset;
buf->size = size;
buf->direction = dir;
buf->pool = pool;
@@ -219,7 +225,8 @@ static struct safe_buffer *find_safe_buffer_dev(struct device *dev,
return find_safe_buffer(dev->archdata.dmabounce, dma_addr);
}
-static inline dma_addr_t map_single(struct device *dev, void *ptr, size_t size,
+static inline dma_addr_t map_single_or_page(struct device *dev, void *ptr,
+ struct page *page, unsigned long offset, size_t size,
enum dma_data_direction dir)
{
struct dmabounce_device_info *device_info = dev->archdata.dmabounce;
@@ -229,46 +236,42 @@ static inline dma_addr_t map_single(struct device *dev, void *ptr, size_t size,
if (device_info)
DO_STATS ( device_info->map_op_count++ );
- dma_addr = virt_to_dma(dev, ptr);
-
- if (dev->dma_mask) {
- unsigned long mask = *dev->dma_mask;
- unsigned long limit;
-
- mask = (mask - 1) | mask;
- limit = (mask + 1) & ~mask;
- if (limit && size > limit) {
- dev_err(dev, "DMA mapping too big (requested %#x "
- "mask %#Lx)\n", size, *dev->dma_mask);
- return ~0;
- }
-
- /*
- * Figure out if we need to bounce from the DMA mask.
- */
- needs_bounce = (dma_addr | (dma_addr + size - 1)) & ~mask;
- }
+ if (page)
+ dma_addr = page_to_dma(dev, page) + offset;
+ else
+ dma_addr = virt_to_dma(dev, ptr);
if (device_info && (needs_bounce || dma_needs_bounce(dev, dma_addr, size))) {
struct safe_buffer *buf;
- buf = alloc_safe_buffer(device_info, ptr, size, dir);
+ buf = alloc_safe_buffer(device_info, ptr, page, offset, size, dir);
if (buf == 0) {
dev_err(dev, "%s: unable to map unsafe buffer %p!\n",
__func__, ptr);
return 0;
}
- dev_dbg(dev,
- "%s: unsafe buffer %p (dma=%#x) mapped to %p (dma=%#x)\n",
- __func__, buf->ptr, virt_to_dma(dev, buf->ptr),
- buf->safe, buf->safe_dma_addr);
+ if (buf->page)
+ dev_dbg(dev, "%s: unsafe buffer %p (dma=%#x) mapped "
+ "to %p (dma=%#x)\n", __func__,
+ page_address(buf->page),
+ page_to_dma(dev, buf->page),
+ buf->safe, buf->safe_dma_addr);
+ else
+ dev_dbg(dev, "%s: unsafe buffer %p (dma=%#x) mapped "
+ "to %p (dma=%#x)\n", __func__,
+ buf->ptr, virt_to_dma(dev, buf->ptr),
+ buf->safe, buf->safe_dma_addr);
if ((dir == DMA_TO_DEVICE) ||
(dir == DMA_BIDIRECTIONAL)) {
+ if (page)
+ ptr = kmap_atomic(page, KM_BOUNCE_READ) + offset;
dev_dbg(dev, "%s: copy unsafe %p to safe %p, size %d\n",
__func__, ptr, buf->safe, size);
memcpy(buf->safe, ptr, size);
+ if (page)
+ kunmap_atomic(ptr - offset, KM_BOUNCE_READ);
}
ptr = buf->safe;
@@ -278,12 +281,50 @@ static inline dma_addr_t map_single(struct device *dev, void *ptr, size_t size,
* We don't need to sync the DMA buffer since
* it was allocated via the coherent allocators.
*/
- __dma_single_cpu_to_dev(ptr, size, dir);
+ if (page)
+ __dma_page_cpu_to_dev(page, offset, size, dir);
+ else
+ __dma_single_cpu_to_dev(ptr, size, dir);
}
return dma_addr;
}
+static inline void unmap_page(struct device *dev, dma_addr_t dma_addr,
+ size_t size, enum dma_data_direction dir)
+{
+ struct safe_buffer *buf = find_safe_buffer_dev(dev, dma_addr, "unmap");
+
+ if (buf) {
+ BUG_ON(buf->size != size);
+ BUG_ON(buf->direction != dir);
+ BUG_ON(!buf->page);
+ BUG_ON(buf->ptr);
+
+ dev_dbg(dev,
+ "%s: unsafe buffer %p (dma=%#x) mapped to %p (dma=%#x)\n",
+ __func__, page_address(buf->page),
+ page_to_dma(dev, buf->page),
+ buf->safe, buf->safe_dma_addr);
+
+ DO_STATS(dev->archdata.dmabounce->bounce_count++);
+ if (dir == DMA_FROM_DEVICE || dir == DMA_BIDIRECTIONAL) {
+ void *ptr;
+ unsigned long phys;
+ ptr = kmap_atomic(buf->page, KM_BOUNCE_READ) + buf->offset;
+ phys = page_to_phys(buf->page) + buf->offset;
+ memcpy(ptr, buf->safe, size);
+ dmac_clean_range(ptr, ptr + size);
+ kunmap_atomic(ptr - buf->offset, KM_BOUNCE_READ);
+ outer_clean_range(phys, phys + size);
+ }
+ free_safe_buffer(dev->archdata.dmabounce, buf);
+ } else {
+ __dma_page_dev_to_cpu(dma_to_page(dev, dma_addr),
+ dma_addr & ~PAGE_MASK, size, dir);
+ }
+}
+
static inline void unmap_single(struct device *dev, dma_addr_t dma_addr,
size_t size, enum dma_data_direction dir)
{
@@ -292,6 +333,8 @@ static inline void unmap_single(struct device *dev, dma_addr_t dma_addr,
if (buf) {
BUG_ON(buf->size != size);
BUG_ON(buf->direction != dir);
+ BUG_ON(buf->page);
+ BUG_ON(!buf->ptr);
dev_dbg(dev,
"%s: unsafe buffer %p (dma=%#x) mapped to %p (dma=%#x)\n",
@@ -341,7 +384,7 @@ dma_addr_t dma_map_single(struct device *dev, void *ptr, size_t size,
BUG_ON(!valid_dma_direction(dir));
- return map_single(dev, ptr, size, dir);
+ return map_single_or_page(dev, ptr, NULL, 0, size, dir);
}
EXPORT_SYMBOL(dma_map_single);
@@ -369,13 +412,7 @@ dma_addr_t dma_map_page(struct device *dev, struct page *page,
BUG_ON(!valid_dma_direction(dir));
- if (PageHighMem(page)) {
- dev_err(dev, "DMA buffer bouncing of HIGHMEM pages "
- "is not supported\n");
- return ~0;
- }
-
- return map_single(dev, page_address(page) + offset, size, dir);
+ return map_single_or_page(dev, NULL, page, offset, size, dir);
}
EXPORT_SYMBOL(dma_map_page);
@@ -391,7 +428,7 @@ void dma_unmap_page(struct device *dev, dma_addr_t dma_addr, size_t size,
dev_dbg(dev, "%s(ptr=%p,size=%d,dir=%x)\n",
__func__, (void *) dma_addr, size, dir);
- unmap_single(dev, dma_addr, size, dir);
+ unmap_page(dev, dma_addr, size, dir);
}
EXPORT_SYMBOL(dma_unmap_page);