WIP FPC-III support
[linux/fpc-iii.git] / sound / soc / uniphier / aio-core.c
blobb8195778953e1eab2fb7fb77ec4b61ff1d6aaaa2
1 // SPDX-License-Identifier: GPL-2.0
2 //
3 // Socionext UniPhier AIO ALSA common driver.
4 //
5 // Copyright (c) 2016-2018 Socionext Inc.
7 #include <linux/bitfield.h>
8 #include <linux/errno.h>
9 #include <linux/kernel.h>
10 #include <linux/module.h>
11 #include <sound/core.h>
12 #include <sound/pcm.h>
13 #include <sound/pcm_params.h>
14 #include <sound/soc.h>
16 #include "aio.h"
17 #include "aio-reg.h"
19 static u64 rb_cnt(u64 wr, u64 rd, u64 len)
21 if (rd <= wr)
22 return wr - rd;
23 else
24 return len - (rd - wr);
27 static u64 rb_cnt_to_end(u64 wr, u64 rd, u64 len)
29 if (rd <= wr)
30 return wr - rd;
31 else
32 return len - rd;
35 static u64 rb_space(u64 wr, u64 rd, u64 len)
37 if (rd <= wr)
38 return len - (wr - rd) - 8;
39 else
40 return rd - wr - 8;
43 static u64 rb_space_to_end(u64 wr, u64 rd, u64 len)
45 if (rd > wr)
46 return rd - wr - 8;
47 else if (rd > 0)
48 return len - wr;
49 else
50 return len - wr - 8;
53 u64 aio_rb_cnt(struct uniphier_aio_sub *sub)
55 return rb_cnt(sub->wr_offs, sub->rd_offs, sub->compr_bytes);
58 u64 aio_rbt_cnt_to_end(struct uniphier_aio_sub *sub)
60 return rb_cnt_to_end(sub->wr_offs, sub->rd_offs, sub->compr_bytes);
63 u64 aio_rb_space(struct uniphier_aio_sub *sub)
65 return rb_space(sub->wr_offs, sub->rd_offs, sub->compr_bytes);
68 u64 aio_rb_space_to_end(struct uniphier_aio_sub *sub)
70 return rb_space_to_end(sub->wr_offs, sub->rd_offs, sub->compr_bytes);
73 /**
74 * aio_iecout_set_enable - setup IEC output via SoC glue
75 * @chip: the AIO chip pointer
76 * @enable: false to stop the output, true to start
78 * Set enabled or disabled S/PDIF signal output to out of SoC via AOnIEC pins.
79 * This function need to call at driver startup.
81 * The regmap of SoC glue is specified by 'socionext,syscon' optional property
82 * of DT. This function has no effect if no property.
84 void aio_iecout_set_enable(struct uniphier_aio_chip *chip, bool enable)
86 struct regmap *r = chip->regmap_sg;
88 if (!r)
89 return;
91 regmap_write(r, SG_AOUTEN, (enable) ? ~0 : 0);
94 /**
95 * aio_chip_set_pll - set frequency to audio PLL
96 * @chip: the AIO chip pointer
97 * @pll_id: PLL
98 * @freq: frequency in Hz, 0 is ignored
100 * Sets frequency of audio PLL. This function can be called anytime,
101 * but it takes time till PLL is locked.
103 * Return: Zero if successful, otherwise a negative value on error.
105 int aio_chip_set_pll(struct uniphier_aio_chip *chip, int pll_id,
106 unsigned int freq)
108 struct device *dev = &chip->pdev->dev;
109 struct regmap *r = chip->regmap;
110 int shift;
111 u32 v;
113 /* Not change */
114 if (freq == 0)
115 return 0;
117 switch (pll_id) {
118 case AUD_PLL_A1:
119 shift = 0;
120 break;
121 case AUD_PLL_F1:
122 shift = 1;
123 break;
124 case AUD_PLL_A2:
125 shift = 2;
126 break;
127 case AUD_PLL_F2:
128 shift = 3;
129 break;
130 default:
131 dev_err(dev, "PLL(%d) not supported\n", pll_id);
132 return -EINVAL;
135 switch (freq) {
136 case 36864000:
137 v = A2APLLCTR1_APLLX_36MHZ;
138 break;
139 case 33868800:
140 v = A2APLLCTR1_APLLX_33MHZ;
141 break;
142 default:
143 dev_err(dev, "PLL frequency not supported(%d)\n", freq);
144 return -EINVAL;
146 chip->plls[pll_id].freq = freq;
148 regmap_update_bits(r, A2APLLCTR1, A2APLLCTR1_APLLX_MASK << shift,
149 v << shift);
151 return 0;
155 * aio_chip_init - initialize AIO whole settings
156 * @chip: the AIO chip pointer
158 * Sets AIO fixed and whole device settings to AIO.
159 * This function need to call once at driver startup.
161 * The register area that is changed by this function is shared by all
162 * modules of AIO. But there is not race condition since this function
163 * has always set the same initialize values.
165 void aio_chip_init(struct uniphier_aio_chip *chip)
167 struct regmap *r = chip->regmap;
169 regmap_update_bits(r, A2APLLCTR0,
170 A2APLLCTR0_APLLXPOW_MASK,
171 A2APLLCTR0_APLLXPOW_PWON);
173 regmap_update_bits(r, A2EXMCLKSEL0,
174 A2EXMCLKSEL0_EXMCLK_MASK,
175 A2EXMCLKSEL0_EXMCLK_OUTPUT);
177 regmap_update_bits(r, A2AIOINPUTSEL, A2AIOINPUTSEL_RXSEL_MASK,
178 A2AIOINPUTSEL_RXSEL_PCMI1_HDMIRX1 |
179 A2AIOINPUTSEL_RXSEL_PCMI2_SIF |
180 A2AIOINPUTSEL_RXSEL_PCMI3_EVEA |
181 A2AIOINPUTSEL_RXSEL_IECI1_HDMIRX1);
183 if (chip->chip_spec->addr_ext)
184 regmap_update_bits(r, CDA2D_TEST, CDA2D_TEST_DDR_MODE_MASK,
185 CDA2D_TEST_DDR_MODE_EXTON0);
186 else
187 regmap_update_bits(r, CDA2D_TEST, CDA2D_TEST_DDR_MODE_MASK,
188 CDA2D_TEST_DDR_MODE_EXTOFF1);
192 * aio_init - initialize AIO substream
193 * @sub: the AIO substream pointer
195 * Sets fixed settings of each AIO substreams.
196 * This function need to call once at substream startup.
198 * Return: Zero if successful, otherwise a negative value on error.
200 int aio_init(struct uniphier_aio_sub *sub)
202 struct device *dev = &sub->aio->chip->pdev->dev;
203 struct regmap *r = sub->aio->chip->regmap;
205 regmap_write(r, A2RBNMAPCTR0(sub->swm->rb.hw),
206 MAPCTR0_EN | sub->swm->rb.map);
207 regmap_write(r, A2CHNMAPCTR0(sub->swm->ch.hw),
208 MAPCTR0_EN | sub->swm->ch.map);
210 switch (sub->swm->type) {
211 case PORT_TYPE_I2S:
212 case PORT_TYPE_SPDIF:
213 case PORT_TYPE_EVE:
214 if (sub->swm->dir == PORT_DIR_INPUT) {
215 regmap_write(r, A2IIFNMAPCTR0(sub->swm->iif.hw),
216 MAPCTR0_EN | sub->swm->iif.map);
217 regmap_write(r, A2IPORTNMAPCTR0(sub->swm->iport.hw),
218 MAPCTR0_EN | sub->swm->iport.map);
219 } else {
220 regmap_write(r, A2OIFNMAPCTR0(sub->swm->oif.hw),
221 MAPCTR0_EN | sub->swm->oif.map);
222 regmap_write(r, A2OPORTNMAPCTR0(sub->swm->oport.hw),
223 MAPCTR0_EN | sub->swm->oport.map);
225 break;
226 case PORT_TYPE_CONV:
227 regmap_write(r, A2OIFNMAPCTR0(sub->swm->oif.hw),
228 MAPCTR0_EN | sub->swm->oif.map);
229 regmap_write(r, A2OPORTNMAPCTR0(sub->swm->oport.hw),
230 MAPCTR0_EN | sub->swm->oport.map);
231 regmap_write(r, A2CHNMAPCTR0(sub->swm->och.hw),
232 MAPCTR0_EN | sub->swm->och.map);
233 regmap_write(r, A2IIFNMAPCTR0(sub->swm->iif.hw),
234 MAPCTR0_EN | sub->swm->iif.map);
235 break;
236 default:
237 dev_err(dev, "Unknown port type %d.\n", sub->swm->type);
238 return -EINVAL;
241 return 0;
245 * aio_port_reset - reset AIO port block
246 * @sub: the AIO substream pointer
248 * Resets the digital signal input/output port block of AIO.
250 void aio_port_reset(struct uniphier_aio_sub *sub)
252 struct regmap *r = sub->aio->chip->regmap;
254 if (sub->swm->dir == PORT_DIR_OUTPUT) {
255 regmap_write(r, AOUTRSTCTR0, BIT(sub->swm->oport.map));
256 regmap_write(r, AOUTRSTCTR1, BIT(sub->swm->oport.map));
257 } else {
258 regmap_update_bits(r, IPORTMXRSTCTR(sub->swm->iport.map),
259 IPORTMXRSTCTR_RSTPI_MASK,
260 IPORTMXRSTCTR_RSTPI_RESET);
261 regmap_update_bits(r, IPORTMXRSTCTR(sub->swm->iport.map),
262 IPORTMXRSTCTR_RSTPI_MASK,
263 IPORTMXRSTCTR_RSTPI_RELEASE);
268 * aio_port_set_ch - set channels of LPCM
269 * @sub: the AIO substream pointer, PCM substream only
271 * Set suitable slot selecting to input/output port block of AIO.
273 * This function may return error if non-PCM substream.
275 * Return: Zero if successful, otherwise a negative value on error.
277 static int aio_port_set_ch(struct uniphier_aio_sub *sub)
279 struct regmap *r = sub->aio->chip->regmap;
280 u32 slotsel_2ch[] = {
281 0, 0, 0, 0, 0,
283 u32 slotsel_multi[] = {
284 OPORTMXTYSLOTCTR_SLOTSEL_SLOT0,
285 OPORTMXTYSLOTCTR_SLOTSEL_SLOT1,
286 OPORTMXTYSLOTCTR_SLOTSEL_SLOT2,
287 OPORTMXTYSLOTCTR_SLOTSEL_SLOT3,
288 OPORTMXTYSLOTCTR_SLOTSEL_SLOT4,
290 u32 mode, *slotsel;
291 int i;
293 switch (params_channels(&sub->params)) {
294 case 8:
295 case 6:
296 mode = OPORTMXTYSLOTCTR_MODE;
297 slotsel = slotsel_multi;
298 break;
299 case 2:
300 mode = 0;
301 slotsel = slotsel_2ch;
302 break;
303 default:
304 return -EINVAL;
307 for (i = 0; i < AUD_MAX_SLOTSEL; i++) {
308 regmap_update_bits(r, OPORTMXTYSLOTCTR(sub->swm->oport.map, i),
309 OPORTMXTYSLOTCTR_MODE, mode);
310 regmap_update_bits(r, OPORTMXTYSLOTCTR(sub->swm->oport.map, i),
311 OPORTMXTYSLOTCTR_SLOTSEL_MASK, slotsel[i]);
314 return 0;
318 * aio_port_set_rate - set sampling rate of LPCM
319 * @sub: the AIO substream pointer, PCM substream only
320 * @rate: Sampling rate in Hz.
322 * Set suitable I2S format settings to input/output port block of AIO.
323 * Parameter is specified by hw_params().
325 * This function may return error if non-PCM substream.
327 * Return: Zero if successful, otherwise a negative value on error.
329 static int aio_port_set_rate(struct uniphier_aio_sub *sub, int rate)
331 struct regmap *r = sub->aio->chip->regmap;
332 struct device *dev = &sub->aio->chip->pdev->dev;
333 u32 v;
335 if (sub->swm->dir == PORT_DIR_OUTPUT) {
336 switch (rate) {
337 case 8000:
338 v = OPORTMXCTR1_FSSEL_8;
339 break;
340 case 11025:
341 v = OPORTMXCTR1_FSSEL_11_025;
342 break;
343 case 12000:
344 v = OPORTMXCTR1_FSSEL_12;
345 break;
346 case 16000:
347 v = OPORTMXCTR1_FSSEL_16;
348 break;
349 case 22050:
350 v = OPORTMXCTR1_FSSEL_22_05;
351 break;
352 case 24000:
353 v = OPORTMXCTR1_FSSEL_24;
354 break;
355 case 32000:
356 v = OPORTMXCTR1_FSSEL_32;
357 break;
358 case 44100:
359 v = OPORTMXCTR1_FSSEL_44_1;
360 break;
361 case 48000:
362 v = OPORTMXCTR1_FSSEL_48;
363 break;
364 case 88200:
365 v = OPORTMXCTR1_FSSEL_88_2;
366 break;
367 case 96000:
368 v = OPORTMXCTR1_FSSEL_96;
369 break;
370 case 176400:
371 v = OPORTMXCTR1_FSSEL_176_4;
372 break;
373 case 192000:
374 v = OPORTMXCTR1_FSSEL_192;
375 break;
376 default:
377 dev_err(dev, "Rate not supported(%d)\n", rate);
378 return -EINVAL;
381 regmap_update_bits(r, OPORTMXCTR1(sub->swm->oport.map),
382 OPORTMXCTR1_FSSEL_MASK, v);
383 } else {
384 switch (rate) {
385 case 8000:
386 v = IPORTMXCTR1_FSSEL_8;
387 break;
388 case 11025:
389 v = IPORTMXCTR1_FSSEL_11_025;
390 break;
391 case 12000:
392 v = IPORTMXCTR1_FSSEL_12;
393 break;
394 case 16000:
395 v = IPORTMXCTR1_FSSEL_16;
396 break;
397 case 22050:
398 v = IPORTMXCTR1_FSSEL_22_05;
399 break;
400 case 24000:
401 v = IPORTMXCTR1_FSSEL_24;
402 break;
403 case 32000:
404 v = IPORTMXCTR1_FSSEL_32;
405 break;
406 case 44100:
407 v = IPORTMXCTR1_FSSEL_44_1;
408 break;
409 case 48000:
410 v = IPORTMXCTR1_FSSEL_48;
411 break;
412 case 88200:
413 v = IPORTMXCTR1_FSSEL_88_2;
414 break;
415 case 96000:
416 v = IPORTMXCTR1_FSSEL_96;
417 break;
418 case 176400:
419 v = IPORTMXCTR1_FSSEL_176_4;
420 break;
421 case 192000:
422 v = IPORTMXCTR1_FSSEL_192;
423 break;
424 default:
425 dev_err(dev, "Rate not supported(%d)\n", rate);
426 return -EINVAL;
429 regmap_update_bits(r, IPORTMXCTR1(sub->swm->iport.map),
430 IPORTMXCTR1_FSSEL_MASK, v);
433 return 0;
437 * aio_port_set_fmt - set format of I2S data
438 * @sub: the AIO substream pointer, PCM substream only
439 * This parameter has no effect if substream is I2S or PCM.
441 * Set suitable I2S format settings to input/output port block of AIO.
442 * Parameter is specified by set_fmt().
444 * This function may return error if non-PCM substream.
446 * Return: Zero if successful, otherwise a negative value on error.
448 static int aio_port_set_fmt(struct uniphier_aio_sub *sub)
450 struct regmap *r = sub->aio->chip->regmap;
451 struct device *dev = &sub->aio->chip->pdev->dev;
452 u32 v;
454 if (sub->swm->dir == PORT_DIR_OUTPUT) {
455 switch (sub->aio->fmt) {
456 case SND_SOC_DAIFMT_LEFT_J:
457 v = OPORTMXCTR1_I2SLRSEL_LEFT;
458 break;
459 case SND_SOC_DAIFMT_RIGHT_J:
460 v = OPORTMXCTR1_I2SLRSEL_RIGHT;
461 break;
462 case SND_SOC_DAIFMT_I2S:
463 v = OPORTMXCTR1_I2SLRSEL_I2S;
464 break;
465 default:
466 dev_err(dev, "Format is not supported(%d)\n",
467 sub->aio->fmt);
468 return -EINVAL;
471 v |= OPORTMXCTR1_OUTBITSEL_24;
472 regmap_update_bits(r, OPORTMXCTR1(sub->swm->oport.map),
473 OPORTMXCTR1_I2SLRSEL_MASK |
474 OPORTMXCTR1_OUTBITSEL_MASK, v);
475 } else {
476 switch (sub->aio->fmt) {
477 case SND_SOC_DAIFMT_LEFT_J:
478 v = IPORTMXCTR1_LRSEL_LEFT;
479 break;
480 case SND_SOC_DAIFMT_RIGHT_J:
481 v = IPORTMXCTR1_LRSEL_RIGHT;
482 break;
483 case SND_SOC_DAIFMT_I2S:
484 v = IPORTMXCTR1_LRSEL_I2S;
485 break;
486 default:
487 dev_err(dev, "Format is not supported(%d)\n",
488 sub->aio->fmt);
489 return -EINVAL;
492 v |= IPORTMXCTR1_OUTBITSEL_24 |
493 IPORTMXCTR1_CHSEL_ALL;
494 regmap_update_bits(r, IPORTMXCTR1(sub->swm->iport.map),
495 IPORTMXCTR1_LRSEL_MASK |
496 IPORTMXCTR1_OUTBITSEL_MASK |
497 IPORTMXCTR1_CHSEL_MASK, v);
500 return 0;
504 * aio_port_set_clk - set clock and divider of AIO port block
505 * @sub: the AIO substream pointer
507 * Set suitable PLL clock divider and relational settings to
508 * input/output port block of AIO. Parameters are specified by
509 * set_sysclk() and set_pll().
511 * Return: Zero if successful, otherwise a negative value on error.
513 static int aio_port_set_clk(struct uniphier_aio_sub *sub)
515 struct uniphier_aio_chip *chip = sub->aio->chip;
516 struct device *dev = &sub->aio->chip->pdev->dev;
517 struct regmap *r = sub->aio->chip->regmap;
518 u32 v_pll[] = {
519 OPORTMXCTR2_ACLKSEL_A1, OPORTMXCTR2_ACLKSEL_F1,
520 OPORTMXCTR2_ACLKSEL_A2, OPORTMXCTR2_ACLKSEL_F2,
521 OPORTMXCTR2_ACLKSEL_A2PLL,
522 OPORTMXCTR2_ACLKSEL_RX1,
524 u32 v_div[] = {
525 OPORTMXCTR2_DACCKSEL_1_2, OPORTMXCTR2_DACCKSEL_1_3,
526 OPORTMXCTR2_DACCKSEL_1_1, OPORTMXCTR2_DACCKSEL_2_3,
528 u32 v;
530 if (sub->swm->dir == PORT_DIR_OUTPUT) {
531 if (sub->swm->type == PORT_TYPE_I2S) {
532 if (sub->aio->pll_out >= ARRAY_SIZE(v_pll)) {
533 dev_err(dev, "PLL(%d) is invalid\n",
534 sub->aio->pll_out);
535 return -EINVAL;
537 if (sub->aio->plldiv >= ARRAY_SIZE(v_div)) {
538 dev_err(dev, "PLL divider(%d) is invalid\n",
539 sub->aio->plldiv);
540 return -EINVAL;
543 v = v_pll[sub->aio->pll_out] |
544 OPORTMXCTR2_MSSEL_MASTER |
545 v_div[sub->aio->plldiv];
547 switch (chip->plls[sub->aio->pll_out].freq) {
548 case 0:
549 case 36864000:
550 case 33868800:
551 v |= OPORTMXCTR2_EXTLSIFSSEL_36;
552 break;
553 default:
554 v |= OPORTMXCTR2_EXTLSIFSSEL_24;
555 break;
557 } else if (sub->swm->type == PORT_TYPE_EVE) {
558 v = OPORTMXCTR2_ACLKSEL_A2PLL |
559 OPORTMXCTR2_MSSEL_MASTER |
560 OPORTMXCTR2_EXTLSIFSSEL_36 |
561 OPORTMXCTR2_DACCKSEL_1_2;
562 } else if (sub->swm->type == PORT_TYPE_SPDIF) {
563 if (sub->aio->pll_out >= ARRAY_SIZE(v_pll)) {
564 dev_err(dev, "PLL(%d) is invalid\n",
565 sub->aio->pll_out);
566 return -EINVAL;
568 v = v_pll[sub->aio->pll_out] |
569 OPORTMXCTR2_MSSEL_MASTER |
570 OPORTMXCTR2_DACCKSEL_1_2;
572 switch (chip->plls[sub->aio->pll_out].freq) {
573 case 0:
574 case 36864000:
575 case 33868800:
576 v |= OPORTMXCTR2_EXTLSIFSSEL_36;
577 break;
578 default:
579 v |= OPORTMXCTR2_EXTLSIFSSEL_24;
580 break;
582 } else {
583 v = OPORTMXCTR2_ACLKSEL_A1 |
584 OPORTMXCTR2_MSSEL_MASTER |
585 OPORTMXCTR2_EXTLSIFSSEL_36 |
586 OPORTMXCTR2_DACCKSEL_1_2;
588 regmap_write(r, OPORTMXCTR2(sub->swm->oport.map), v);
589 } else {
590 v = IPORTMXCTR2_ACLKSEL_A1 |
591 IPORTMXCTR2_MSSEL_SLAVE |
592 IPORTMXCTR2_EXTLSIFSSEL_36 |
593 IPORTMXCTR2_DACCKSEL_1_2;
594 regmap_write(r, IPORTMXCTR2(sub->swm->iport.map), v);
597 return 0;
601 * aio_port_set_param - set parameters of AIO port block
602 * @sub: the AIO substream pointer
603 * @pass_through: Zero if sound data is LPCM, otherwise if data is not LPCM.
604 * This parameter has no effect if substream is I2S or PCM.
605 * @params: hardware parameters of ALSA
607 * Set suitable setting to input/output port block of AIO to process the
608 * specified in params.
610 * Return: Zero if successful, otherwise a negative value on error.
612 int aio_port_set_param(struct uniphier_aio_sub *sub, int pass_through,
613 const struct snd_pcm_hw_params *params)
615 struct regmap *r = sub->aio->chip->regmap;
616 unsigned int rate;
617 u32 v;
618 int ret;
620 if (!pass_through) {
621 if (sub->swm->type == PORT_TYPE_EVE ||
622 sub->swm->type == PORT_TYPE_CONV) {
623 rate = 48000;
624 } else {
625 rate = params_rate(params);
628 ret = aio_port_set_ch(sub);
629 if (ret)
630 return ret;
632 ret = aio_port_set_rate(sub, rate);
633 if (ret)
634 return ret;
636 ret = aio_port_set_fmt(sub);
637 if (ret)
638 return ret;
641 ret = aio_port_set_clk(sub);
642 if (ret)
643 return ret;
645 if (sub->swm->dir == PORT_DIR_OUTPUT) {
646 if (pass_through)
647 v = OPORTMXCTR3_SRCSEL_STREAM |
648 OPORTMXCTR3_VALID_STREAM;
649 else
650 v = OPORTMXCTR3_SRCSEL_PCM |
651 OPORTMXCTR3_VALID_PCM;
653 v |= OPORTMXCTR3_IECTHUR_IECOUT |
654 OPORTMXCTR3_PMSEL_PAUSE |
655 OPORTMXCTR3_PMSW_MUTE_OFF;
656 regmap_write(r, OPORTMXCTR3(sub->swm->oport.map), v);
657 } else {
658 regmap_write(r, IPORTMXACLKSEL0EX(sub->swm->iport.map),
659 IPORTMXACLKSEL0EX_ACLKSEL0EX_INTERNAL);
660 regmap_write(r, IPORTMXEXNOE(sub->swm->iport.map),
661 IPORTMXEXNOE_PCMINOE_INPUT);
664 return 0;
668 * aio_port_set_enable - start or stop of AIO port block
669 * @sub: the AIO substream pointer
670 * @enable: zero to stop the block, otherwise to start
672 * Start or stop the signal input/output port block of AIO.
674 void aio_port_set_enable(struct uniphier_aio_sub *sub, int enable)
676 struct regmap *r = sub->aio->chip->regmap;
678 if (sub->swm->dir == PORT_DIR_OUTPUT) {
679 regmap_write(r, OPORTMXPATH(sub->swm->oport.map),
680 sub->swm->oif.map);
682 regmap_update_bits(r, OPORTMXMASK(sub->swm->oport.map),
683 OPORTMXMASK_IUDXMSK_MASK |
684 OPORTMXMASK_IUXCKMSK_MASK |
685 OPORTMXMASK_DXMSK_MASK |
686 OPORTMXMASK_XCKMSK_MASK,
687 OPORTMXMASK_IUDXMSK_OFF |
688 OPORTMXMASK_IUXCKMSK_OFF |
689 OPORTMXMASK_DXMSK_OFF |
690 OPORTMXMASK_XCKMSK_OFF);
692 if (enable)
693 regmap_write(r, AOUTENCTR0, BIT(sub->swm->oport.map));
694 else
695 regmap_write(r, AOUTENCTR1, BIT(sub->swm->oport.map));
696 } else {
697 regmap_update_bits(r, IPORTMXMASK(sub->swm->iport.map),
698 IPORTMXMASK_IUXCKMSK_MASK |
699 IPORTMXMASK_XCKMSK_MASK,
700 IPORTMXMASK_IUXCKMSK_OFF |
701 IPORTMXMASK_XCKMSK_OFF);
703 if (enable)
704 regmap_update_bits(r,
705 IPORTMXCTR2(sub->swm->iport.map),
706 IPORTMXCTR2_REQEN_MASK,
707 IPORTMXCTR2_REQEN_ENABLE);
708 else
709 regmap_update_bits(r,
710 IPORTMXCTR2(sub->swm->iport.map),
711 IPORTMXCTR2_REQEN_MASK,
712 IPORTMXCTR2_REQEN_DISABLE);
717 * aio_port_get_volume - get volume of AIO port block
718 * @sub: the AIO substream pointer
720 * Return: current volume, range is 0x0000 - 0xffff
722 int aio_port_get_volume(struct uniphier_aio_sub *sub)
724 struct regmap *r = sub->aio->chip->regmap;
725 u32 v;
727 regmap_read(r, OPORTMXTYVOLGAINSTATUS(sub->swm->oport.map, 0), &v);
729 return FIELD_GET(OPORTMXTYVOLGAINSTATUS_CUR_MASK, v);
733 * aio_port_set_volume - set volume of AIO port block
734 * @sub: the AIO substream pointer
735 * @vol: target volume, range is 0x0000 - 0xffff.
737 * Change digital volume and perfome fade-out/fade-in effect for specified
738 * output slot of port. Gained PCM value can calculate as the following:
739 * Gained = Original * vol / 0x4000
741 void aio_port_set_volume(struct uniphier_aio_sub *sub, int vol)
743 struct regmap *r = sub->aio->chip->regmap;
744 int oport_map = sub->swm->oport.map;
745 int cur, diff, slope = 0, fs;
747 if (sub->swm->dir == PORT_DIR_INPUT)
748 return;
750 cur = aio_port_get_volume(sub);
751 diff = abs(vol - cur);
752 fs = params_rate(&sub->params);
753 if (fs)
754 slope = diff / AUD_VOL_FADE_TIME * 1000 / fs;
755 slope = max(1, slope);
757 regmap_update_bits(r, OPORTMXTYVOLPARA1(oport_map, 0),
758 OPORTMXTYVOLPARA1_SLOPEU_MASK, slope << 16);
759 regmap_update_bits(r, OPORTMXTYVOLPARA2(oport_map, 0),
760 OPORTMXTYVOLPARA2_TARGET_MASK, vol);
762 if (cur < vol)
763 regmap_update_bits(r, OPORTMXTYVOLPARA2(oport_map, 0),
764 OPORTMXTYVOLPARA2_FADE_MASK,
765 OPORTMXTYVOLPARA2_FADE_FADEIN);
766 else
767 regmap_update_bits(r, OPORTMXTYVOLPARA2(oport_map, 0),
768 OPORTMXTYVOLPARA2_FADE_MASK,
769 OPORTMXTYVOLPARA2_FADE_FADEOUT);
771 regmap_write(r, AOUTFADECTR0, BIT(oport_map));
775 * aio_if_set_param - set parameters of AIO DMA I/F block
776 * @sub: the AIO substream pointer
777 * @pass_through: Zero if sound data is LPCM, otherwise if data is not LPCM.
778 * This parameter has no effect if substream is I2S or PCM.
780 * Set suitable setting to DMA interface block of AIO to process the
781 * specified in settings.
783 * Return: Zero if successful, otherwise a negative value on error.
785 int aio_if_set_param(struct uniphier_aio_sub *sub, int pass_through)
787 struct regmap *r = sub->aio->chip->regmap;
788 u32 memfmt, v;
790 if (sub->swm->dir == PORT_DIR_OUTPUT) {
791 if (pass_through) {
792 v = PBOUTMXCTR0_ENDIAN_0123 |
793 PBOUTMXCTR0_MEMFMT_STREAM;
794 } else {
795 switch (params_channels(&sub->params)) {
796 case 2:
797 memfmt = PBOUTMXCTR0_MEMFMT_2CH;
798 break;
799 case 6:
800 memfmt = PBOUTMXCTR0_MEMFMT_6CH;
801 break;
802 case 8:
803 memfmt = PBOUTMXCTR0_MEMFMT_8CH;
804 break;
805 default:
806 return -EINVAL;
808 v = PBOUTMXCTR0_ENDIAN_3210 | memfmt;
811 regmap_write(r, PBOUTMXCTR0(sub->swm->oif.map), v);
812 regmap_write(r, PBOUTMXCTR1(sub->swm->oif.map), 0);
813 } else {
814 regmap_write(r, PBINMXCTR(sub->swm->iif.map),
815 PBINMXCTR_NCONNECT_CONNECT |
816 PBINMXCTR_INOUTSEL_IN |
817 (sub->swm->iport.map << PBINMXCTR_PBINSEL_SHIFT) |
818 PBINMXCTR_ENDIAN_3210 |
819 PBINMXCTR_MEMFMT_D0);
822 return 0;
826 * aio_oport_set_stream_type - set parameters of AIO playback port block
827 * @sub: the AIO substream pointer
828 * @pc: Pc type of IEC61937
830 * Set special setting to output port block of AIO to output the stream
831 * via S/PDIF.
833 * Return: Zero if successful, otherwise a negative value on error.
835 int aio_oport_set_stream_type(struct uniphier_aio_sub *sub,
836 enum IEC61937_PC pc)
838 struct regmap *r = sub->aio->chip->regmap;
839 u32 repet = 0, pause = OPORTMXPAUDAT_PAUSEPC_CMN;
841 switch (pc) {
842 case IEC61937_PC_AC3:
843 repet = OPORTMXREPET_STRLENGTH_AC3 |
844 OPORTMXREPET_PMLENGTH_AC3;
845 pause |= OPORTMXPAUDAT_PAUSEPD_AC3;
846 break;
847 case IEC61937_PC_MPA:
848 repet = OPORTMXREPET_STRLENGTH_MPA |
849 OPORTMXREPET_PMLENGTH_MPA;
850 pause |= OPORTMXPAUDAT_PAUSEPD_MPA;
851 break;
852 case IEC61937_PC_MP3:
853 repet = OPORTMXREPET_STRLENGTH_MP3 |
854 OPORTMXREPET_PMLENGTH_MP3;
855 pause |= OPORTMXPAUDAT_PAUSEPD_MP3;
856 break;
857 case IEC61937_PC_DTS1:
858 repet = OPORTMXREPET_STRLENGTH_DTS1 |
859 OPORTMXREPET_PMLENGTH_DTS1;
860 pause |= OPORTMXPAUDAT_PAUSEPD_DTS1;
861 break;
862 case IEC61937_PC_DTS2:
863 repet = OPORTMXREPET_STRLENGTH_DTS2 |
864 OPORTMXREPET_PMLENGTH_DTS2;
865 pause |= OPORTMXPAUDAT_PAUSEPD_DTS2;
866 break;
867 case IEC61937_PC_DTS3:
868 repet = OPORTMXREPET_STRLENGTH_DTS3 |
869 OPORTMXREPET_PMLENGTH_DTS3;
870 pause |= OPORTMXPAUDAT_PAUSEPD_DTS3;
871 break;
872 case IEC61937_PC_AAC:
873 repet = OPORTMXREPET_STRLENGTH_AAC |
874 OPORTMXREPET_PMLENGTH_AAC;
875 pause |= OPORTMXPAUDAT_PAUSEPD_AAC;
876 break;
877 case IEC61937_PC_PAUSE:
878 /* Do nothing */
879 break;
882 regmap_write(r, OPORTMXREPET(sub->swm->oport.map), repet);
883 regmap_write(r, OPORTMXPAUDAT(sub->swm->oport.map), pause);
885 return 0;
889 * aio_src_reset - reset AIO SRC block
890 * @sub: the AIO substream pointer
892 * Resets the digital signal input/output port with sampling rate converter
893 * block of AIO.
894 * This function has no effect if substream is not supported rate converter.
896 void aio_src_reset(struct uniphier_aio_sub *sub)
898 struct regmap *r = sub->aio->chip->regmap;
900 if (sub->swm->dir != PORT_DIR_OUTPUT)
901 return;
903 regmap_write(r, AOUTSRCRSTCTR0, BIT(sub->swm->oport.map));
904 regmap_write(r, AOUTSRCRSTCTR1, BIT(sub->swm->oport.map));
908 * aio_src_set_param - set parameters of AIO SRC block
909 * @sub: the AIO substream pointer
910 * @params: hardware parameters of ALSA
912 * Set suitable setting to input/output port with sampling rate converter
913 * block of AIO to process the specified in params.
914 * This function has no effect if substream is not supported rate converter.
916 * Return: Zero if successful, otherwise a negative value on error.
918 int aio_src_set_param(struct uniphier_aio_sub *sub,
919 const struct snd_pcm_hw_params *params)
921 struct regmap *r = sub->aio->chip->regmap;
922 u32 v;
924 if (sub->swm->dir != PORT_DIR_OUTPUT)
925 return 0;
927 regmap_write(r, OPORTMXSRC1CTR(sub->swm->oport.map),
928 OPORTMXSRC1CTR_THMODE_SRC |
929 OPORTMXSRC1CTR_SRCPATH_CALC |
930 OPORTMXSRC1CTR_SYNC_ASYNC |
931 OPORTMXSRC1CTR_FSIIPSEL_INNER |
932 OPORTMXSRC1CTR_FSISEL_ACLK);
934 switch (params_rate(params)) {
935 default:
936 case 48000:
937 v = OPORTMXRATE_I_ACLKSEL_APLLA1 |
938 OPORTMXRATE_I_MCKSEL_36 |
939 OPORTMXRATE_I_FSSEL_48;
940 break;
941 case 44100:
942 v = OPORTMXRATE_I_ACLKSEL_APLLA2 |
943 OPORTMXRATE_I_MCKSEL_33 |
944 OPORTMXRATE_I_FSSEL_44_1;
945 break;
946 case 32000:
947 v = OPORTMXRATE_I_ACLKSEL_APLLA1 |
948 OPORTMXRATE_I_MCKSEL_36 |
949 OPORTMXRATE_I_FSSEL_32;
950 break;
953 regmap_write(r, OPORTMXRATE_I(sub->swm->oport.map),
954 v | OPORTMXRATE_I_ACLKSRC_APLL |
955 OPORTMXRATE_I_LRCKSTP_STOP);
956 regmap_update_bits(r, OPORTMXRATE_I(sub->swm->oport.map),
957 OPORTMXRATE_I_LRCKSTP_MASK,
958 OPORTMXRATE_I_LRCKSTP_START);
960 return 0;
963 int aio_srcif_set_param(struct uniphier_aio_sub *sub)
965 struct regmap *r = sub->aio->chip->regmap;
967 regmap_write(r, PBINMXCTR(sub->swm->iif.map),
968 PBINMXCTR_NCONNECT_CONNECT |
969 PBINMXCTR_INOUTSEL_OUT |
970 (sub->swm->oport.map << PBINMXCTR_PBINSEL_SHIFT) |
971 PBINMXCTR_ENDIAN_3210 |
972 PBINMXCTR_MEMFMT_D0);
974 return 0;
977 int aio_srcch_set_param(struct uniphier_aio_sub *sub)
979 struct regmap *r = sub->aio->chip->regmap;
981 regmap_write(r, CDA2D_CHMXCTRL1(sub->swm->och.map),
982 CDA2D_CHMXCTRL1_INDSIZE_INFINITE);
984 regmap_write(r, CDA2D_CHMXSRCAMODE(sub->swm->och.map),
985 CDA2D_CHMXAMODE_ENDIAN_3210 |
986 CDA2D_CHMXAMODE_AUPDT_FIX |
987 CDA2D_CHMXAMODE_TYPE_NORMAL);
989 regmap_write(r, CDA2D_CHMXDSTAMODE(sub->swm->och.map),
990 CDA2D_CHMXAMODE_ENDIAN_3210 |
991 CDA2D_CHMXAMODE_AUPDT_INC |
992 CDA2D_CHMXAMODE_TYPE_RING |
993 (sub->swm->och.map << CDA2D_CHMXAMODE_RSSEL_SHIFT));
995 return 0;
998 void aio_srcch_set_enable(struct uniphier_aio_sub *sub, int enable)
1000 struct regmap *r = sub->aio->chip->regmap;
1001 u32 v;
1003 if (enable)
1004 v = CDA2D_STRT0_STOP_START;
1005 else
1006 v = CDA2D_STRT0_STOP_STOP;
1008 regmap_write(r, CDA2D_STRT0,
1009 v | BIT(sub->swm->och.map));
1012 int aiodma_ch_set_param(struct uniphier_aio_sub *sub)
1014 struct regmap *r = sub->aio->chip->regmap;
1015 u32 v;
1017 regmap_write(r, CDA2D_CHMXCTRL1(sub->swm->ch.map),
1018 CDA2D_CHMXCTRL1_INDSIZE_INFINITE);
1020 v = CDA2D_CHMXAMODE_ENDIAN_3210 |
1021 CDA2D_CHMXAMODE_AUPDT_INC |
1022 CDA2D_CHMXAMODE_TYPE_NORMAL |
1023 (sub->swm->rb.map << CDA2D_CHMXAMODE_RSSEL_SHIFT);
1024 if (sub->swm->dir == PORT_DIR_OUTPUT)
1025 regmap_write(r, CDA2D_CHMXSRCAMODE(sub->swm->ch.map), v);
1026 else
1027 regmap_write(r, CDA2D_CHMXDSTAMODE(sub->swm->ch.map), v);
1029 return 0;
1032 void aiodma_ch_set_enable(struct uniphier_aio_sub *sub, int enable)
1034 struct regmap *r = sub->aio->chip->regmap;
1036 if (enable) {
1037 regmap_write(r, CDA2D_STRT0,
1038 CDA2D_STRT0_STOP_START | BIT(sub->swm->ch.map));
1040 regmap_update_bits(r, INTRBIM(0),
1041 BIT(sub->swm->rb.map),
1042 BIT(sub->swm->rb.map));
1043 } else {
1044 regmap_write(r, CDA2D_STRT0,
1045 CDA2D_STRT0_STOP_STOP | BIT(sub->swm->ch.map));
1047 regmap_update_bits(r, INTRBIM(0),
1048 BIT(sub->swm->rb.map),
1053 static u64 aiodma_rb_get_rp(struct uniphier_aio_sub *sub)
1055 struct regmap *r = sub->aio->chip->regmap;
1056 u32 pos_u, pos_l;
1057 int i;
1059 regmap_write(r, CDA2D_RDPTRLOAD,
1060 CDA2D_RDPTRLOAD_LSFLAG_STORE | BIT(sub->swm->rb.map));
1061 /* Wait for setup */
1062 for (i = 0; i < 6; i++)
1063 regmap_read(r, CDA2D_RBMXRDPTR(sub->swm->rb.map), &pos_l);
1065 regmap_read(r, CDA2D_RBMXRDPTR(sub->swm->rb.map), &pos_l);
1066 regmap_read(r, CDA2D_RBMXRDPTRU(sub->swm->rb.map), &pos_u);
1067 pos_u = FIELD_GET(CDA2D_RBMXPTRU_PTRU_MASK, pos_u);
1069 return ((u64)pos_u << 32) | pos_l;
1072 static void aiodma_rb_set_rp(struct uniphier_aio_sub *sub, u64 pos)
1074 struct regmap *r = sub->aio->chip->regmap;
1075 u32 tmp;
1076 int i;
1078 regmap_write(r, CDA2D_RBMXRDPTR(sub->swm->rb.map), (u32)pos);
1079 regmap_write(r, CDA2D_RBMXRDPTRU(sub->swm->rb.map), (u32)(pos >> 32));
1080 regmap_write(r, CDA2D_RDPTRLOAD, BIT(sub->swm->rb.map));
1081 /* Wait for setup */
1082 for (i = 0; i < 6; i++)
1083 regmap_read(r, CDA2D_RBMXRDPTR(sub->swm->rb.map), &tmp);
1086 static u64 aiodma_rb_get_wp(struct uniphier_aio_sub *sub)
1088 struct regmap *r = sub->aio->chip->regmap;
1089 u32 pos_u, pos_l;
1090 int i;
1092 regmap_write(r, CDA2D_WRPTRLOAD,
1093 CDA2D_WRPTRLOAD_LSFLAG_STORE | BIT(sub->swm->rb.map));
1094 /* Wait for setup */
1095 for (i = 0; i < 6; i++)
1096 regmap_read(r, CDA2D_RBMXWRPTR(sub->swm->rb.map), &pos_l);
1098 regmap_read(r, CDA2D_RBMXWRPTR(sub->swm->rb.map), &pos_l);
1099 regmap_read(r, CDA2D_RBMXWRPTRU(sub->swm->rb.map), &pos_u);
1100 pos_u = FIELD_GET(CDA2D_RBMXPTRU_PTRU_MASK, pos_u);
1102 return ((u64)pos_u << 32) | pos_l;
1105 static void aiodma_rb_set_wp(struct uniphier_aio_sub *sub, u64 pos)
1107 struct regmap *r = sub->aio->chip->regmap;
1108 u32 tmp;
1109 int i;
1111 regmap_write(r, CDA2D_RBMXWRPTR(sub->swm->rb.map),
1112 lower_32_bits(pos));
1113 regmap_write(r, CDA2D_RBMXWRPTRU(sub->swm->rb.map),
1114 upper_32_bits(pos));
1115 regmap_write(r, CDA2D_WRPTRLOAD, BIT(sub->swm->rb.map));
1116 /* Wait for setup */
1117 for (i = 0; i < 6; i++)
1118 regmap_read(r, CDA2D_RBMXWRPTR(sub->swm->rb.map), &tmp);
1121 int aiodma_rb_set_threshold(struct uniphier_aio_sub *sub, u64 size, u32 th)
1123 struct regmap *r = sub->aio->chip->regmap;
1125 if (size <= th)
1126 return -EINVAL;
1128 regmap_write(r, CDA2D_RBMXBTH(sub->swm->rb.map), th);
1129 regmap_write(r, CDA2D_RBMXRTH(sub->swm->rb.map), th);
1131 return 0;
1134 int aiodma_rb_set_buffer(struct uniphier_aio_sub *sub, u64 start, u64 end,
1135 int period)
1137 struct regmap *r = sub->aio->chip->regmap;
1138 u64 size = end - start;
1139 int ret;
1141 if (end < start || period < 0)
1142 return -EINVAL;
1144 regmap_write(r, CDA2D_RBMXCNFG(sub->swm->rb.map), 0);
1145 regmap_write(r, CDA2D_RBMXBGNADRS(sub->swm->rb.map),
1146 lower_32_bits(start));
1147 regmap_write(r, CDA2D_RBMXBGNADRSU(sub->swm->rb.map),
1148 upper_32_bits(start));
1149 regmap_write(r, CDA2D_RBMXENDADRS(sub->swm->rb.map),
1150 lower_32_bits(end));
1151 regmap_write(r, CDA2D_RBMXENDADRSU(sub->swm->rb.map),
1152 upper_32_bits(end));
1154 regmap_write(r, CDA2D_RBADRSLOAD, BIT(sub->swm->rb.map));
1156 ret = aiodma_rb_set_threshold(sub, size, 2 * period);
1157 if (ret)
1158 return ret;
1160 if (sub->swm->dir == PORT_DIR_OUTPUT) {
1161 aiodma_rb_set_rp(sub, start);
1162 aiodma_rb_set_wp(sub, end - period);
1164 regmap_update_bits(r, CDA2D_RBMXIE(sub->swm->rb.map),
1165 CDA2D_RBMXIX_SPACE,
1166 CDA2D_RBMXIX_SPACE);
1167 } else {
1168 aiodma_rb_set_rp(sub, end - period);
1169 aiodma_rb_set_wp(sub, start);
1171 regmap_update_bits(r, CDA2D_RBMXIE(sub->swm->rb.map),
1172 CDA2D_RBMXIX_REMAIN,
1173 CDA2D_RBMXIX_REMAIN);
1176 sub->threshold = 2 * period;
1177 sub->rd_offs = 0;
1178 sub->wr_offs = 0;
1179 sub->rd_org = 0;
1180 sub->wr_org = 0;
1181 sub->rd_total = 0;
1182 sub->wr_total = 0;
1184 return 0;
1187 void aiodma_rb_sync(struct uniphier_aio_sub *sub, u64 start, u64 size,
1188 int period)
1190 if (sub->swm->dir == PORT_DIR_OUTPUT) {
1191 sub->rd_offs = aiodma_rb_get_rp(sub) - start;
1193 if (sub->use_mmap) {
1194 sub->threshold = 2 * period;
1195 aiodma_rb_set_threshold(sub, size, 2 * period);
1197 sub->wr_offs = sub->rd_offs - period;
1198 if (sub->rd_offs < period)
1199 sub->wr_offs += size;
1201 aiodma_rb_set_wp(sub, sub->wr_offs + start);
1202 } else {
1203 sub->wr_offs = aiodma_rb_get_wp(sub) - start;
1205 if (sub->use_mmap) {
1206 sub->threshold = 2 * period;
1207 aiodma_rb_set_threshold(sub, size, 2 * period);
1209 sub->rd_offs = sub->wr_offs - period;
1210 if (sub->wr_offs < period)
1211 sub->rd_offs += size;
1213 aiodma_rb_set_rp(sub, sub->rd_offs + start);
1216 sub->rd_total += sub->rd_offs - sub->rd_org;
1217 if (sub->rd_offs < sub->rd_org)
1218 sub->rd_total += size;
1219 sub->wr_total += sub->wr_offs - sub->wr_org;
1220 if (sub->wr_offs < sub->wr_org)
1221 sub->wr_total += size;
1223 sub->rd_org = sub->rd_offs;
1224 sub->wr_org = sub->wr_offs;
1227 bool aiodma_rb_is_irq(struct uniphier_aio_sub *sub)
1229 struct regmap *r = sub->aio->chip->regmap;
1230 u32 ir;
1232 regmap_read(r, CDA2D_RBMXIR(sub->swm->rb.map), &ir);
1234 if (sub->swm->dir == PORT_DIR_OUTPUT)
1235 return !!(ir & CDA2D_RBMXIX_SPACE);
1236 else
1237 return !!(ir & CDA2D_RBMXIX_REMAIN);
1240 void aiodma_rb_clear_irq(struct uniphier_aio_sub *sub)
1242 struct regmap *r = sub->aio->chip->regmap;
1244 if (sub->swm->dir == PORT_DIR_OUTPUT)
1245 regmap_write(r, CDA2D_RBMXIR(sub->swm->rb.map),
1246 CDA2D_RBMXIX_SPACE);
1247 else
1248 regmap_write(r, CDA2D_RBMXIR(sub->swm->rb.map),
1249 CDA2D_RBMXIX_REMAIN);