Skip to content

Commit

Permalink
ASoC: wm_adsp: Detach compressed stream on free
Browse files Browse the repository at this point in the history
If someone powers down the DSP core (through routing changes
say) whilst a compressed record is in progress we can end up
using a freed pointer to the buffer object. When a compressed
audio stream is triggered we attach it to a buffer on a physical
DSP. This patch adds a detach of the buffer from the stream when
the stream is freed or when the DSP is powered down which avoids
the situation where we use a buffer when it is no longer valid.

Signed-off-by: Charles Keepax <ckeepax@opensource.wolfsonmicro.com>
Signed-off-by: Mark Brown <broonie@kernel.org>
  • Loading branch information
Charles Keepax authored and Mark Brown committed May 4, 2016
1 parent edd7135 commit 721be3b
Showing 1 changed file with 22 additions and 0 deletions.
22 changes: 22 additions & 0 deletions sound/soc/codecs/wm_adsp.c
Original file line number Diff line number Diff line change
Expand Up @@ -273,8 +273,11 @@ struct wm_adsp_buffer {
__be32 words_written[2]; /* total words written (64 bit) */
};

struct wm_adsp_compr;

struct wm_adsp_compr_buf {
struct wm_adsp *dsp;
struct wm_adsp_compr *compr;

struct wm_adsp_buffer_region *regions;
u32 host_buf_ptr;
Expand Down Expand Up @@ -2467,10 +2470,26 @@ static int wm_adsp_compr_attach(struct wm_adsp_compr *compr)
return -EINVAL;

compr->buf = compr->dsp->buffer;
compr->buf->compr = compr;

return 0;
}

static void wm_adsp_compr_detach(struct wm_adsp_compr *compr)
{
if (!compr)
return;

/* Wake the poll so it can see buffer is no longer attached */
if (compr->stream)
snd_compr_fragment_elapsed(compr->stream);

if (wm_adsp_compr_attached(compr)) {
compr->buf->compr = NULL;
compr->buf = NULL;
}
}

int wm_adsp_compr_open(struct wm_adsp *dsp, struct snd_compr_stream *stream)
{
struct wm_adsp_compr *compr;
Expand Down Expand Up @@ -2524,6 +2543,7 @@ int wm_adsp_compr_free(struct snd_compr_stream *stream)

mutex_lock(&dsp->pwr_lock);

wm_adsp_compr_detach(compr);
dsp->compr = NULL;

kfree(compr->raw_buf);
Expand Down Expand Up @@ -2820,6 +2840,8 @@ static int wm_adsp_buffer_init(struct wm_adsp *dsp)
static int wm_adsp_buffer_free(struct wm_adsp *dsp)
{
if (dsp->buffer) {
wm_adsp_compr_detach(dsp->buffer->compr);

kfree(dsp->buffer->regions);
kfree(dsp->buffer);

Expand Down

0 comments on commit 721be3b

Please sign in to comment.