[ARM] Clean up dmabounce
Encapsulate pool data into dmabounce_pool. Only account successful allocations. Use dma_mapping_error(). Signed-off-by: Russell King <rmk+kernel@arm.linux.org.uk>
This commit is contained in:
parent
1d1fd66c45
commit
cb7610d018
|
@ -33,8 +33,8 @@
|
||||||
#include <asm/cacheflush.h>
|
#include <asm/cacheflush.h>
|
||||||
|
|
||||||
#undef DEBUG
|
#undef DEBUG
|
||||||
|
|
||||||
#undef STATS
|
#undef STATS
|
||||||
|
|
||||||
#ifdef STATS
|
#ifdef STATS
|
||||||
#define DO_STATS(X) do { X ; } while (0)
|
#define DO_STATS(X) do { X ; } while (0)
|
||||||
#else
|
#else
|
||||||
|
@ -52,26 +52,31 @@ struct safe_buffer {
|
||||||
int direction;
|
int direction;
|
||||||
|
|
||||||
/* safe buffer info */
|
/* safe buffer info */
|
||||||
struct dma_pool *pool;
|
struct dmabounce_pool *pool;
|
||||||
void *safe;
|
void *safe;
|
||||||
dma_addr_t safe_dma_addr;
|
dma_addr_t safe_dma_addr;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct dmabounce_pool {
|
||||||
|
unsigned long size;
|
||||||
|
struct dma_pool *pool;
|
||||||
|
#ifdef STATS
|
||||||
|
unsigned long allocs;
|
||||||
|
#endif
|
||||||
|
};
|
||||||
|
|
||||||
struct dmabounce_device_info {
|
struct dmabounce_device_info {
|
||||||
struct list_head node;
|
struct list_head node;
|
||||||
|
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
struct dma_pool *small_buffer_pool;
|
|
||||||
struct dma_pool *large_buffer_pool;
|
|
||||||
struct list_head safe_buffers;
|
struct list_head safe_buffers;
|
||||||
unsigned long small_buffer_size, large_buffer_size;
|
|
||||||
#ifdef STATS
|
#ifdef STATS
|
||||||
unsigned long sbp_allocs;
|
|
||||||
unsigned long lbp_allocs;
|
|
||||||
unsigned long total_allocs;
|
unsigned long total_allocs;
|
||||||
unsigned long map_op_count;
|
unsigned long map_op_count;
|
||||||
unsigned long bounce_count;
|
unsigned long bounce_count;
|
||||||
#endif
|
#endif
|
||||||
|
struct dmabounce_pool small;
|
||||||
|
struct dmabounce_pool large;
|
||||||
};
|
};
|
||||||
|
|
||||||
static LIST_HEAD(dmabounce_devs);
|
static LIST_HEAD(dmabounce_devs);
|
||||||
|
@ -82,9 +87,9 @@ static void print_alloc_stats(struct dmabounce_device_info *device_info)
|
||||||
printk(KERN_INFO
|
printk(KERN_INFO
|
||||||
"%s: dmabounce: sbp: %lu, lbp: %lu, other: %lu, total: %lu\n",
|
"%s: dmabounce: sbp: %lu, lbp: %lu, other: %lu, total: %lu\n",
|
||||||
device_info->dev->bus_id,
|
device_info->dev->bus_id,
|
||||||
device_info->sbp_allocs, device_info->lbp_allocs,
|
device_info->small.allocs, device_info->large.allocs,
|
||||||
device_info->total_allocs - device_info->sbp_allocs -
|
device_info->total_allocs - device_info->small.allocs -
|
||||||
device_info->lbp_allocs,
|
device_info->large.allocs,
|
||||||
device_info->total_allocs);
|
device_info->total_allocs);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
@ -106,18 +111,22 @@ find_dmabounce_dev(struct device *dev)
|
||||||
/* allocate a 'safe' buffer and keep track of it */
|
/* allocate a 'safe' buffer and keep track of it */
|
||||||
static inline struct safe_buffer *
|
static inline struct safe_buffer *
|
||||||
alloc_safe_buffer(struct dmabounce_device_info *device_info, void *ptr,
|
alloc_safe_buffer(struct dmabounce_device_info *device_info, void *ptr,
|
||||||
size_t size, enum dma_data_direction dir)
|
size_t size, enum dma_data_direction dir)
|
||||||
{
|
{
|
||||||
struct safe_buffer *buf;
|
struct safe_buffer *buf;
|
||||||
struct dma_pool *pool;
|
struct dmabounce_pool *pool;
|
||||||
struct device *dev = device_info->dev;
|
struct device *dev = device_info->dev;
|
||||||
void *safe;
|
|
||||||
dma_addr_t safe_dma_addr;
|
|
||||||
|
|
||||||
dev_dbg(dev, "%s(ptr=%p, size=%d, dir=%d)\n",
|
dev_dbg(dev, "%s(ptr=%p, size=%d, dir=%d)\n",
|
||||||
__func__, ptr, size, dir);
|
__func__, ptr, size, dir);
|
||||||
|
|
||||||
DO_STATS ( device_info->total_allocs++ );
|
if (size <= device_info->small.size) {
|
||||||
|
pool = &device_info->small;
|
||||||
|
} else if (size <= device_info->large.size) {
|
||||||
|
pool = &device_info->large;
|
||||||
|
} else {
|
||||||
|
pool = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
buf = kmalloc(sizeof(struct safe_buffer), GFP_ATOMIC);
|
buf = kmalloc(sizeof(struct safe_buffer), GFP_ATOMIC);
|
||||||
if (buf == NULL) {
|
if (buf == NULL) {
|
||||||
|
@ -125,41 +134,35 @@ alloc_safe_buffer(struct dmabounce_device_info *device_info, void *ptr,
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (size <= device_info->small_buffer_size) {
|
buf->ptr = ptr;
|
||||||
pool = device_info->small_buffer_pool;
|
buf->size = size;
|
||||||
safe = dma_pool_alloc(pool, GFP_ATOMIC, &safe_dma_addr);
|
buf->direction = dir;
|
||||||
|
buf->pool = pool;
|
||||||
|
|
||||||
DO_STATS ( device_info->sbp_allocs++ );
|
if (pool) {
|
||||||
} else if (size <= device_info->large_buffer_size) {
|
buf->safe = dma_pool_alloc(pool->pool, GFP_ATOMIC,
|
||||||
pool = device_info->large_buffer_pool;
|
&buf->safe_dma_addr);
|
||||||
safe = dma_pool_alloc(pool, GFP_ATOMIC, &safe_dma_addr);
|
|
||||||
|
|
||||||
DO_STATS ( device_info->lbp_allocs++ );
|
|
||||||
} else {
|
} else {
|
||||||
pool = NULL;
|
buf->safe = dma_alloc_coherent(dev, size, &buf->safe_dma_addr,
|
||||||
safe = dma_alloc_coherent(dev, size, &safe_dma_addr, GFP_ATOMIC);
|
GFP_ATOMIC);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (safe == NULL) {
|
if (buf->safe == NULL) {
|
||||||
dev_warn(device_info->dev,
|
dev_warn(dev,
|
||||||
"%s: could not alloc dma memory (size=%d)\n",
|
"%s: could not alloc dma memory (size=%d)\n",
|
||||||
__func__, size);
|
__func__, size);
|
||||||
kfree(buf);
|
kfree(buf);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef STATS
|
#ifdef STATS
|
||||||
|
if (pool)
|
||||||
|
pool->allocs++;
|
||||||
|
device_info->total_allocs++;
|
||||||
if (device_info->total_allocs % 1000 == 0)
|
if (device_info->total_allocs % 1000 == 0)
|
||||||
print_alloc_stats(device_info);
|
print_alloc_stats(device_info);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
buf->ptr = ptr;
|
|
||||||
buf->size = size;
|
|
||||||
buf->direction = dir;
|
|
||||||
buf->pool = pool;
|
|
||||||
buf->safe = safe;
|
|
||||||
buf->safe_dma_addr = safe_dma_addr;
|
|
||||||
|
|
||||||
list_add(&buf->node, &device_info->safe_buffers);
|
list_add(&buf->node, &device_info->safe_buffers);
|
||||||
|
|
||||||
return buf;
|
return buf;
|
||||||
|
@ -186,7 +189,7 @@ free_safe_buffer(struct dmabounce_device_info *device_info, struct safe_buffer *
|
||||||
list_del(&buf->node);
|
list_del(&buf->node);
|
||||||
|
|
||||||
if (buf->pool)
|
if (buf->pool)
|
||||||
dma_pool_free(buf->pool, buf->safe, buf->safe_dma_addr);
|
dma_pool_free(buf->pool->pool, buf->safe, buf->safe_dma_addr);
|
||||||
else
|
else
|
||||||
dma_free_coherent(device_info->dev, buf->size, buf->safe,
|
dma_free_coherent(device_info->dev, buf->size, buf->safe,
|
||||||
buf->safe_dma_addr);
|
buf->safe_dma_addr);
|
||||||
|
@ -197,12 +200,10 @@ free_safe_buffer(struct dmabounce_device_info *device_info, struct safe_buffer *
|
||||||
/* ************************************************** */
|
/* ************************************************** */
|
||||||
|
|
||||||
#ifdef STATS
|
#ifdef STATS
|
||||||
|
|
||||||
static void print_map_stats(struct dmabounce_device_info *device_info)
|
static void print_map_stats(struct dmabounce_device_info *device_info)
|
||||||
{
|
{
|
||||||
printk(KERN_INFO
|
dev_info(device_info->dev,
|
||||||
"%s: dmabounce: map_op_count=%lu, bounce_count=%lu\n",
|
"dmabounce: map_op_count=%lu, bounce_count=%lu\n",
|
||||||
device_info->dev->bus_id,
|
|
||||||
device_info->map_op_count, device_info->bounce_count);
|
device_info->map_op_count, device_info->bounce_count);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
@ -258,13 +259,13 @@ map_single(struct device *dev, void *ptr, size_t size,
|
||||||
__func__, ptr, buf->safe, size);
|
__func__, ptr, buf->safe, size);
|
||||||
memcpy(buf->safe, ptr, size);
|
memcpy(buf->safe, ptr, size);
|
||||||
}
|
}
|
||||||
consistent_sync(buf->safe, size, dir);
|
ptr = buf->safe;
|
||||||
|
|
||||||
dma_addr = buf->safe_dma_addr;
|
dma_addr = buf->safe_dma_addr;
|
||||||
} else {
|
|
||||||
consistent_sync(ptr, size, dir);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
consistent_sync(ptr, size, dir);
|
||||||
|
|
||||||
return dma_addr;
|
return dma_addr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -278,7 +279,7 @@ unmap_single(struct device *dev, dma_addr_t dma_addr, size_t size,
|
||||||
/*
|
/*
|
||||||
* Trying to unmap an invalid mapping
|
* Trying to unmap an invalid mapping
|
||||||
*/
|
*/
|
||||||
if (dma_addr == ~0) {
|
if (dma_mapping_error(dma_addr)) {
|
||||||
dev_err(dev, "Trying to unmap invalid mapping\n");
|
dev_err(dev, "Trying to unmap invalid mapping\n");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -570,11 +571,25 @@ dma_sync_sg_for_device(struct device *dev, struct scatterlist *sg, int nents,
|
||||||
local_irq_restore(flags);
|
local_irq_restore(flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
dmabounce_init_pool(struct dmabounce_pool *pool, struct device *dev, const char *name,
|
||||||
|
unsigned long size)
|
||||||
|
{
|
||||||
|
pool->size = size;
|
||||||
|
DO_STATS(pool->allocs = 0);
|
||||||
|
pool->pool = dma_pool_create(name, dev, size,
|
||||||
|
0 /* byte alignment */,
|
||||||
|
0 /* no page-crossing issues */);
|
||||||
|
|
||||||
|
return pool->pool ? 0 : -ENOMEM;
|
||||||
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
dmabounce_register_dev(struct device *dev, unsigned long small_buffer_size,
|
dmabounce_register_dev(struct device *dev, unsigned long small_buffer_size,
|
||||||
unsigned long large_buffer_size)
|
unsigned long large_buffer_size)
|
||||||
{
|
{
|
||||||
struct dmabounce_device_info *device_info;
|
struct dmabounce_device_info *device_info;
|
||||||
|
int ret;
|
||||||
|
|
||||||
device_info = kmalloc(sizeof(struct dmabounce_device_info), GFP_ATOMIC);
|
device_info = kmalloc(sizeof(struct dmabounce_device_info), GFP_ATOMIC);
|
||||||
if (!device_info) {
|
if (!device_info) {
|
||||||
|
@ -584,45 +599,31 @@ dmabounce_register_dev(struct device *dev, unsigned long small_buffer_size,
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
device_info->small_buffer_pool =
|
ret = dmabounce_init_pool(&device_info->small, dev,
|
||||||
dma_pool_create("small_dmabounce_pool",
|
"small_dmabounce_pool", small_buffer_size);
|
||||||
dev,
|
if (ret) {
|
||||||
small_buffer_size,
|
dev_err(dev,
|
||||||
0 /* byte alignment */,
|
"dmabounce: could not allocate DMA pool for %ld byte objects\n",
|
||||||
0 /* no page-crossing issues */);
|
small_buffer_size);
|
||||||
if (!device_info->small_buffer_pool) {
|
goto err_free;
|
||||||
printk(KERN_ERR
|
|
||||||
"dmabounce: could not allocate small DMA pool for %s\n",
|
|
||||||
dev->bus_id);
|
|
||||||
kfree(device_info);
|
|
||||||
return -ENOMEM;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (large_buffer_size) {
|
if (large_buffer_size) {
|
||||||
device_info->large_buffer_pool =
|
ret = dmabounce_init_pool(&device_info->large, dev,
|
||||||
dma_pool_create("large_dmabounce_pool",
|
"large_dmabounce_pool",
|
||||||
dev,
|
large_buffer_size);
|
||||||
large_buffer_size,
|
if (ret) {
|
||||||
0 /* byte alignment */,
|
dev_err(dev,
|
||||||
0 /* no page-crossing issues */);
|
"dmabounce: could not allocate DMA pool for %ld byte objects\n",
|
||||||
if (!device_info->large_buffer_pool) {
|
large_buffer_size);
|
||||||
printk(KERN_ERR
|
goto err_destroy;
|
||||||
"dmabounce: could not allocate large DMA pool for %s\n",
|
|
||||||
dev->bus_id);
|
|
||||||
dma_pool_destroy(device_info->small_buffer_pool);
|
|
||||||
|
|
||||||
return -ENOMEM;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
device_info->dev = dev;
|
device_info->dev = dev;
|
||||||
device_info->small_buffer_size = small_buffer_size;
|
|
||||||
device_info->large_buffer_size = large_buffer_size;
|
|
||||||
INIT_LIST_HEAD(&device_info->safe_buffers);
|
INIT_LIST_HEAD(&device_info->safe_buffers);
|
||||||
|
|
||||||
#ifdef STATS
|
#ifdef STATS
|
||||||
device_info->sbp_allocs = 0;
|
|
||||||
device_info->lbp_allocs = 0;
|
|
||||||
device_info->total_allocs = 0;
|
device_info->total_allocs = 0;
|
||||||
device_info->map_op_count = 0;
|
device_info->map_op_count = 0;
|
||||||
device_info->bounce_count = 0;
|
device_info->bounce_count = 0;
|
||||||
|
@ -634,6 +635,12 @@ dmabounce_register_dev(struct device *dev, unsigned long small_buffer_size,
|
||||||
dev->bus_id, dev->bus->name);
|
dev->bus_id, dev->bus->name);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
err_destroy:
|
||||||
|
dma_pool_destroy(device_info->small.pool);
|
||||||
|
err_free:
|
||||||
|
kfree(device_info);
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
@ -655,10 +662,10 @@ dmabounce_unregister_dev(struct device *dev)
|
||||||
BUG();
|
BUG();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (device_info->small_buffer_pool)
|
if (device_info->small.pool)
|
||||||
dma_pool_destroy(device_info->small_buffer_pool);
|
dma_pool_destroy(device_info->small.pool);
|
||||||
if (device_info->large_buffer_pool)
|
if (device_info->large.pool)
|
||||||
dma_pool_destroy(device_info->large_buffer_pool);
|
dma_pool_destroy(device_info->large.pool);
|
||||||
|
|
||||||
#ifdef STATS
|
#ifdef STATS
|
||||||
print_alloc_stats(device_info);
|
print_alloc_stats(device_info);
|
||||||
|
|
Loading…
Reference in New Issue