Commit 188d9cae authored by Herve Codina's avatar Herve Codina Committed by Mark Brown

ASoC: fsl: fsl_qmc_audio: Add support for non-interleaved mode.

The current fsl_qmc_audio works in interleaved mode. The audio samples
are interleaved and all data are sent to (received from) one QMC
channel.

Using several QMC channels, non interleaved mode can be easily
supported. In that case, data related to ch0 are sent to (received from)
the first QMC channel, data related to ch1 use the next QMC channel and
so on up to the last channel.

In terms of constraints and settings, the two modes are slightly
different:
- Interleaved mode:
    - The sample size should fit in the number of time-slots available
      for the QMC channel.
    - The number of audio channels should fit in the number of
      time-slots (taking into account the sample size) available for the
      QMC channel.
- Non-interleaved mode:
    - The number of audio channels is the number of available QMC
      channels.
    - Each QMC channel should have the same number of time-slots.
    - The sample size equals the number of time-slots of one QMC
      channel.

Add support for the non-interleaved mode allowing multiple QMC channel
per DAI. The DAI switches in non-interleaved mode when more that one QMC
channel is available.
Signed-off-by: default avatarHerve Codina <herve.codina@bootlin.com>
Link: https://patch.msgid.link/20240701113038.55144-11-herve.codina@bootlin.comSigned-off-by: default avatarMark Brown <broonie@kernel.org>
parent fb601316
...@@ -29,7 +29,11 @@ struct qmc_dai { ...@@ -29,7 +29,11 @@ struct qmc_dai {
struct device *dev; struct device *dev;
unsigned int nb_tx_ts; unsigned int nb_tx_ts;
unsigned int nb_rx_ts; unsigned int nb_rx_ts;
struct qmc_dai_chan chan;
unsigned int nb_chans_avail;
unsigned int nb_chans_used_tx;
unsigned int nb_chans_used_rx;
struct qmc_dai_chan *chans;
}; };
struct qmc_audio { struct qmc_audio {
...@@ -50,7 +54,10 @@ struct qmc_dai_prtd { ...@@ -50,7 +54,10 @@ struct qmc_dai_prtd {
dma_addr_t ch_dma_addr_current; dma_addr_t ch_dma_addr_current;
dma_addr_t ch_dma_addr_end; dma_addr_t ch_dma_addr_end;
size_t ch_dma_size; size_t ch_dma_size;
size_t ch_dma_offset;
unsigned int channels;
DECLARE_BITMAP(chans_pending, 64);
struct snd_pcm_substream *substream; struct snd_pcm_substream *substream;
}; };
...@@ -69,6 +76,18 @@ static int qmc_audio_pcm_construct(struct snd_soc_component *component, ...@@ -69,6 +76,18 @@ static int qmc_audio_pcm_construct(struct snd_soc_component *component,
return 0; return 0;
} }
static bool qmc_audio_access_is_interleaved(snd_pcm_access_t access)
{
switch (access) {
case SNDRV_PCM_ACCESS_MMAP_INTERLEAVED:
case SNDRV_PCM_ACCESS_RW_INTERLEAVED:
return true;
default:
break;
}
return false;
}
static int qmc_audio_pcm_hw_params(struct snd_soc_component *component, static int qmc_audio_pcm_hw_params(struct snd_soc_component *component,
struct snd_pcm_substream *substream, struct snd_pcm_substream *substream,
struct snd_pcm_hw_params *params) struct snd_pcm_hw_params *params)
...@@ -76,6 +95,14 @@ static int qmc_audio_pcm_hw_params(struct snd_soc_component *component, ...@@ -76,6 +95,14 @@ static int qmc_audio_pcm_hw_params(struct snd_soc_component *component,
struct snd_pcm_runtime *runtime = substream->runtime; struct snd_pcm_runtime *runtime = substream->runtime;
struct qmc_dai_prtd *prtd = substream->runtime->private_data; struct qmc_dai_prtd *prtd = substream->runtime->private_data;
/*
* In interleaved mode, the driver uses one QMC channel for all audio
* channels whereas in non-interleaved mode, it uses one QMC channel per
* audio channel.
*/
prtd->channels = qmc_audio_access_is_interleaved(params_access(params)) ?
1 : params_channels(params);
prtd->substream = substream; prtd->substream = substream;
prtd->buffer_ended = 0; prtd->buffer_ended = 0;
...@@ -83,9 +110,10 @@ static int qmc_audio_pcm_hw_params(struct snd_soc_component *component, ...@@ -83,9 +110,10 @@ static int qmc_audio_pcm_hw_params(struct snd_soc_component *component,
prtd->period_size = params_period_size(params); prtd->period_size = params_period_size(params);
prtd->ch_dma_addr_start = runtime->dma_addr; prtd->ch_dma_addr_start = runtime->dma_addr;
prtd->ch_dma_addr_end = runtime->dma_addr + params_buffer_bytes(params); prtd->ch_dma_offset = params_buffer_bytes(params) / prtd->channels;
prtd->ch_dma_addr_end = runtime->dma_addr + prtd->ch_dma_offset;
prtd->ch_dma_addr_current = prtd->ch_dma_addr_start; prtd->ch_dma_addr_current = prtd->ch_dma_addr_start;
prtd->ch_dma_size = params_period_bytes(params); prtd->ch_dma_size = params_period_bytes(params) / prtd->channels;
return 0; return 0;
} }
...@@ -94,16 +122,23 @@ static void qmc_audio_pcm_write_complete(void *context); ...@@ -94,16 +122,23 @@ static void qmc_audio_pcm_write_complete(void *context);
static int qmc_audio_pcm_write_submit(struct qmc_dai_prtd *prtd) static int qmc_audio_pcm_write_submit(struct qmc_dai_prtd *prtd)
{ {
unsigned int i;
int ret; int ret;
ret = qmc_chan_write_submit(prtd->qmc_dai->chan.qmc_chan, for (i = 0; i < prtd->channels; i++) {
prtd->ch_dma_addr_current, prtd->ch_dma_size, bitmap_set(prtd->chans_pending, i, 1);
qmc_audio_pcm_write_complete,
&prtd->qmc_dai->chan); ret = qmc_chan_write_submit(prtd->qmc_dai->chans[i].qmc_chan,
if (ret) { prtd->ch_dma_addr_current + i * prtd->ch_dma_offset,
dev_err(prtd->qmc_dai->dev, "write_submit failed %d\n", prtd->ch_dma_size,
ret); qmc_audio_pcm_write_complete,
return ret; &prtd->qmc_dai->chans[i]);
if (ret) {
dev_err(prtd->qmc_dai->dev, "write_submit %u failed %d\n",
i, ret);
bitmap_clear(prtd->chans_pending, i, 1);
return ret;
}
} }
return 0; return 0;
...@@ -116,6 +151,16 @@ static void qmc_audio_pcm_write_complete(void *context) ...@@ -116,6 +151,16 @@ static void qmc_audio_pcm_write_complete(void *context)
prtd = chan->prtd_tx; prtd = chan->prtd_tx;
/* Mark the current channel as completed */
bitmap_clear(prtd->chans_pending, chan - prtd->qmc_dai->chans, 1);
/*
* All QMC channels involved must have completed their transfer before
* submitting a new one.
*/
if (!bitmap_empty(prtd->chans_pending, 64))
return;
prtd->buffer_ended += prtd->period_size; prtd->buffer_ended += prtd->period_size;
if (prtd->buffer_ended >= prtd->buffer_size) if (prtd->buffer_ended >= prtd->buffer_size)
prtd->buffer_ended = 0; prtd->buffer_ended = 0;
...@@ -133,15 +178,23 @@ static void qmc_audio_pcm_read_complete(void *context, size_t length, unsigned i ...@@ -133,15 +178,23 @@ static void qmc_audio_pcm_read_complete(void *context, size_t length, unsigned i
static int qmc_audio_pcm_read_submit(struct qmc_dai_prtd *prtd) static int qmc_audio_pcm_read_submit(struct qmc_dai_prtd *prtd)
{ {
unsigned int i;
int ret; int ret;
ret = qmc_chan_read_submit(prtd->qmc_dai->chan.qmc_chan, for (i = 0; i < prtd->channels; i++) {
prtd->ch_dma_addr_current, prtd->ch_dma_size, bitmap_set(prtd->chans_pending, i, 1);
qmc_audio_pcm_read_complete,
&prtd->qmc_dai->chan); ret = qmc_chan_read_submit(prtd->qmc_dai->chans[i].qmc_chan,
if (ret) { prtd->ch_dma_addr_current + i * prtd->ch_dma_offset,
dev_err(prtd->qmc_dai->dev, "read_submit failed %d\n", prtd->ch_dma_size,
ret); qmc_audio_pcm_read_complete,
&prtd->qmc_dai->chans[i]);
if (ret) {
dev_err(prtd->qmc_dai->dev, "read_submit %u failed %d\n",
i, ret);
bitmap_clear(prtd->chans_pending, i, 1);
return ret;
}
} }
return 0; return 0;
...@@ -154,11 +207,21 @@ static void qmc_audio_pcm_read_complete(void *context, size_t length, unsigned i ...@@ -154,11 +207,21 @@ static void qmc_audio_pcm_read_complete(void *context, size_t length, unsigned i
prtd = chan->prtd_rx; prtd = chan->prtd_rx;
/* Mark the current channel as completed */
bitmap_clear(prtd->chans_pending, chan - prtd->qmc_dai->chans, 1);
if (length != prtd->ch_dma_size) { if (length != prtd->ch_dma_size) {
dev_err(prtd->qmc_dai->dev, "read complete length = %zu, exp %zu\n", dev_err(prtd->qmc_dai->dev, "read complete length = %zu, exp %zu\n",
length, prtd->ch_dma_size); length, prtd->ch_dma_size);
} }
/*
* All QMC channels involved must have completed their transfer before
* submitting a new one.
*/
if (!bitmap_empty(prtd->chans_pending, 64))
return;
prtd->buffer_ended += prtd->period_size; prtd->buffer_ended += prtd->period_size;
if (prtd->buffer_ended >= prtd->buffer_size) if (prtd->buffer_ended >= prtd->buffer_size)
prtd->buffer_ended = 0; prtd->buffer_ended = 0;
...@@ -176,6 +239,7 @@ static int qmc_audio_pcm_trigger(struct snd_soc_component *component, ...@@ -176,6 +239,7 @@ static int qmc_audio_pcm_trigger(struct snd_soc_component *component,
struct snd_pcm_substream *substream, int cmd) struct snd_pcm_substream *substream, int cmd)
{ {
struct qmc_dai_prtd *prtd = substream->runtime->private_data; struct qmc_dai_prtd *prtd = substream->runtime->private_data;
unsigned int i;
int ret; int ret;
if (!prtd->qmc_dai) { if (!prtd->qmc_dai) {
...@@ -185,8 +249,10 @@ static int qmc_audio_pcm_trigger(struct snd_soc_component *component, ...@@ -185,8 +249,10 @@ static int qmc_audio_pcm_trigger(struct snd_soc_component *component,
switch (cmd) { switch (cmd) {
case SNDRV_PCM_TRIGGER_START: case SNDRV_PCM_TRIGGER_START:
bitmap_zero(prtd->chans_pending, 64);
if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
prtd->qmc_dai->chan.prtd_tx = prtd; for (i = 0; i < prtd->channels; i++)
prtd->qmc_dai->chans[i].prtd_tx = prtd;
/* Submit first chunk ... */ /* Submit first chunk ... */
ret = qmc_audio_pcm_write_submit(prtd); ret = qmc_audio_pcm_write_submit(prtd);
...@@ -203,7 +269,8 @@ static int qmc_audio_pcm_trigger(struct snd_soc_component *component, ...@@ -203,7 +269,8 @@ static int qmc_audio_pcm_trigger(struct snd_soc_component *component,
if (ret) if (ret)
return ret; return ret;
} else { } else {
prtd->qmc_dai->chan.prtd_rx = prtd; for (i = 0; i < prtd->channels; i++)
prtd->qmc_dai->chans[i].prtd_rx = prtd;
/* Submit first chunk ... */ /* Submit first chunk ... */
ret = qmc_audio_pcm_read_submit(prtd); ret = qmc_audio_pcm_read_submit(prtd);
...@@ -270,6 +337,7 @@ static const struct snd_pcm_hardware qmc_audio_pcm_hardware = { ...@@ -270,6 +337,7 @@ static const struct snd_pcm_hardware qmc_audio_pcm_hardware = {
.info = SNDRV_PCM_INFO_MMAP | .info = SNDRV_PCM_INFO_MMAP |
SNDRV_PCM_INFO_MMAP_VALID | SNDRV_PCM_INFO_MMAP_VALID |
SNDRV_PCM_INFO_INTERLEAVED | SNDRV_PCM_INFO_INTERLEAVED |
SNDRV_PCM_INFO_NONINTERLEAVED |
SNDRV_PCM_INFO_PAUSE, SNDRV_PCM_INFO_PAUSE,
.period_bytes_min = 32, .period_bytes_min = 32,
.period_bytes_max = 64 * 1024, .period_bytes_max = 64 * 1024,
...@@ -442,6 +510,7 @@ static int qmc_dai_constraints_interleaved(struct snd_pcm_substream *substream, ...@@ -442,6 +510,7 @@ static int qmc_dai_constraints_interleaved(struct snd_pcm_substream *substream,
snd_pcm_hw_rule_func_t hw_rule_channels_by_format; snd_pcm_hw_rule_func_t hw_rule_channels_by_format;
snd_pcm_hw_rule_func_t hw_rule_format_by_channels; snd_pcm_hw_rule_func_t hw_rule_format_by_channels;
unsigned int frame_bits; unsigned int frame_bits;
u64 access;
int ret; int ret;
if (substream->stream == SNDRV_PCM_STREAM_CAPTURE) { if (substream->stream == SNDRV_PCM_STREAM_CAPTURE) {
...@@ -478,6 +547,44 @@ static int qmc_dai_constraints_interleaved(struct snd_pcm_substream *substream, ...@@ -478,6 +547,44 @@ static int qmc_dai_constraints_interleaved(struct snd_pcm_substream *substream,
return ret; return ret;
} }
access = 1ULL << (__force int)SNDRV_PCM_ACCESS_MMAP_INTERLEAVED |
1ULL << (__force int)SNDRV_PCM_ACCESS_RW_INTERLEAVED;
ret = snd_pcm_hw_constraint_mask64(substream->runtime, SNDRV_PCM_HW_PARAM_ACCESS,
access);
if (ret) {
dev_err(qmc_dai->dev, "Failed to add hw_param_access constraint (%d)\n", ret);
return ret;
}
return 0;
}
static int qmc_dai_constraints_noninterleaved(struct snd_pcm_substream *substream,
struct qmc_dai *qmc_dai)
{
unsigned int frame_bits;
u64 access;
int ret;
frame_bits = (substream->stream == SNDRV_PCM_STREAM_CAPTURE) ?
qmc_dai->nb_rx_ts * 8 : qmc_dai->nb_tx_ts * 8;
ret = snd_pcm_hw_constraint_single(substream->runtime,
SNDRV_PCM_HW_PARAM_FRAME_BITS,
frame_bits);
if (ret < 0) {
dev_err(qmc_dai->dev, "Failed to add frame_bits constraint (%d)\n", ret);
return ret;
}
access = 1ULL << (__force int)SNDRV_PCM_ACCESS_MMAP_NONINTERLEAVED |
1ULL << (__force int)SNDRV_PCM_ACCESS_RW_NONINTERLEAVED;
ret = snd_pcm_hw_constraint_mask64(substream->runtime, SNDRV_PCM_HW_PARAM_ACCESS,
access);
if (ret) {
dev_err(qmc_dai->dev, "Failed to add hw_param_access constraint (%d)\n", ret);
return ret;
}
return 0; return 0;
} }
...@@ -495,7 +602,9 @@ static int qmc_dai_startup(struct snd_pcm_substream *substream, ...@@ -495,7 +602,9 @@ static int qmc_dai_startup(struct snd_pcm_substream *substream,
prtd->qmc_dai = qmc_dai; prtd->qmc_dai = qmc_dai;
return qmc_dai_constraints_interleaved(substream, qmc_dai); return qmc_dai->nb_chans_avail > 1 ?
qmc_dai_constraints_noninterleaved(substream, qmc_dai) :
qmc_dai_constraints_interleaved(substream, qmc_dai);
} }
static int qmc_dai_hw_params(struct snd_pcm_substream *substream, static int qmc_dai_hw_params(struct snd_pcm_substream *substream,
...@@ -503,7 +612,9 @@ static int qmc_dai_hw_params(struct snd_pcm_substream *substream, ...@@ -503,7 +612,9 @@ static int qmc_dai_hw_params(struct snd_pcm_substream *substream,
struct snd_soc_dai *dai) struct snd_soc_dai *dai)
{ {
struct qmc_chan_param chan_param = {0}; struct qmc_chan_param chan_param = {0};
unsigned int nb_chans_used;
struct qmc_dai *qmc_dai; struct qmc_dai *qmc_dai;
unsigned int i;
int ret; int ret;
qmc_dai = qmc_dai_get_data(dai); qmc_dai = qmc_dai_get_data(dai);
...@@ -512,15 +623,34 @@ static int qmc_dai_hw_params(struct snd_pcm_substream *substream, ...@@ -512,15 +623,34 @@ static int qmc_dai_hw_params(struct snd_pcm_substream *substream,
return -EINVAL; return -EINVAL;
} }
/*
* In interleaved mode, the driver uses one QMC channel for all audio
* channels whereas in non-interleaved mode, it uses one QMC channel per
* audio channel.
*/
nb_chans_used = qmc_audio_access_is_interleaved(params_access(params)) ?
1 : params_channels(params);
if (nb_chans_used > qmc_dai->nb_chans_avail) {
dev_err(dai->dev, "Not enough qmc_chans. Need %u, avail %u\n",
nb_chans_used, qmc_dai->nb_chans_avail);
return -EINVAL;
}
if (substream->stream == SNDRV_PCM_STREAM_CAPTURE) { if (substream->stream == SNDRV_PCM_STREAM_CAPTURE) {
chan_param.mode = QMC_TRANSPARENT; chan_param.mode = QMC_TRANSPARENT;
chan_param.transp.max_rx_buf_size = params_period_bytes(params); chan_param.transp.max_rx_buf_size = params_period_bytes(params) / nb_chans_used;
ret = qmc_chan_set_param(qmc_dai->chan.qmc_chan, &chan_param); for (i = 0; i < nb_chans_used; i++) {
if (ret) { ret = qmc_chan_set_param(qmc_dai->chans[i].qmc_chan, &chan_param);
dev_err(dai->dev, "set param failed %d\n", if (ret) {
ret); dev_err(dai->dev, "chans[%u], set param failed %d\n",
return ret; i, ret);
return ret;
}
} }
qmc_dai->nb_chans_used_rx = nb_chans_used;
} else {
qmc_dai->nb_chans_used_tx = nb_chans_used;
} }
return 0; return 0;
...@@ -529,9 +659,12 @@ static int qmc_dai_hw_params(struct snd_pcm_substream *substream, ...@@ -529,9 +659,12 @@ static int qmc_dai_hw_params(struct snd_pcm_substream *substream,
static int qmc_dai_trigger(struct snd_pcm_substream *substream, int cmd, static int qmc_dai_trigger(struct snd_pcm_substream *substream, int cmd,
struct snd_soc_dai *dai) struct snd_soc_dai *dai)
{ {
unsigned int nb_chans_used;
struct qmc_dai *qmc_dai; struct qmc_dai *qmc_dai;
unsigned int i;
int direction; int direction;
int ret; int ret = 0;
int ret_tmp;
qmc_dai = qmc_dai_get_data(dai); qmc_dai = qmc_dai_get_data(dai);
if (!qmc_dai) { if (!qmc_dai) {
...@@ -539,30 +672,50 @@ static int qmc_dai_trigger(struct snd_pcm_substream *substream, int cmd, ...@@ -539,30 +672,50 @@ static int qmc_dai_trigger(struct snd_pcm_substream *substream, int cmd,
return -EINVAL; return -EINVAL;
} }
direction = (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ? if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
QMC_CHAN_WRITE : QMC_CHAN_READ; direction = QMC_CHAN_WRITE;
nb_chans_used = qmc_dai->nb_chans_used_tx;
} else {
direction = QMC_CHAN_READ;
nb_chans_used = qmc_dai->nb_chans_used_rx;
}
switch (cmd) { switch (cmd) {
case SNDRV_PCM_TRIGGER_START: case SNDRV_PCM_TRIGGER_START:
case SNDRV_PCM_TRIGGER_RESUME: case SNDRV_PCM_TRIGGER_RESUME:
case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:
ret = qmc_chan_start(qmc_dai->chan.qmc_chan, direction); for (i = 0; i < nb_chans_used; i++) {
if (ret) ret = qmc_chan_start(qmc_dai->chans[i].qmc_chan, direction);
return ret; if (ret)
goto err_stop;
}
break; break;
case SNDRV_PCM_TRIGGER_STOP: case SNDRV_PCM_TRIGGER_STOP:
ret = qmc_chan_stop(qmc_dai->chan.qmc_chan, direction); /* Stop and reset all QMC channels and return the first error encountered */
if (ret) for (i = 0; i < nb_chans_used; i++) {
return ret; ret_tmp = qmc_chan_stop(qmc_dai->chans[i].qmc_chan, direction);
ret = qmc_chan_reset(qmc_dai->chan.qmc_chan, direction); if (!ret)
ret = ret_tmp;
if (ret_tmp)
continue;
ret_tmp = qmc_chan_reset(qmc_dai->chans[i].qmc_chan, direction);
if (!ret)
ret = ret_tmp;
}
if (ret) if (ret)
return ret; return ret;
break; break;
case SNDRV_PCM_TRIGGER_SUSPEND: case SNDRV_PCM_TRIGGER_SUSPEND:
case SNDRV_PCM_TRIGGER_PAUSE_PUSH: case SNDRV_PCM_TRIGGER_PAUSE_PUSH:
ret = qmc_chan_stop(qmc_dai->chan.qmc_chan, direction); /* Stop all QMC channels and return the first error encountered */
for (i = 0; i < nb_chans_used; i++) {
ret_tmp = qmc_chan_stop(qmc_dai->chans[i].qmc_chan, direction);
if (!ret)
ret = ret_tmp;
}
if (ret) if (ret)
return ret; return ret;
break; break;
...@@ -572,6 +725,13 @@ static int qmc_dai_trigger(struct snd_pcm_substream *substream, int cmd, ...@@ -572,6 +725,13 @@ static int qmc_dai_trigger(struct snd_pcm_substream *substream, int cmd,
} }
return 0; return 0;
err_stop:
while (i--) {
qmc_chan_stop(qmc_dai->chans[i].qmc_chan, direction);
qmc_chan_reset(qmc_dai->chans[i].qmc_chan, direction);
}
return ret;
} }
static const struct snd_soc_dai_ops qmc_dai_ops = { static const struct snd_soc_dai_ops qmc_dai_ops = {
...@@ -580,7 +740,7 @@ static const struct snd_soc_dai_ops qmc_dai_ops = { ...@@ -580,7 +740,7 @@ static const struct snd_soc_dai_ops qmc_dai_ops = {
.hw_params = qmc_dai_hw_params, .hw_params = qmc_dai_hw_params,
}; };
static u64 qmc_audio_formats(u8 nb_ts) static u64 qmc_audio_formats(u8 nb_ts, bool is_noninterleaved)
{ {
unsigned int format_width; unsigned int format_width;
unsigned int chan_width; unsigned int chan_width;
...@@ -612,6 +772,13 @@ static u64 qmc_audio_formats(u8 nb_ts) ...@@ -612,6 +772,13 @@ static u64 qmc_audio_formats(u8 nb_ts)
if (format_width > chan_width || chan_width % format_width) if (format_width > chan_width || chan_width % format_width)
continue; continue;
/*
* In non interleaved mode, we can only support formats that
* can fit only 1 time in the channel
*/
if (is_noninterleaved && format_width != chan_width)
continue;
formats_mask |= pcm_format_to_bits(format); formats_mask |= pcm_format_to_bits(format);
} }
return formats_mask; return formats_mask;
...@@ -622,6 +789,12 @@ static int qmc_audio_dai_parse(struct qmc_audio *qmc_audio, struct device_node * ...@@ -622,6 +789,12 @@ static int qmc_audio_dai_parse(struct qmc_audio *qmc_audio, struct device_node *
struct snd_soc_dai_driver *qmc_soc_dai_driver) struct snd_soc_dai_driver *qmc_soc_dai_driver)
{ {
struct qmc_chan_info info; struct qmc_chan_info info;
unsigned long rx_fs_rate;
unsigned long tx_fs_rate;
unsigned int nb_tx_ts;
unsigned int nb_rx_ts;
unsigned int i;
int count;
u32 val; u32 val;
int ret; int ret;
...@@ -639,56 +812,105 @@ static int qmc_audio_dai_parse(struct qmc_audio *qmc_audio, struct device_node * ...@@ -639,56 +812,105 @@ static int qmc_audio_dai_parse(struct qmc_audio *qmc_audio, struct device_node *
if (!qmc_dai->name) if (!qmc_dai->name)
return -ENOMEM; return -ENOMEM;
qmc_dai->chan.qmc_chan = devm_qmc_chan_get_byphandle(qmc_audio->dev, np, count = qmc_chan_count_phandles(np, "fsl,qmc-chan");
"fsl,qmc-chan"); if (count < 0)
if (IS_ERR(qmc_dai->chan.qmc_chan)) { return dev_err_probe(qmc_audio->dev, count,
ret = PTR_ERR(qmc_dai->chan.qmc_chan); "dai %d get number of QMC channel failed\n", qmc_dai->id);
return dev_err_probe(qmc_audio->dev, ret, if (!count)
"dai %d get QMC channel failed\n", qmc_dai->id); return dev_err_probe(qmc_audio->dev, -EINVAL,
} "dai %d no QMC channel defined\n", qmc_dai->id);
qmc_soc_dai_driver->id = qmc_dai->id; qmc_dai->chans = devm_kcalloc(qmc_audio->dev, count, sizeof(*qmc_dai->chans), GFP_KERNEL);
qmc_soc_dai_driver->name = qmc_dai->name; if (!qmc_dai->chans)
return -ENOMEM;
ret = qmc_chan_get_info(qmc_dai->chan.qmc_chan, &info); for (i = 0; i < count; i++) {
if (ret) { qmc_dai->chans[i].qmc_chan = devm_qmc_chan_get_byphandles_index(qmc_audio->dev, np,
dev_err(qmc_audio->dev, "dai %d get QMC channel info failed %d\n", "fsl,qmc-chan", i);
qmc_dai->id, ret); if (IS_ERR(qmc_dai->chans[i].qmc_chan)) {
return ret; return dev_err_probe(qmc_audio->dev, PTR_ERR(qmc_dai->chans[i].qmc_chan),
} "dai %d get QMC channel %d failed\n", qmc_dai->id, i);
dev_info(qmc_audio->dev, "dai %d QMC channel mode %d, nb_tx_ts %u, nb_rx_ts %u\n", }
qmc_dai->id, info.mode, info.nb_tx_ts, info.nb_rx_ts);
if (info.mode != QMC_TRANSPARENT) { ret = qmc_chan_get_info(qmc_dai->chans[i].qmc_chan, &info);
dev_err(qmc_audio->dev, "dai %d QMC chan mode %d is not QMC_TRANSPARENT\n", if (ret) {
qmc_dai->id, info.mode); dev_err(qmc_audio->dev, "dai %d get QMC %d channel info failed %d\n",
return -EINVAL; qmc_dai->id, i, ret);
return ret;
}
dev_info(qmc_audio->dev, "dai %d QMC channel %d mode %d, nb_tx_ts %u, nb_rx_ts %u\n",
qmc_dai->id, i, info.mode, info.nb_tx_ts, info.nb_rx_ts);
if (info.mode != QMC_TRANSPARENT) {
dev_err(qmc_audio->dev, "dai %d QMC chan %d mode %d is not QMC_TRANSPARENT\n",
qmc_dai->id, i, info.mode);
return -EINVAL;
}
/*
* All channels must have the same number of Tx slots and the
* same numbers of Rx slots.
*/
if (i == 0) {
nb_tx_ts = info.nb_tx_ts;
nb_rx_ts = info.nb_rx_ts;
tx_fs_rate = info.tx_fs_rate;
rx_fs_rate = info.rx_fs_rate;
} else {
if (nb_tx_ts != info.nb_tx_ts) {
dev_err(qmc_audio->dev, "dai %d QMC chan %d inconsistent number of Tx timeslots (%u instead of %u)\n",
qmc_dai->id, i, info.nb_tx_ts, nb_tx_ts);
return -EINVAL;
}
if (nb_rx_ts != info.nb_rx_ts) {
dev_err(qmc_audio->dev, "dai %d QMC chan %d inconsistent number of Rx timeslots (%u instead of %u)\n",
qmc_dai->id, i, info.nb_rx_ts, nb_rx_ts);
return -EINVAL;
}
if (tx_fs_rate != info.tx_fs_rate) {
dev_err(qmc_audio->dev, "dai %d QMC chan %d inconsistent Tx frame sample rate (%lu instead of %lu)\n",
qmc_dai->id, i, info.tx_fs_rate, tx_fs_rate);
return -EINVAL;
}
if (rx_fs_rate != info.rx_fs_rate) {
dev_err(qmc_audio->dev, "dai %d QMC chan %d inconsistent Rx frame sample rate (%lu instead of %lu)\n",
qmc_dai->id, i, info.rx_fs_rate, rx_fs_rate);
return -EINVAL;
}
}
} }
qmc_dai->nb_tx_ts = info.nb_tx_ts;
qmc_dai->nb_rx_ts = info.nb_rx_ts; qmc_dai->nb_chans_avail = count;
qmc_dai->nb_tx_ts = nb_tx_ts * count;
qmc_dai->nb_rx_ts = nb_rx_ts * count;
qmc_soc_dai_driver->id = qmc_dai->id;
qmc_soc_dai_driver->name = qmc_dai->name;
qmc_soc_dai_driver->playback.channels_min = 0; qmc_soc_dai_driver->playback.channels_min = 0;
qmc_soc_dai_driver->playback.channels_max = 0; qmc_soc_dai_driver->playback.channels_max = 0;
if (qmc_dai->nb_tx_ts) { if (nb_tx_ts) {
qmc_soc_dai_driver->playback.channels_min = 1; qmc_soc_dai_driver->playback.channels_min = 1;
qmc_soc_dai_driver->playback.channels_max = qmc_dai->nb_tx_ts; qmc_soc_dai_driver->playback.channels_max = count > 1 ? count : nb_tx_ts;
} }
qmc_soc_dai_driver->playback.formats = qmc_audio_formats(qmc_dai->nb_tx_ts); qmc_soc_dai_driver->playback.formats = qmc_audio_formats(nb_tx_ts,
count > 1 ? true : false);
qmc_soc_dai_driver->capture.channels_min = 0; qmc_soc_dai_driver->capture.channels_min = 0;
qmc_soc_dai_driver->capture.channels_max = 0; qmc_soc_dai_driver->capture.channels_max = 0;
if (qmc_dai->nb_rx_ts) { if (nb_rx_ts) {
qmc_soc_dai_driver->capture.channels_min = 1; qmc_soc_dai_driver->capture.channels_min = 1;
qmc_soc_dai_driver->capture.channels_max = qmc_dai->nb_rx_ts; qmc_soc_dai_driver->capture.channels_max = count > 1 ? count : nb_rx_ts;
} }
qmc_soc_dai_driver->capture.formats = qmc_audio_formats(qmc_dai->nb_rx_ts); qmc_soc_dai_driver->capture.formats = qmc_audio_formats(nb_rx_ts,
count > 1 ? true : false);
qmc_soc_dai_driver->playback.rates = snd_pcm_rate_to_rate_bit(info.tx_fs_rate);
qmc_soc_dai_driver->playback.rate_min = info.tx_fs_rate; qmc_soc_dai_driver->playback.rates = snd_pcm_rate_to_rate_bit(tx_fs_rate);
qmc_soc_dai_driver->playback.rate_max = info.tx_fs_rate; qmc_soc_dai_driver->playback.rate_min = tx_fs_rate;
qmc_soc_dai_driver->capture.rates = snd_pcm_rate_to_rate_bit(info.rx_fs_rate); qmc_soc_dai_driver->playback.rate_max = tx_fs_rate;
qmc_soc_dai_driver->capture.rate_min = info.rx_fs_rate; qmc_soc_dai_driver->capture.rates = snd_pcm_rate_to_rate_bit(rx_fs_rate);
qmc_soc_dai_driver->capture.rate_max = info.rx_fs_rate; qmc_soc_dai_driver->capture.rate_min = rx_fs_rate;
qmc_soc_dai_driver->capture.rate_max = rx_fs_rate;
qmc_soc_dai_driver->ops = &qmc_dai_ops; qmc_soc_dai_driver->ops = &qmc_dai_ops;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment