diff options
author | Lars-Peter Clausen <lars@metafoo.de> | 2013-04-20 13:29:05 -0400 |
---|---|---|
committer | Mark Brown <broonie@opensource.wolfsonmicro.com> | 2013-05-13 10:20:56 -0400 |
commit | e27e8a60cb4ca8e3b047c5d6ee9afff9c4c5b61a (patch) | |
tree | b8bc47d5ae0bddbd8eef0528d308033308e3b119 | |
parent | 785d81e29bd237b4e76ca27c3ebcc3281e663596 (diff) |
ASoC: ep93xx: Use generic dmaengine PCM
Use the generic dmaengine PCM driver instead of a custom implementation.
Signed-off-by: Lars-Peter Clausen <lars@metafoo.de>
Signed-off-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
-rw-r--r-- | sound/soc/cirrus/Kconfig | 2 | ||||
-rw-r--r-- | sound/soc/cirrus/ep93xx-pcm.c | 138 |
2 files changed, 12 insertions, 128 deletions
diff --git a/sound/soc/cirrus/Kconfig b/sound/soc/cirrus/Kconfig index 88143db7e753..2c20f01e1f7e 100644 --- a/sound/soc/cirrus/Kconfig +++ b/sound/soc/cirrus/Kconfig | |||
@@ -1,7 +1,7 @@ | |||
1 | config SND_EP93XX_SOC | 1 | config SND_EP93XX_SOC |
2 | tristate "SoC Audio support for the Cirrus Logic EP93xx series" | 2 | tristate "SoC Audio support for the Cirrus Logic EP93xx series" |
3 | depends on ARCH_EP93XX && SND_SOC | 3 | depends on ARCH_EP93XX && SND_SOC |
4 | select SND_SOC_DMAENGINE_PCM | 4 | select SND_SOC_GENERIC_DMAENGINE_PCM |
5 | help | 5 | help |
6 | Say Y or M if you want to add support for codecs attached to | 6 | Say Y or M if you want to add support for codecs attached to |
7 | the EP93xx I2S or AC97 interfaces. | 7 | the EP93xx I2S or AC97 interfaces. |
diff --git a/sound/soc/cirrus/ep93xx-pcm.c b/sound/soc/cirrus/ep93xx-pcm.c index 488032690378..0e9f56e0d4b2 100644 --- a/sound/soc/cirrus/ep93xx-pcm.c +++ b/sound/soc/cirrus/ep93xx-pcm.c | |||
@@ -14,20 +14,14 @@ | |||
14 | 14 | ||
15 | #include <linux/module.h> | 15 | #include <linux/module.h> |
16 | #include <linux/init.h> | 16 | #include <linux/init.h> |
17 | #include <linux/device.h> | 17 | #include <linux/platform_device.h> |
18 | #include <linux/slab.h> | ||
19 | #include <linux/dmaengine.h> | 18 | #include <linux/dmaengine.h> |
20 | #include <linux/dma-mapping.h> | ||
21 | 19 | ||
22 | #include <sound/core.h> | ||
23 | #include <sound/pcm.h> | 20 | #include <sound/pcm.h> |
24 | #include <sound/pcm_params.h> | ||
25 | #include <sound/soc.h> | 21 | #include <sound/soc.h> |
26 | #include <sound/dmaengine_pcm.h> | 22 | #include <sound/dmaengine_pcm.h> |
27 | 23 | ||
28 | #include <linux/platform_data/dma-ep93xx.h> | 24 | #include <linux/platform_data/dma-ep93xx.h> |
29 | #include <mach/hardware.h> | ||
30 | #include <mach/ep93xx-regs.h> | ||
31 | 25 | ||
32 | static const struct snd_pcm_hardware ep93xx_pcm_hardware = { | 26 | static const struct snd_pcm_hardware ep93xx_pcm_hardware = { |
33 | .info = (SNDRV_PCM_INFO_MMAP | | 27 | .info = (SNDRV_PCM_INFO_MMAP | |
@@ -63,134 +57,24 @@ static bool ep93xx_pcm_dma_filter(struct dma_chan *chan, void *filter_param) | |||
63 | return false; | 57 | return false; |
64 | } | 58 | } |
65 | 59 | ||
66 | static int ep93xx_pcm_open(struct snd_pcm_substream *substream) | 60 | static const struct snd_dmaengine_pcm_config ep93xx_dmaengine_pcm_config = { |
67 | { | 61 | .pcm_hardware = &ep93xx_pcm_hardware, |
68 | struct snd_soc_pcm_runtime *rtd = substream->private_data; | 62 | .compat_filter_fn = ep93xx_pcm_dma_filter, |
69 | 63 | .prealloc_buffer_size = 131072, | |
70 | snd_soc_set_runtime_hwparams(substream, &ep93xx_pcm_hardware); | ||
71 | |||
72 | return snd_dmaengine_pcm_open_request_chan(substream, | ||
73 | ep93xx_pcm_dma_filter, | ||
74 | snd_soc_dai_get_dma_data(rtd->cpu_dai, substream)); | ||
75 | } | ||
76 | |||
77 | static int ep93xx_pcm_hw_params(struct snd_pcm_substream *substream, | ||
78 | struct snd_pcm_hw_params *params) | ||
79 | { | ||
80 | snd_pcm_set_runtime_buffer(substream, &substream->dma_buffer); | ||
81 | |||
82 | return 0; | ||
83 | } | ||
84 | |||
85 | static int ep93xx_pcm_hw_free(struct snd_pcm_substream *substream) | ||
86 | { | ||
87 | snd_pcm_set_runtime_buffer(substream, NULL); | ||
88 | return 0; | ||
89 | } | ||
90 | |||
91 | static int ep93xx_pcm_mmap(struct snd_pcm_substream *substream, | ||
92 | struct vm_area_struct *vma) | ||
93 | { | ||
94 | struct snd_pcm_runtime *runtime = substream->runtime; | ||
95 | |||
96 | return dma_mmap_writecombine(substream->pcm->card->dev, vma, | ||
97 | runtime->dma_area, | ||
98 | runtime->dma_addr, | ||
99 | runtime->dma_bytes); | ||
100 | } | ||
101 | |||
102 | static struct snd_pcm_ops ep93xx_pcm_ops = { | ||
103 | .open = ep93xx_pcm_open, | ||
104 | .close = snd_dmaengine_pcm_close_release_chan, | ||
105 | .ioctl = snd_pcm_lib_ioctl, | ||
106 | .hw_params = ep93xx_pcm_hw_params, | ||
107 | .hw_free = ep93xx_pcm_hw_free, | ||
108 | .trigger = snd_dmaengine_pcm_trigger, | ||
109 | .pointer = snd_dmaengine_pcm_pointer_no_residue, | ||
110 | .mmap = ep93xx_pcm_mmap, | ||
111 | }; | ||
112 | |||
113 | static int ep93xx_pcm_preallocate_dma_buffer(struct snd_pcm *pcm, int stream) | ||
114 | { | ||
115 | struct snd_pcm_substream *substream = pcm->streams[stream].substream; | ||
116 | struct snd_dma_buffer *buf = &substream->dma_buffer; | ||
117 | size_t size = ep93xx_pcm_hardware.buffer_bytes_max; | ||
118 | |||
119 | buf->dev.type = SNDRV_DMA_TYPE_DEV; | ||
120 | buf->dev.dev = pcm->card->dev; | ||
121 | buf->private_data = NULL; | ||
122 | buf->area = dma_alloc_writecombine(pcm->card->dev, size, | ||
123 | &buf->addr, GFP_KERNEL); | ||
124 | buf->bytes = size; | ||
125 | |||
126 | return (buf->area == NULL) ? -ENOMEM : 0; | ||
127 | } | ||
128 | |||
129 | static void ep93xx_pcm_free_dma_buffers(struct snd_pcm *pcm) | ||
130 | { | ||
131 | struct snd_pcm_substream *substream; | ||
132 | struct snd_dma_buffer *buf; | ||
133 | int stream; | ||
134 | |||
135 | for (stream = 0; stream < 2; stream++) { | ||
136 | substream = pcm->streams[stream].substream; | ||
137 | if (!substream) | ||
138 | continue; | ||
139 | |||
140 | buf = &substream->dma_buffer; | ||
141 | if (!buf->area) | ||
142 | continue; | ||
143 | |||
144 | dma_free_writecombine(pcm->card->dev, buf->bytes, buf->area, | ||
145 | buf->addr); | ||
146 | buf->area = NULL; | ||
147 | } | ||
148 | } | ||
149 | |||
150 | static u64 ep93xx_pcm_dmamask = DMA_BIT_MASK(32); | ||
151 | |||
152 | static int ep93xx_pcm_new(struct snd_soc_pcm_runtime *rtd) | ||
153 | { | ||
154 | struct snd_card *card = rtd->card->snd_card; | ||
155 | struct snd_pcm *pcm = rtd->pcm; | ||
156 | int ret = 0; | ||
157 | |||
158 | if (!card->dev->dma_mask) | ||
159 | card->dev->dma_mask = &ep93xx_pcm_dmamask; | ||
160 | if (!card->dev->coherent_dma_mask) | ||
161 | card->dev->coherent_dma_mask = DMA_BIT_MASK(32); | ||
162 | |||
163 | if (pcm->streams[SNDRV_PCM_STREAM_PLAYBACK].substream) { | ||
164 | ret = ep93xx_pcm_preallocate_dma_buffer(pcm, | ||
165 | SNDRV_PCM_STREAM_PLAYBACK); | ||
166 | if (ret) | ||
167 | return ret; | ||
168 | } | ||
169 | |||
170 | if (pcm->streams[SNDRV_PCM_STREAM_CAPTURE].substream) { | ||
171 | ret = ep93xx_pcm_preallocate_dma_buffer(pcm, | ||
172 | SNDRV_PCM_STREAM_CAPTURE); | ||
173 | if (ret) | ||
174 | return ret; | ||
175 | } | ||
176 | |||
177 | return 0; | ||
178 | } | ||
179 | |||
180 | static struct snd_soc_platform_driver ep93xx_soc_platform = { | ||
181 | .ops = &ep93xx_pcm_ops, | ||
182 | .pcm_new = &ep93xx_pcm_new, | ||
183 | .pcm_free = &ep93xx_pcm_free_dma_buffers, | ||
184 | }; | 64 | }; |
185 | 65 | ||
186 | static int ep93xx_soc_platform_probe(struct platform_device *pdev) | 66 | static int ep93xx_soc_platform_probe(struct platform_device *pdev) |
187 | { | 67 | { |
188 | return snd_soc_register_platform(&pdev->dev, &ep93xx_soc_platform); | 68 | return snd_dmaengine_pcm_register(&pdev->dev, |
69 | &ep93xx_dmaengine_pcm_config, | ||
70 | SND_DMAENGINE_PCM_FLAG_NO_RESIDUE | | ||
71 | SND_DMAENGINE_PCM_FLAG_NO_DT | | ||
72 | SND_DMAENGINE_PCM_FLAG_COMPAT); | ||
189 | } | 73 | } |
190 | 74 | ||
191 | static int ep93xx_soc_platform_remove(struct platform_device *pdev) | 75 | static int ep93xx_soc_platform_remove(struct platform_device *pdev) |
192 | { | 76 | { |
193 | snd_soc_unregister_platform(&pdev->dev); | 77 | snd_dmaengine_pcm_unregister(&pdev->dev); |
194 | return 0; | 78 | return 0; |
195 | } | 79 | } |
196 | 80 | ||