ASoC: Intel: Fix Haswell/Broadwell DSP page table creation.
Fix page table creation on Haswell and Broadwell to remove unsafe virt_to_phys mappings and use more portable SG buffer. Use audio buffer APIs to allocate DMA buffers. Signed-off-by: Liam Girdwood <liam.r.girdwood@linux.intel.com> Signed-off-by: Mark Brown <broonie@linaro.org>
This commit is contained in:
parent
84fbdd5861
commit
0b708c87f6
1 changed files with 31 additions and 27 deletions
|
@ -107,7 +107,7 @@ struct hsw_priv_data {
|
||||||
struct sst_hsw *hsw;
|
struct sst_hsw *hsw;
|
||||||
|
|
||||||
/* page tables */
|
/* page tables */
|
||||||
unsigned char *pcm_pg[HSW_PCM_COUNT][2];
|
struct snd_dma_buffer dmab[HSW_PCM_COUNT][2];
|
||||||
|
|
||||||
/* DAI data */
|
/* DAI data */
|
||||||
struct hsw_pcm_data pcm[HSW_PCM_COUNT];
|
struct hsw_pcm_data pcm[HSW_PCM_COUNT];
|
||||||
|
@ -273,28 +273,26 @@ static const struct snd_kcontrol_new hsw_volume_controls[] = {
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Create DMA buffer page table for DSP */
|
/* Create DMA buffer page table for DSP */
|
||||||
static int create_adsp_page_table(struct hsw_priv_data *pdata,
|
static int create_adsp_page_table(struct snd_pcm_substream *substream,
|
||||||
struct snd_soc_pcm_runtime *rtd,
|
struct hsw_priv_data *pdata, struct snd_soc_pcm_runtime *rtd,
|
||||||
unsigned char *dma_area, size_t size, int pcm, int stream)
|
unsigned char *dma_area, size_t size, int pcm)
|
||||||
{
|
{
|
||||||
int i, pages;
|
struct snd_dma_buffer *dmab = snd_pcm_get_dma_buf(substream);
|
||||||
|
int i, pages, stream = substream->stream;
|
||||||
|
|
||||||
if (size % PAGE_SIZE)
|
pages = snd_sgbuf_aligned_pages(size);
|
||||||
pages = (size / PAGE_SIZE) + 1;
|
|
||||||
else
|
|
||||||
pages = size / PAGE_SIZE;
|
|
||||||
|
|
||||||
dev_dbg(rtd->dev, "generating page table for %p size 0x%zu pages %d\n",
|
dev_dbg(rtd->dev, "generating page table for %p size 0x%zu pages %d\n",
|
||||||
dma_area, size, pages);
|
dma_area, size, pages);
|
||||||
|
|
||||||
for (i = 0; i < pages; i++) {
|
for (i = 0; i < pages; i++) {
|
||||||
u32 idx = (((i << 2) + i)) >> 1;
|
u32 idx = (((i << 2) + i)) >> 1;
|
||||||
u32 pfn = (virt_to_phys(dma_area + i * PAGE_SIZE)) >> PAGE_SHIFT;
|
u32 pfn = snd_sgbuf_get_addr(dmab, i * PAGE_SIZE) >> PAGE_SHIFT;
|
||||||
u32 *pg_table;
|
u32 *pg_table;
|
||||||
|
|
||||||
dev_dbg(rtd->dev, "pfn i %i idx %d pfn %x\n", i, idx, pfn);
|
dev_dbg(rtd->dev, "pfn i %i idx %d pfn %x\n", i, idx, pfn);
|
||||||
|
|
||||||
pg_table = (u32*)(pdata->pcm_pg[pcm][stream] + idx);
|
pg_table = (u32 *)(pdata->dmab[pcm][stream].area + idx);
|
||||||
|
|
||||||
if (i & 1)
|
if (i & 1)
|
||||||
*pg_table |= (pfn << 4);
|
*pg_table |= (pfn << 4);
|
||||||
|
@ -317,6 +315,7 @@ static int hsw_pcm_hw_params(struct snd_pcm_substream *substream,
|
||||||
struct sst_hsw *hsw = pdata->hsw;
|
struct sst_hsw *hsw = pdata->hsw;
|
||||||
struct sst_module *module_data;
|
struct sst_module *module_data;
|
||||||
struct sst_dsp *dsp;
|
struct sst_dsp *dsp;
|
||||||
|
struct snd_dma_buffer *dmab;
|
||||||
enum sst_hsw_stream_type stream_type;
|
enum sst_hsw_stream_type stream_type;
|
||||||
enum sst_hsw_stream_path_id path_id;
|
enum sst_hsw_stream_path_id path_id;
|
||||||
u32 rate, bits, map, pages, module_id;
|
u32 rate, bits, map, pages, module_id;
|
||||||
|
@ -416,8 +415,10 @@ static int hsw_pcm_hw_params(struct snd_pcm_substream *substream,
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = create_adsp_page_table(pdata, rtd, runtime->dma_area,
|
dmab = snd_pcm_get_dma_buf(substream);
|
||||||
runtime->dma_bytes, rtd->cpu_dai->id, substream->stream);
|
|
||||||
|
ret = create_adsp_page_table(substream, pdata, rtd, runtime->dma_area,
|
||||||
|
runtime->dma_bytes, rtd->cpu_dai->id);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
|
@ -430,9 +431,9 @@ static int hsw_pcm_hw_params(struct snd_pcm_substream *substream,
|
||||||
pages = runtime->dma_bytes / PAGE_SIZE;
|
pages = runtime->dma_bytes / PAGE_SIZE;
|
||||||
|
|
||||||
ret = sst_hsw_stream_buffer(hsw, pcm_data->stream,
|
ret = sst_hsw_stream_buffer(hsw, pcm_data->stream,
|
||||||
virt_to_phys(pdata->pcm_pg[rtd->cpu_dai->id][substream->stream]),
|
pdata->dmab[rtd->cpu_dai->id][substream->stream].addr,
|
||||||
pages, runtime->dma_bytes, 0,
|
pages, runtime->dma_bytes, 0,
|
||||||
(u32)(virt_to_phys(runtime->dma_area) >> PAGE_SHIFT));
|
snd_sgbuf_get_addr(dmab, 0) >> PAGE_SHIFT);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
dev_err(rtd->dev, "error: failed to set DMA buffer %d\n", ret);
|
dev_err(rtd->dev, "error: failed to set DMA buffer %d\n", ret);
|
||||||
return ret;
|
return ret;
|
||||||
|
@ -621,7 +622,7 @@ static struct snd_pcm_ops hsw_pcm_ops = {
|
||||||
.hw_free = hsw_pcm_hw_free,
|
.hw_free = hsw_pcm_hw_free,
|
||||||
.trigger = hsw_pcm_trigger,
|
.trigger = hsw_pcm_trigger,
|
||||||
.pointer = hsw_pcm_pointer,
|
.pointer = hsw_pcm_pointer,
|
||||||
.mmap = snd_pcm_lib_default_mmap,
|
.page = snd_pcm_sgbuf_ops_page,
|
||||||
};
|
};
|
||||||
|
|
||||||
static void hsw_pcm_free(struct snd_pcm *pcm)
|
static void hsw_pcm_free(struct snd_pcm *pcm)
|
||||||
|
@ -641,7 +642,7 @@ static int hsw_pcm_new(struct snd_soc_pcm_runtime *rtd)
|
||||||
if (pcm->streams[SNDRV_PCM_STREAM_PLAYBACK].substream ||
|
if (pcm->streams[SNDRV_PCM_STREAM_PLAYBACK].substream ||
|
||||||
pcm->streams[SNDRV_PCM_STREAM_CAPTURE].substream) {
|
pcm->streams[SNDRV_PCM_STREAM_CAPTURE].substream) {
|
||||||
ret = snd_pcm_lib_preallocate_pages_for_all(pcm,
|
ret = snd_pcm_lib_preallocate_pages_for_all(pcm,
|
||||||
SNDRV_DMA_TYPE_DEV,
|
SNDRV_DMA_TYPE_DEV_SG,
|
||||||
rtd->card->dev,
|
rtd->card->dev,
|
||||||
hsw_pcm_hardware.buffer_bytes_max,
|
hsw_pcm_hardware.buffer_bytes_max,
|
||||||
hsw_pcm_hardware.buffer_bytes_max);
|
hsw_pcm_hardware.buffer_bytes_max);
|
||||||
|
@ -742,7 +743,8 @@ static int hsw_pcm_probe(struct snd_soc_platform *platform)
|
||||||
{
|
{
|
||||||
struct sst_pdata *pdata = dev_get_platdata(platform->dev);
|
struct sst_pdata *pdata = dev_get_platdata(platform->dev);
|
||||||
struct hsw_priv_data *priv_data;
|
struct hsw_priv_data *priv_data;
|
||||||
int i;
|
struct device *dma_dev = pdata->dma_dev;
|
||||||
|
int i, ret = 0;
|
||||||
|
|
||||||
if (!pdata)
|
if (!pdata)
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
|
@ -758,15 +760,17 @@ static int hsw_pcm_probe(struct snd_soc_platform *platform)
|
||||||
|
|
||||||
/* playback */
|
/* playback */
|
||||||
if (hsw_dais[i].playback.channels_min) {
|
if (hsw_dais[i].playback.channels_min) {
|
||||||
priv_data->pcm_pg[i][0] = kzalloc(PAGE_SIZE, GFP_DMA);
|
ret = snd_dma_alloc_pages(SNDRV_DMA_TYPE_DEV, dma_dev,
|
||||||
if (priv_data->pcm_pg[i][0] == NULL)
|
PAGE_SIZE, &priv_data->dmab[i][0]);
|
||||||
|
if (ret < 0)
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* capture */
|
/* capture */
|
||||||
if (hsw_dais[i].capture.channels_min) {
|
if (hsw_dais[i].capture.channels_min) {
|
||||||
priv_data->pcm_pg[i][1] = kzalloc(PAGE_SIZE, GFP_DMA);
|
ret = snd_dma_alloc_pages(SNDRV_DMA_TYPE_DEV, dma_dev,
|
||||||
if (priv_data->pcm_pg[i][1] == NULL)
|
PAGE_SIZE, &priv_data->dmab[i][1]);
|
||||||
|
if (ret < 0)
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -776,11 +780,11 @@ static int hsw_pcm_probe(struct snd_soc_platform *platform)
|
||||||
err:
|
err:
|
||||||
for (;i >= 0; i--) {
|
for (;i >= 0; i--) {
|
||||||
if (hsw_dais[i].playback.channels_min)
|
if (hsw_dais[i].playback.channels_min)
|
||||||
kfree(priv_data->pcm_pg[i][0]);
|
snd_dma_free_pages(&priv_data->dmab[i][0]);
|
||||||
if (hsw_dais[i].capture.channels_min)
|
if (hsw_dais[i].capture.channels_min)
|
||||||
kfree(priv_data->pcm_pg[i][1]);
|
snd_dma_free_pages(&priv_data->dmab[i][1]);
|
||||||
}
|
}
|
||||||
return -ENOMEM;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int hsw_pcm_remove(struct snd_soc_platform *platform)
|
static int hsw_pcm_remove(struct snd_soc_platform *platform)
|
||||||
|
@ -791,9 +795,9 @@ static int hsw_pcm_remove(struct snd_soc_platform *platform)
|
||||||
|
|
||||||
for (i = 0; i < ARRAY_SIZE(hsw_dais); i++) {
|
for (i = 0; i < ARRAY_SIZE(hsw_dais); i++) {
|
||||||
if (hsw_dais[i].playback.channels_min)
|
if (hsw_dais[i].playback.channels_min)
|
||||||
kfree(priv_data->pcm_pg[i][0]);
|
snd_dma_free_pages(&priv_data->dmab[i][0]);
|
||||||
if (hsw_dais[i].capture.channels_min)
|
if (hsw_dais[i].capture.channels_min)
|
||||||
kfree(priv_data->pcm_pg[i][1]);
|
snd_dma_free_pages(&priv_data->dmab[i][1]);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue