Skip to content

Commit

Permalink
vchiq_2835_arm: Implement a DMA pool for small bulk transfers (#2699)
Browse files Browse the repository at this point in the history
During a bulk transfer we request a DMA allocation to hold the
scatter-gather list.  Most of the time, this allocation is small
(<< PAGE_SIZE), however it can be requested at a high enough frequency
to cause fragmentation and/or stress the CMA allocator (think time
spent in compaction here, or during allocations elsewhere).

Implement a pool to serve up small DMA allocations, falling back
to a coherent allocation if the request is greater than
VCHIQ_DMA_POOL_SIZE.

Signed-off-by: Oliver Gjoneski <ogjoneski@gmail.com>
  • Loading branch information
detule authored and popcornmix committed Dec 10, 2018
1 parent 9069025 commit 74da962
Showing 1 changed file with 33 additions and 7 deletions.
40 changes: 33 additions & 7 deletions drivers/staging/vc04_services/interface/vchiq_arm/vchiq_2835_arm.c
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,7 @@
#include <linux/interrupt.h>
#include <linux/pagemap.h>
#include <linux/dma-mapping.h>
#include <linux/dmapool.h>
#include <linux/io.h>
#include <linux/platform_device.h>
#include <linux/uaccess.h>
Expand All @@ -59,6 +60,8 @@
#define BELL0 0x00
#define BELL2 0x08

#define VCHIQ_DMA_POOL_SIZE PAGE_SIZE

struct vchiq_2835_state {
int inited;
VCHIQ_ARM_STATE_T arm_state;
Expand All @@ -68,6 +71,7 @@ struct vchiq_pagelist_info {
PAGELIST_T *pagelist;
size_t pagelist_buffer_size;
dma_addr_t dma_addr;
bool is_from_pool;
enum dma_data_direction dma_dir;
unsigned int num_pages;
unsigned int pages_need_release;
Expand All @@ -88,6 +92,7 @@ static void __iomem *g_regs;
* of 32.
*/
static unsigned int g_cache_line_size = 32;
static struct dma_pool *g_dma_pool;
static unsigned int g_fragments_size;
static char *g_fragments_base;
static char *g_free_fragments;
Expand Down Expand Up @@ -194,6 +199,14 @@ int vchiq_platform_init(struct platform_device *pdev, VCHIQ_STATE_T *state)
}

g_dev = dev;
g_dma_pool = dmam_pool_create("vchiq_scatter_pool", dev,
VCHIQ_DMA_POOL_SIZE, g_cache_line_size,
0);
if (!g_dma_pool) {
dev_err(dev, "failed to create dma pool");
return -ENOMEM;
}

vchiq_log_info(vchiq_arm_log_level,
"vchiq_init - done (slots %pK, phys %pad)",
vchiq_slot_zero, &slot_phys);
Expand Down Expand Up @@ -382,9 +395,14 @@ cleanup_pagelistinfo(struct vchiq_pagelist_info *pagelistinfo)
for (i = 0; i < pagelistinfo->num_pages; i++)
put_page(pagelistinfo->pages[i]);
}

dma_free_coherent(g_dev, pagelistinfo->pagelist_buffer_size,
pagelistinfo->pagelist, pagelistinfo->dma_addr);
if (pagelistinfo->is_from_pool) {
dma_pool_free(g_dma_pool, pagelistinfo->pagelist,
pagelistinfo->dma_addr);
} else {
dma_free_coherent(g_dev, pagelistinfo->pagelist_buffer_size,
pagelistinfo->pagelist,
pagelistinfo->dma_addr);
}
}

/* There is a potential problem with partial cache lines (pages?)
Expand All @@ -404,6 +422,7 @@ create_pagelist(char __user *buf, size_t count, unsigned short type)
u32 *addrs;
unsigned int num_pages, offset, i, k;
int actual_pages;
bool is_from_pool;
size_t pagelist_size;
struct scatterlist *scatterlist, *sg;
int dma_buffers;
Expand All @@ -421,10 +440,16 @@ create_pagelist(char __user *buf, size_t count, unsigned short type)
/* Allocate enough storage to hold the page pointers and the page
* list
*/
pagelist = dma_zalloc_coherent(g_dev,
pagelist_size,
&dma_addr,
GFP_KERNEL);
if (pagelist_size > VCHIQ_DMA_POOL_SIZE) {
pagelist = dma_zalloc_coherent(g_dev,
pagelist_size,
&dma_addr,
GFP_KERNEL);
is_from_pool = false;
} else {
pagelist = dma_pool_zalloc(g_dma_pool, GFP_KERNEL, &dma_addr);
is_from_pool = true;
}

vchiq_log_trace(vchiq_arm_log_level, "%s - %pK", __func__, pagelist);

Expand All @@ -445,6 +470,7 @@ create_pagelist(char __user *buf, size_t count, unsigned short type)
pagelistinfo->pagelist = pagelist;
pagelistinfo->pagelist_buffer_size = pagelist_size;
pagelistinfo->dma_addr = dma_addr;
pagelistinfo->is_from_pool = is_from_pool;
pagelistinfo->dma_dir = (type == PAGELIST_WRITE) ?
DMA_TO_DEVICE : DMA_FROM_DEVICE;
pagelistinfo->num_pages = num_pages;
Expand Down

0 comments on commit 74da962

Please sign in to comment.