2 * Copyright (c) 2010-2011,2013-2015 The Linux Foundation. All rights reserved.
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2 and
6 * only version 2 as published by the Free Software Foundation.
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
13 * lpass-platform.c -- ALSA SoC platform driver for QTi LPASS
16 #include <linux/dma-mapping.h>
17 #include <linux/export.h>
18 #include <linux/kernel.h>
19 #include <linux/module.h>
20 #include <linux/platform_device.h>
21 #include <sound/pcm_params.h>
22 #include <linux/regmap.h>
23 #include <sound/soc.h>
24 #include "lpass-lpaif-reg.h"
27 struct lpass_pcm_data {
32 #define LPASS_PLATFORM_BUFFER_SIZE (16 * 1024)
33 #define LPASS_PLATFORM_PERIODS 2
35 static struct snd_pcm_hardware lpass_platform_pcm_hardware = {
36 .info = SNDRV_PCM_INFO_MMAP |
37 SNDRV_PCM_INFO_MMAP_VALID |
38 SNDRV_PCM_INFO_INTERLEAVED |
39 SNDRV_PCM_INFO_PAUSE |
40 SNDRV_PCM_INFO_RESUME,
41 .formats = SNDRV_PCM_FMTBIT_S16 |
42 SNDRV_PCM_FMTBIT_S24 |
44 .rates = SNDRV_PCM_RATE_8000_192000,
49 .buffer_bytes_max = LPASS_PLATFORM_BUFFER_SIZE,
50 .period_bytes_max = LPASS_PLATFORM_BUFFER_SIZE /
51 LPASS_PLATFORM_PERIODS,
52 .period_bytes_min = LPASS_PLATFORM_BUFFER_SIZE /
53 LPASS_PLATFORM_PERIODS,
54 .periods_min = LPASS_PLATFORM_PERIODS,
55 .periods_max = LPASS_PLATFORM_PERIODS,
59 static int lpass_platform_pcmops_open(struct snd_pcm_substream *substream)
61 struct snd_pcm_runtime *runtime = substream->runtime;
62 struct snd_soc_pcm_runtime *soc_runtime = substream->private_data;
63 struct snd_soc_dai *cpu_dai = soc_runtime->cpu_dai;
64 struct lpass_data *drvdata =
65 snd_soc_platform_get_drvdata(soc_runtime->platform);
66 struct lpass_variant *v = drvdata->variant;
67 int ret, dma_ch, dir = substream->stream;
68 struct lpass_pcm_data *data;
70 data = devm_kzalloc(soc_runtime->dev, sizeof(*data), GFP_KERNEL);
74 data->i2s_port = cpu_dai->driver->id;
75 runtime->private_data = data;
77 if (v->alloc_dma_channel)
78 dma_ch = v->alloc_dma_channel(drvdata, dir);
82 drvdata->substream[dma_ch] = substream;
84 ret = regmap_write(drvdata->lpaif_map,
85 LPAIF_DMACTL_REG(v, dma_ch, dir), 0);
87 dev_err(soc_runtime->dev,
88 "%s() error writing to rdmactl reg: %d\n",
93 data->dma_ch = dma_ch;
95 snd_soc_set_runtime_hwparams(substream, &lpass_platform_pcm_hardware);
97 runtime->dma_bytes = lpass_platform_pcm_hardware.buffer_bytes_max;
99 ret = snd_pcm_hw_constraint_integer(runtime,
100 SNDRV_PCM_HW_PARAM_PERIODS);
102 dev_err(soc_runtime->dev, "%s() setting constraints failed: %d\n",
107 snd_pcm_set_runtime_buffer(substream, &substream->dma_buffer);
112 static int lpass_platform_pcmops_close(struct snd_pcm_substream *substream)
114 struct snd_pcm_runtime *runtime = substream->runtime;
115 struct snd_soc_pcm_runtime *soc_runtime = substream->private_data;
116 struct lpass_data *drvdata =
117 snd_soc_platform_get_drvdata(soc_runtime->platform);
118 struct lpass_variant *v = drvdata->variant;
119 struct lpass_pcm_data *data;
121 data = runtime->private_data;
122 v = drvdata->variant;
123 drvdata->substream[data->dma_ch] = NULL;
124 if (v->free_dma_channel)
125 v->free_dma_channel(drvdata, data->dma_ch);
130 static int lpass_platform_pcmops_hw_params(struct snd_pcm_substream *substream,
131 struct snd_pcm_hw_params *params)
133 struct snd_soc_pcm_runtime *soc_runtime = substream->private_data;
134 struct lpass_data *drvdata =
135 snd_soc_platform_get_drvdata(soc_runtime->platform);
136 struct snd_pcm_runtime *rt = substream->runtime;
137 struct lpass_pcm_data *pcm_data = rt->private_data;
138 struct lpass_variant *v = drvdata->variant;
139 snd_pcm_format_t format = params_format(params);
140 unsigned int channels = params_channels(params);
142 int ch, dir = substream->stream;
144 int ret, dma_port = pcm_data->i2s_port + v->dmactl_audif_start;
146 ch = pcm_data->dma_ch;
148 bitwidth = snd_pcm_format_width(format);
150 dev_err(soc_runtime->dev, "%s() invalid bit width given: %d\n",
155 regval = LPAIF_DMACTL_BURSTEN_INCR4 |
156 LPAIF_DMACTL_AUDINTF(dma_port) |
157 LPAIF_DMACTL_FIFOWM_8;
164 regval |= LPAIF_DMACTL_WPSCNT_ONE;
167 regval |= LPAIF_DMACTL_WPSCNT_TWO;
170 regval |= LPAIF_DMACTL_WPSCNT_THREE;
173 regval |= LPAIF_DMACTL_WPSCNT_FOUR;
176 dev_err(soc_runtime->dev, "%s() invalid PCM config given: bw=%d, ch=%u\n",
177 __func__, bitwidth, channels);
185 regval |= LPAIF_DMACTL_WPSCNT_ONE;
188 regval |= LPAIF_DMACTL_WPSCNT_TWO;
191 regval |= LPAIF_DMACTL_WPSCNT_FOUR;
194 regval |= LPAIF_DMACTL_WPSCNT_SIX;
197 regval |= LPAIF_DMACTL_WPSCNT_EIGHT;
200 dev_err(soc_runtime->dev, "%s() invalid PCM config given: bw=%d, ch=%u\n",
201 __func__, bitwidth, channels);
206 dev_err(soc_runtime->dev, "%s() invalid PCM config given: bw=%d, ch=%u\n",
207 __func__, bitwidth, channels);
211 ret = regmap_write(drvdata->lpaif_map,
212 LPAIF_DMACTL_REG(v, ch, dir), regval);
214 dev_err(soc_runtime->dev, "%s() error writing to rdmactl reg: %d\n",
222 static int lpass_platform_pcmops_hw_free(struct snd_pcm_substream *substream)
224 struct snd_soc_pcm_runtime *soc_runtime = substream->private_data;
225 struct lpass_data *drvdata =
226 snd_soc_platform_get_drvdata(soc_runtime->platform);
227 struct snd_pcm_runtime *rt = substream->runtime;
228 struct lpass_pcm_data *pcm_data = rt->private_data;
229 struct lpass_variant *v = drvdata->variant;
233 reg = LPAIF_DMACTL_REG(v, pcm_data->dma_ch, substream->stream);
234 ret = regmap_write(drvdata->lpaif_map, reg, 0);
236 dev_err(soc_runtime->dev, "%s() error writing to rdmactl reg: %d\n",
242 static int lpass_platform_pcmops_prepare(struct snd_pcm_substream *substream)
244 struct snd_pcm_runtime *runtime = substream->runtime;
245 struct snd_soc_pcm_runtime *soc_runtime = substream->private_data;
246 struct lpass_data *drvdata =
247 snd_soc_platform_get_drvdata(soc_runtime->platform);
248 struct snd_pcm_runtime *rt = substream->runtime;
249 struct lpass_pcm_data *pcm_data = rt->private_data;
250 struct lpass_variant *v = drvdata->variant;
251 int ret, ch, dir = substream->stream;
253 ch = pcm_data->dma_ch;
255 ret = regmap_write(drvdata->lpaif_map,
256 LPAIF_DMABASE_REG(v, ch, dir),
259 dev_err(soc_runtime->dev, "%s() error writing to rdmabase reg: %d\n",
264 ret = regmap_write(drvdata->lpaif_map,
265 LPAIF_DMABUFF_REG(v, ch, dir),
266 (snd_pcm_lib_buffer_bytes(substream) >> 2) - 1);
268 dev_err(soc_runtime->dev, "%s() error writing to rdmabuff reg: %d\n",
273 ret = regmap_write(drvdata->lpaif_map,
274 LPAIF_DMAPER_REG(v, ch, dir),
275 (snd_pcm_lib_period_bytes(substream) >> 2) - 1);
277 dev_err(soc_runtime->dev, "%s() error writing to rdmaper reg: %d\n",
282 ret = regmap_update_bits(drvdata->lpaif_map,
283 LPAIF_DMACTL_REG(v, ch, dir),
284 LPAIF_DMACTL_ENABLE_MASK, LPAIF_DMACTL_ENABLE_ON);
286 dev_err(soc_runtime->dev, "%s() error writing to rdmactl reg: %d\n",
294 static int lpass_platform_pcmops_trigger(struct snd_pcm_substream *substream,
297 struct snd_soc_pcm_runtime *soc_runtime = substream->private_data;
298 struct lpass_data *drvdata =
299 snd_soc_platform_get_drvdata(soc_runtime->platform);
300 struct snd_pcm_runtime *rt = substream->runtime;
301 struct lpass_pcm_data *pcm_data = rt->private_data;
302 struct lpass_variant *v = drvdata->variant;
303 int ret, ch, dir = substream->stream;
305 ch = pcm_data->dma_ch;
308 case SNDRV_PCM_TRIGGER_START:
309 case SNDRV_PCM_TRIGGER_RESUME:
310 case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:
311 /* clear status before enabling interrupts */
312 ret = regmap_write(drvdata->lpaif_map,
313 LPAIF_IRQCLEAR_REG(v, LPAIF_IRQ_PORT_HOST),
316 dev_err(soc_runtime->dev, "%s() error writing to irqclear reg: %d\n",
321 ret = regmap_update_bits(drvdata->lpaif_map,
322 LPAIF_IRQEN_REG(v, LPAIF_IRQ_PORT_HOST),
326 dev_err(soc_runtime->dev, "%s() error writing to irqen reg: %d\n",
331 ret = regmap_update_bits(drvdata->lpaif_map,
332 LPAIF_DMACTL_REG(v, ch, dir),
333 LPAIF_DMACTL_ENABLE_MASK,
334 LPAIF_DMACTL_ENABLE_ON);
336 dev_err(soc_runtime->dev, "%s() error writing to rdmactl reg: %d\n",
341 case SNDRV_PCM_TRIGGER_STOP:
342 case SNDRV_PCM_TRIGGER_SUSPEND:
343 case SNDRV_PCM_TRIGGER_PAUSE_PUSH:
344 ret = regmap_update_bits(drvdata->lpaif_map,
345 LPAIF_DMACTL_REG(v, ch, dir),
346 LPAIF_DMACTL_ENABLE_MASK,
347 LPAIF_DMACTL_ENABLE_OFF);
349 dev_err(soc_runtime->dev, "%s() error writing to rdmactl reg: %d\n",
354 ret = regmap_update_bits(drvdata->lpaif_map,
355 LPAIF_IRQEN_REG(v, LPAIF_IRQ_PORT_HOST),
356 LPAIF_IRQ_ALL(ch), 0);
358 dev_err(soc_runtime->dev, "%s() error writing to irqen reg: %d\n",
368 static snd_pcm_uframes_t lpass_platform_pcmops_pointer(
369 struct snd_pcm_substream *substream)
371 struct snd_soc_pcm_runtime *soc_runtime = substream->private_data;
372 struct lpass_data *drvdata =
373 snd_soc_platform_get_drvdata(soc_runtime->platform);
374 struct snd_pcm_runtime *rt = substream->runtime;
375 struct lpass_pcm_data *pcm_data = rt->private_data;
376 struct lpass_variant *v = drvdata->variant;
377 unsigned int base_addr, curr_addr;
378 int ret, ch, dir = substream->stream;
380 ch = pcm_data->dma_ch;
382 ret = regmap_read(drvdata->lpaif_map,
383 LPAIF_DMABASE_REG(v, ch, dir), &base_addr);
385 dev_err(soc_runtime->dev, "%s() error reading from rdmabase reg: %d\n",
390 ret = regmap_read(drvdata->lpaif_map,
391 LPAIF_DMACURR_REG(v, ch, dir), &curr_addr);
393 dev_err(soc_runtime->dev, "%s() error reading from rdmacurr reg: %d\n",
398 return bytes_to_frames(substream->runtime, curr_addr - base_addr);
401 static int lpass_platform_pcmops_mmap(struct snd_pcm_substream *substream,
402 struct vm_area_struct *vma)
404 struct snd_pcm_runtime *runtime = substream->runtime;
406 return dma_mmap_coherent(substream->pcm->card->dev, vma,
407 runtime->dma_area, runtime->dma_addr,
411 static const struct snd_pcm_ops lpass_platform_pcm_ops = {
412 .open = lpass_platform_pcmops_open,
413 .close = lpass_platform_pcmops_close,
414 .ioctl = snd_pcm_lib_ioctl,
415 .hw_params = lpass_platform_pcmops_hw_params,
416 .hw_free = lpass_platform_pcmops_hw_free,
417 .prepare = lpass_platform_pcmops_prepare,
418 .trigger = lpass_platform_pcmops_trigger,
419 .pointer = lpass_platform_pcmops_pointer,
420 .mmap = lpass_platform_pcmops_mmap,
423 static irqreturn_t lpass_dma_interrupt_handler(
424 struct snd_pcm_substream *substream,
425 struct lpass_data *drvdata,
426 int chan, u32 interrupts)
428 struct snd_soc_pcm_runtime *soc_runtime = substream->private_data;
429 struct lpass_variant *v = drvdata->variant;
430 irqreturn_t ret = IRQ_NONE;
433 if (interrupts & LPAIF_IRQ_PER(chan)) {
434 rv = regmap_write(drvdata->lpaif_map,
435 LPAIF_IRQCLEAR_REG(v, LPAIF_IRQ_PORT_HOST),
436 LPAIF_IRQ_PER(chan));
438 dev_err(soc_runtime->dev, "%s() error writing to irqclear reg: %d\n",
442 snd_pcm_period_elapsed(substream);
446 if (interrupts & LPAIF_IRQ_XRUN(chan)) {
447 rv = regmap_write(drvdata->lpaif_map,
448 LPAIF_IRQCLEAR_REG(v, LPAIF_IRQ_PORT_HOST),
449 LPAIF_IRQ_XRUN(chan));
451 dev_err(soc_runtime->dev, "%s() error writing to irqclear reg: %d\n",
455 dev_warn(soc_runtime->dev, "%s() xrun warning\n", __func__);
456 snd_pcm_stop(substream, SNDRV_PCM_STATE_XRUN);
460 if (interrupts & LPAIF_IRQ_ERR(chan)) {
461 rv = regmap_write(drvdata->lpaif_map,
462 LPAIF_IRQCLEAR_REG(v, LPAIF_IRQ_PORT_HOST),
463 LPAIF_IRQ_ERR(chan));
465 dev_err(soc_runtime->dev, "%s() error writing to irqclear reg: %d\n",
469 dev_err(soc_runtime->dev, "%s() bus access error\n", __func__);
470 snd_pcm_stop(substream, SNDRV_PCM_STATE_DISCONNECTED);
477 static irqreturn_t lpass_platform_lpaif_irq(int irq, void *data)
479 struct lpass_data *drvdata = data;
480 struct lpass_variant *v = drvdata->variant;
484 rv = regmap_read(drvdata->lpaif_map,
485 LPAIF_IRQSTAT_REG(v, LPAIF_IRQ_PORT_HOST), &irqs);
487 pr_err("%s() error reading from irqstat reg: %d\n",
492 /* Handle per channel interrupts */
493 for (chan = 0; chan < LPASS_MAX_DMA_CHANNELS; chan++) {
494 if (irqs & LPAIF_IRQ_ALL(chan) && drvdata->substream[chan]) {
495 rv = lpass_dma_interrupt_handler(
496 drvdata->substream[chan],
497 drvdata, chan, irqs);
498 if (rv != IRQ_HANDLED)
506 static int lpass_platform_pcm_new(struct snd_soc_pcm_runtime *soc_runtime)
508 struct snd_pcm *pcm = soc_runtime->pcm;
509 struct snd_pcm_substream *psubstream, *csubstream;
511 size_t size = lpass_platform_pcm_hardware.buffer_bytes_max;
513 psubstream = pcm->streams[SNDRV_PCM_STREAM_PLAYBACK].substream;
515 ret = snd_dma_alloc_pages(SNDRV_DMA_TYPE_DEV,
516 soc_runtime->platform->dev,
517 size, &psubstream->dma_buffer);
519 dev_err(soc_runtime->dev, "Cannot allocate buffer(s)\n");
524 csubstream = pcm->streams[SNDRV_PCM_STREAM_CAPTURE].substream;
526 ret = snd_dma_alloc_pages(SNDRV_DMA_TYPE_DEV,
527 soc_runtime->platform->dev,
528 size, &csubstream->dma_buffer);
530 dev_err(soc_runtime->dev, "Cannot allocate buffer(s)\n");
532 snd_dma_free_pages(&psubstream->dma_buffer);
541 static void lpass_platform_pcm_free(struct snd_pcm *pcm)
543 struct snd_pcm_substream *substream;
546 for (i = 0; i < ARRAY_SIZE(pcm->streams); i++) {
547 substream = pcm->streams[i].substream;
549 snd_dma_free_pages(&substream->dma_buffer);
550 substream->dma_buffer.area = NULL;
551 substream->dma_buffer.addr = 0;
556 static struct snd_soc_platform_driver lpass_platform_driver = {
557 .pcm_new = lpass_platform_pcm_new,
558 .pcm_free = lpass_platform_pcm_free,
559 .ops = &lpass_platform_pcm_ops,
562 int asoc_qcom_lpass_platform_register(struct platform_device *pdev)
564 struct lpass_data *drvdata = platform_get_drvdata(pdev);
565 struct lpass_variant *v = drvdata->variant;
568 drvdata->lpaif_irq = platform_get_irq_byname(pdev, "lpass-irq-lpaif");
569 if (drvdata->lpaif_irq < 0) {
570 dev_err(&pdev->dev, "%s() error getting irq handle: %d\n",
571 __func__, drvdata->lpaif_irq);
575 /* ensure audio hardware is disabled */
576 ret = regmap_write(drvdata->lpaif_map,
577 LPAIF_IRQEN_REG(v, LPAIF_IRQ_PORT_HOST), 0);
579 dev_err(&pdev->dev, "%s() error writing to irqen reg: %d\n",
584 ret = devm_request_irq(&pdev->dev, drvdata->lpaif_irq,
585 lpass_platform_lpaif_irq, IRQF_TRIGGER_RISING,
586 "lpass-irq-lpaif", drvdata);
588 dev_err(&pdev->dev, "%s() irq request failed: %d\n",
594 return devm_snd_soc_register_platform(&pdev->dev,
595 &lpass_platform_driver);
597 EXPORT_SYMBOL_GPL(asoc_qcom_lpass_platform_register);
599 MODULE_DESCRIPTION("QTi LPASS Platform Driver");
600 MODULE_LICENSE("GPL v2");