Blame view
sound/arm/pxa2xx-pcm-lib.c
4.88 KB
a6d773176 ALSA: Separate co... |
1 2 3 4 5 |
/* * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. */ |
5a0e3ad6a include cleanup: ... |
6 |
#include <linux/slab.h> |
a6d773176 ALSA: Separate co... |
7 8 |
#include <linux/module.h> #include <linux/dma-mapping.h> |
d65a14587 ASoC: pxa: use sn... |
9 |
#include <linux/dmaengine.h> |
58ceb57ec ASoC: pxa: pxa-pc... |
10 |
#include <linux/dma/pxa-dma.h> |
a6d773176 ALSA: Separate co... |
11 12 13 14 15 |
#include <sound/core.h> #include <sound/pcm.h> #include <sound/pcm_params.h> #include <sound/pxa2xx-lib.h> |
d65a14587 ASoC: pxa: use sn... |
16 |
#include <sound/dmaengine_pcm.h> |
a6d773176 ALSA: Separate co... |
17 |
|
a6d773176 ALSA: Separate co... |
18 19 20 21 22 23 24 25 26 27 28 29 30 31 |
#include "pxa2xx-pcm.h" static const struct snd_pcm_hardware pxa2xx_pcm_hardware = { .info = SNDRV_PCM_INFO_MMAP | SNDRV_PCM_INFO_MMAP_VALID | SNDRV_PCM_INFO_INTERLEAVED | SNDRV_PCM_INFO_PAUSE | SNDRV_PCM_INFO_RESUME, .formats = SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_S32_LE, .period_bytes_min = 32, .period_bytes_max = 8192 - 32, .periods_min = 1, |
58ceb57ec ASoC: pxa: pxa-pc... |
32 |
.periods_max = 256, |
a6d773176 ALSA: Separate co... |
33 34 35 36 37 38 39 |
.buffer_bytes_max = 128 * 1024, .fifo_size = 32, }; int __pxa2xx_pcm_hw_params(struct snd_pcm_substream *substream, struct snd_pcm_hw_params *params) { |
58ceb57ec ASoC: pxa: pxa-pc... |
40 41 42 43 44 45 46 47 48 |
struct dma_chan *chan = snd_dmaengine_pcm_get_chan(substream); struct snd_soc_pcm_runtime *rtd = substream->private_data; struct snd_dmaengine_dai_dma_data *dma_params; struct dma_slave_config config; int ret; dma_params = snd_soc_dai_get_dma_data(rtd->cpu_dai, substream); if (!dma_params) return 0; |
d65a14587 ASoC: pxa: use sn... |
49 |
|
58ceb57ec ASoC: pxa: pxa-pc... |
50 51 52 |
ret = snd_hwparams_to_dma_slave_config(substream, params, &config); if (ret) return ret; |
d65a14587 ASoC: pxa: use sn... |
53 |
|
58ceb57ec ASoC: pxa: pxa-pc... |
54 55 56 |
snd_dmaengine_pcm_set_config_from_dai_data(substream, snd_soc_dai_get_dma_data(rtd->cpu_dai, substream), &config); |
a6d773176 ALSA: Separate co... |
57 |
|
58ceb57ec ASoC: pxa: pxa-pc... |
58 59 60 |
ret = dmaengine_slave_config(chan, &config); if (ret) return ret; |
a6d773176 ALSA: Separate co... |
61 |
|
58ceb57ec ASoC: pxa: pxa-pc... |
62 |
snd_pcm_set_runtime_buffer(substream, &substream->dma_buffer); |
a6d773176 ALSA: Separate co... |
63 64 65 66 67 68 69 |
return 0; } EXPORT_SYMBOL(__pxa2xx_pcm_hw_params); int __pxa2xx_pcm_hw_free(struct snd_pcm_substream *substream) { |
a6d773176 ALSA: Separate co... |
70 71 72 73 74 75 76 |
snd_pcm_set_runtime_buffer(substream, NULL); return 0; } EXPORT_SYMBOL(__pxa2xx_pcm_hw_free); int pxa2xx_pcm_trigger(struct snd_pcm_substream *substream, int cmd) { |
58ceb57ec ASoC: pxa: pxa-pc... |
77 |
return snd_dmaengine_pcm_trigger(substream, cmd); |
a6d773176 ALSA: Separate co... |
78 79 80 81 82 83 |
} EXPORT_SYMBOL(pxa2xx_pcm_trigger); snd_pcm_uframes_t pxa2xx_pcm_pointer(struct snd_pcm_substream *substream) { |
58ceb57ec ASoC: pxa: pxa-pc... |
84 |
return snd_dmaengine_pcm_pointer(substream); |
a6d773176 ALSA: Separate co... |
85 86 87 88 89 |
} EXPORT_SYMBOL(pxa2xx_pcm_pointer); int __pxa2xx_pcm_prepare(struct snd_pcm_substream *substream) { |
a6d773176 ALSA: Separate co... |
90 91 92 |
return 0; } EXPORT_SYMBOL(__pxa2xx_pcm_prepare); |
a6d773176 ALSA: Separate co... |
93 94 |
int __pxa2xx_pcm_open(struct snd_pcm_substream *substream) { |
58ceb57ec ASoC: pxa: pxa-pc... |
95 |
struct snd_soc_pcm_runtime *rtd = substream->private_data; |
a6d773176 ALSA: Separate co... |
96 |
struct snd_pcm_runtime *runtime = substream->runtime; |
58ceb57ec ASoC: pxa: pxa-pc... |
97 |
struct snd_dmaengine_dai_dma_data *dma_params; |
a6d773176 ALSA: Separate co... |
98 99 100 |
int ret; runtime->hw = pxa2xx_pcm_hardware; |
58ceb57ec ASoC: pxa: pxa-pc... |
101 102 103 |
dma_params = snd_soc_dai_get_dma_data(rtd->cpu_dai, substream); if (!dma_params) return 0; |
a6d773176 ALSA: Separate co... |
104 105 106 107 108 109 110 111 |
/* * For mysterious reasons (and despite what the manual says) * playback samples are lost if the DMA count is not a multiple * of the DMA burst size. Let's add a rule to enforce that. */ ret = snd_pcm_hw_constraint_step(runtime, 0, SNDRV_PCM_HW_PARAM_PERIOD_BYTES, 32); if (ret) |
58ceb57ec ASoC: pxa: pxa-pc... |
112 |
return ret; |
a6d773176 ALSA: Separate co... |
113 114 115 116 |
ret = snd_pcm_hw_constraint_step(runtime, 0, SNDRV_PCM_HW_PARAM_BUFFER_BYTES, 32); if (ret) |
58ceb57ec ASoC: pxa: pxa-pc... |
117 |
return ret; |
a6d773176 ALSA: Separate co... |
118 119 120 121 |
ret = snd_pcm_hw_constraint_integer(runtime, SNDRV_PCM_HW_PARAM_PERIODS); if (ret < 0) |
58ceb57ec ASoC: pxa: pxa-pc... |
122 |
return ret; |
a6d773176 ALSA: Separate co... |
123 |
|
58ceb57ec ASoC: pxa: pxa-pc... |
124 125 126 |
return snd_dmaengine_pcm_open_request_chan(substream, pxad_filter_fn, dma_params->filter_data); |
a6d773176 ALSA: Separate co... |
127 128 129 130 131 |
} EXPORT_SYMBOL(__pxa2xx_pcm_open); int __pxa2xx_pcm_close(struct snd_pcm_substream *substream) { |
58ceb57ec ASoC: pxa: pxa-pc... |
132 |
return snd_dmaengine_pcm_close_release_chan(substream); |
a6d773176 ALSA: Separate co... |
133 134 135 136 137 138 139 |
} EXPORT_SYMBOL(__pxa2xx_pcm_close); int pxa2xx_pcm_mmap(struct snd_pcm_substream *substream, struct vm_area_struct *vma) { struct snd_pcm_runtime *runtime = substream->runtime; |
f6e45661f dma, mm/pat: Rena... |
140 141 |
return dma_mmap_wc(substream->pcm->card->dev, vma, runtime->dma_area, runtime->dma_addr, runtime->dma_bytes); |
a6d773176 ALSA: Separate co... |
142 143 144 145 146 147 148 149 150 151 152 |
} EXPORT_SYMBOL(pxa2xx_pcm_mmap); int pxa2xx_pcm_preallocate_dma_buffer(struct snd_pcm *pcm, int stream) { struct snd_pcm_substream *substream = pcm->streams[stream].substream; struct snd_dma_buffer *buf = &substream->dma_buffer; size_t size = pxa2xx_pcm_hardware.buffer_bytes_max; buf->dev.type = SNDRV_DMA_TYPE_DEV; buf->dev.dev = pcm->card->dev; buf->private_data = NULL; |
f6e45661f dma, mm/pat: Rena... |
153 |
buf->area = dma_alloc_wc(pcm->card->dev, size, &buf->addr, GFP_KERNEL); |
a6d773176 ALSA: Separate co... |
154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 |
if (!buf->area) return -ENOMEM; buf->bytes = size; return 0; } EXPORT_SYMBOL(pxa2xx_pcm_preallocate_dma_buffer); void pxa2xx_pcm_free_dma_buffers(struct snd_pcm *pcm) { struct snd_pcm_substream *substream; struct snd_dma_buffer *buf; int stream; for (stream = 0; stream < 2; stream++) { substream = pcm->streams[stream].substream; if (!substream) continue; buf = &substream->dma_buffer; if (!buf->area) continue; |
f6e45661f dma, mm/pat: Rena... |
174 |
dma_free_wc(pcm->card->dev, buf->bytes, buf->area, buf->addr); |
a6d773176 ALSA: Separate co... |
175 176 177 178 179 180 181 182 |
buf->area = NULL; } } EXPORT_SYMBOL(pxa2xx_pcm_free_dma_buffers); MODULE_AUTHOR("Nicolas Pitre"); MODULE_DESCRIPTION("Intel PXA2xx sound library"); MODULE_LICENSE("GPL"); |