Commit cdcd7f72 authored by Charles Keepax's avatar Charles Keepax Committed by Mark Brown

ASoC: wm_adsp: Use vmalloc to allocate firmware download buffer

Use vmalloc to allocate the buffer for firmware/coefficient download and
rely on the SPI core to split this up into DMA-able chunks. This should
give better performance and means we no longer need to manually split
the download into page size chunks to avoid allocating overly large
continuous memory regions.
Signed-off-by: default avatarCharles Keepax <ckeepax@opensource.wolfsonmicro.com>
Signed-off-by: default avatarMark Brown <broonie@kernel.org>
parent f114040e
...@@ -21,6 +21,7 @@ ...@@ -21,6 +21,7 @@
#include <linux/regmap.h> #include <linux/regmap.h>
#include <linux/regulator/consumer.h> #include <linux/regulator/consumer.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/vmalloc.h>
#include <linux/workqueue.h> #include <linux/workqueue.h>
#include <sound/core.h> #include <sound/core.h>
#include <sound/pcm.h> #include <sound/pcm.h>
...@@ -169,11 +170,12 @@ static struct wm_adsp_buf *wm_adsp_buf_alloc(const void *src, size_t len, ...@@ -169,11 +170,12 @@ static struct wm_adsp_buf *wm_adsp_buf_alloc(const void *src, size_t len,
if (buf == NULL) if (buf == NULL)
return NULL; return NULL;
buf->buf = kmemdup(src, len, GFP_KERNEL | GFP_DMA); buf->buf = vmalloc(len);
if (!buf->buf) { if (!buf->buf) {
kfree(buf); vfree(buf);
return NULL; return NULL;
} }
memcpy(buf->buf, src, len);
if (list) if (list)
list_add_tail(&buf->list, list); list_add_tail(&buf->list, list);
...@@ -188,7 +190,7 @@ static void wm_adsp_buf_free(struct list_head *list) ...@@ -188,7 +190,7 @@ static void wm_adsp_buf_free(struct list_head *list)
struct wm_adsp_buf, struct wm_adsp_buf,
list); list);
list_del(&buf->list); list_del(&buf->list);
kfree(buf->buf); vfree(buf->buf);
kfree(buf); kfree(buf);
} }
} }
...@@ -684,16 +686,8 @@ static int wm_adsp_load(struct wm_adsp *dsp) ...@@ -684,16 +686,8 @@ static int wm_adsp_load(struct wm_adsp *dsp)
} }
if (reg) { if (reg) {
size_t to_write = PAGE_SIZE; buf = wm_adsp_buf_alloc(region->data,
size_t remain = le32_to_cpu(region->len); le32_to_cpu(region->len),
const u8 *data = region->data;
while (remain > 0) {
if (remain < PAGE_SIZE)
to_write = remain;
buf = wm_adsp_buf_alloc(data,
to_write,
&buf_list); &buf_list);
if (!buf) { if (!buf) {
adsp_err(dsp, "Out of memory\n"); adsp_err(dsp, "Out of memory\n");
...@@ -701,22 +695,16 @@ static int wm_adsp_load(struct wm_adsp *dsp) ...@@ -701,22 +695,16 @@ static int wm_adsp_load(struct wm_adsp *dsp)
goto out_fw; goto out_fw;
} }
ret = regmap_raw_write_async(regmap, reg, ret = regmap_raw_write_async(regmap, reg, buf->buf,
buf->buf, le32_to_cpu(region->len));
to_write);
if (ret != 0) { if (ret != 0) {
adsp_err(dsp, adsp_err(dsp,
"%s.%d: Failed to write %zd bytes at %d in %s: %d\n", "%s.%d: Failed to write %d bytes at %d in %s: %d\n",
file, regions, file, regions,
to_write, offset, le32_to_cpu(region->len), offset,
region_name, ret); region_name, ret);
goto out_fw; goto out_fw;
} }
data += to_write;
reg += to_write / 2;
remain -= to_write;
}
} }
pos += le32_to_cpu(region->len) + sizeof(*region); pos += le32_to_cpu(region->len) + sizeof(*region);
......
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