On Tue, Apr 26, 2016 at 03:22:29PM +0200, Lars-Peter Clausen wrote: > On 04/26/2016 03:09 PM, Matthias Reichl wrote: > > Hi Lars, > > > > first of all thanks a lot for your detailled feedback! > > > > On Tue, Apr 26, 2016 at 10:47:05AM +0200, Lars-Peter Clausen wrote: > >>>>> + .periods_min = 2, > >>>>> + .periods_max = 255, > >>>>> + .buffer_bytes_max = 128 * PAGE_SIZE, > >>>>> +}; > >>>>> + > >>>>> +static const struct snd_dmaengine_pcm_config bcm2835_dmaengine_pcm_config = { > >>>>> + .prepare_slave_config = snd_dmaengine_pcm_prepare_slave_config, > >>>>> + .pcm_hardware = &bcm2835_pcm_hardware, > >>>>> + .prealloc_buffer_size = 256 * PAGE_SIZE, > >>>>> +}; > >>>> > >>>> The generic dmaengine PCM driver auto-discovers these things, no need to > >>>> provide them. The code is OK as it is. > >>> > >>> With the auto-discover code we loose the S16_LE format. > >>> > >>> If I understood the code in dmaengine_pcm_set_runtime_hwparams > >>> correctly, this is because the DMA controller doesn't support > >>> 16bit transfers (only multiples of 32bit are allowed). > >>> > >>> But since the I2S driver needs exactly 2 channels S16_LE actually > >>> works fine (one 32bit transfer per frame). > >>> > >>> Do you know of a better way to get S16_LE support? It could well > >>> be that I missed something important... > >> > >> With your patch we should just get an error when trying to configure a > >> stream with 16-bit samples since when setting up the DMA controller the > >> generic code will still choose a 16-bit device port size and this will be > >> rejected by the DMA controller. > > > > We had that code in downstream RPi kernel for ages (IIRC since > > 3.10) and so far it worked fine. > > > > I traced the code for the S16_LE case to find out why: > > > > snd_hwparams_to_dma_slave_config() sets src/dst_addr_width to 16bit. > > > > But immediately afterwards snd_dmaengine_pcm_set_config_from_dai_data() > > overrides addr_width to 32bit. > > > > Ok, makes sense. > > > bcm2835-i2s only supports 32bit access to the FIFO/data register > > and dma_data.addr_width is set to 32bit in the I2S driver - that > > code is also in mainline since the initial bcm2835-i2s commit. > > > > Of course all this only works because the number of channels > > is always 2. > > > >> When you look at peripherals that have DMA support there are basically two > >> types. > >> > >> Type A expect that each write (same applies for read) to the memory mapped > >> FIFO corresponds to exactly one sample. So if the sample width is 16-bit a > >> 16-bit write is done, if the sample width is 32-bit a 32-bit write is done. > >> In this case it is up to the DMA controller to fragment the byte stream into > >> individual samples. > >> > >> Type B on the other hand has a fixed port width (usually the bus size) and > >> expects that each write to the memory mapped FIFO uses the port width. It > >> then internally unpacks the data into the sample data. E.g. if the FIFO is > >> 32-bit wide and the sample width is 16-bit it will unpack the 32-bit entry > >> into two samples. > >> > >> Currently the generic code only supports type A. It would be great if you > >> could add support for type B to support the BCM2835 I2S controller properly. > > > > Do you have a particular solution in mind? > > > > Introducing a flag to just auto-add some packed formats would be > > easy, but a generic, robust solution might be tricky. We'd have > > to make sure that unsupported configurations (like an odd number > > of 16bit channels on 32bit-only setups) get rejected or we might > > be in trouble. > > I think in this case the DMA shouldn't limit the supported sample types. > Since the unpacking is done by the peripheral the peripheral is the one > component that limits what is supported and what is not and the DMA itself > has no influence on this. In the peripheral driver you have all the > information available there to specify the proper constraints and can handle > all the corner cases. Do you mean let the DAI driver check for and reject invalid configurations in hw_params? Yes, I think that should do it. > The overall constraints are the intersection of the > DMA and peripheral constraints, so by having no DMA constraints the > peripheral is the one providing all the constraints. > > We could either say that we assume that when the addr_width is fixed the DMA > shouldn't supply any constraints, or we could introduce a new flag in > snd_dmaengine_dai_dma_data that the peripheral has unpack support and the > DMA constraints don't matter. The additional flag sounds like a good idea. How about something like the patch below? If the ...PACK_16_32 flag is set, 16-bit data will always be packed into 32bit and 32bit DMA is done instead of 16bit. No further checks are done. I only did a very quick test on downstream kernel 4.4.8 and it seemed to work fine, S16_LE was available and usable. Setting addr_width in the DAI driver was also no longer necessary (but that still can be used as a final override if needed). I'm not sure if that approach is able to support other packed configurations as well or if I missed something important so I'd be glad to get some feedback on that. so long, Hias diff --git a/include/sound/dmaengine_pcm.h b/include/sound/dmaengine_pcm.h index f86ef5e..d22b54a 100644 --- a/include/sound/dmaengine_pcm.h +++ b/include/sound/dmaengine_pcm.h @@ -51,6 +51,11 @@ struct dma_chan *snd_dmaengine_pcm_request_channel(dma_filter_fn filter_fn, void *filter_data); struct dma_chan *snd_dmaengine_pcm_get_chan(struct snd_pcm_substream *substream); +/* + * The DAI accepts 2 16-bit values packed into a 32bit word. + */ +#define SND_DMAENGINE_PCM_DAI_FLAG_PACK_16_32 BIT(0) + /** * struct snd_dmaengine_dai_dma_data - DAI DMA configuration data * @addr: Address of the DAI data source or destination register. @@ -63,6 +68,8 @@ struct dma_chan *snd_dmaengine_pcm_get_chan(struct snd_pcm_substream *substream) * requesting the DMA channel. * @chan_name: Custom channel name to use when requesting DMA channel. * @fifo_size: FIFO size of the DAI controller in bytes + * @flags: PCM_DAI flags, only SND_DMAENGINE_PCM_DAI_FLAG_PACK_16_32 + * is currently checked */ struct snd_dmaengine_dai_dma_data { dma_addr_t addr; @@ -72,6 +79,7 @@ struct snd_dmaengine_dai_dma_data { void *filter_data; const char *chan_name; unsigned int fifo_size; + unsigned int flags; }; void snd_dmaengine_pcm_set_config_from_dai_data( diff --git a/sound/core/pcm_dmaengine.c b/sound/core/pcm_dmaengine.c index fba365a..a429f94 100644 --- a/sound/core/pcm_dmaengine.c +++ b/sound/core/pcm_dmaengine.c @@ -117,11 +117,21 @@ void snd_dmaengine_pcm_set_config_from_dai_data( if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { slave_config->dst_addr = dma_data->addr; slave_config->dst_maxburst = dma_data->maxburst; + if ((slave_config->dst_addr_width == + DMA_SLAVE_BUSWIDTH_2_BYTES) && + (dma_data->flags & SND_DMAENGINE_PCM_DAI_FLAG_PACK_16_32)) + slave_config->dst_addr_width = + DMA_SLAVE_BUSWIDTH_4_BYTES; if (dma_data->addr_width != DMA_SLAVE_BUSWIDTH_UNDEFINED) slave_config->dst_addr_width = dma_data->addr_width; } else { slave_config->src_addr = dma_data->addr; slave_config->src_maxburst = dma_data->maxburst; + if ((slave_config->src_addr_width == + DMA_SLAVE_BUSWIDTH_2_BYTES) && + (dma_data->flags & SND_DMAENGINE_PCM_DAI_FLAG_PACK_16_32)) + slave_config->src_addr_width = + DMA_SLAVE_BUSWIDTH_4_BYTES; if (dma_data->addr_width != DMA_SLAVE_BUSWIDTH_UNDEFINED) slave_config->src_addr_width = dma_data->addr_width; } diff --git a/sound/soc/bcm/bcm2835-i2s.c b/sound/soc/bcm/bcm2835-i2s.c index c413973..03d4ad1 100644 --- a/sound/soc/bcm/bcm2835-i2s.c +++ b/sound/soc/bcm/bcm2835-i2s.c @@ -877,11 +877,11 @@ static int bcm2835_i2s_probe(struct platform_device *pdev) dev->dma_data[SNDRV_PCM_STREAM_CAPTURE].addr = dma_reg_base + BCM2835_I2S_FIFO_A_REG; - /* Set the bus width */ - dev->dma_data[SNDRV_PCM_STREAM_PLAYBACK].addr_width = - DMA_SLAVE_BUSWIDTH_4_BYTES; - dev->dma_data[SNDRV_PCM_STREAM_CAPTURE].addr_width = - DMA_SLAVE_BUSWIDTH_4_BYTES; + /* signal that the DAI needs 2 16-bit values packed into 32bit */ + dev->dma_data[SNDRV_PCM_STREAM_PLAYBACK].flags = + SND_DMAENGINE_PCM_DAI_FLAG_PACK_16_32; + dev->dma_data[SNDRV_PCM_STREAM_CAPTURE].flags = + SND_DMAENGINE_PCM_DAI_FLAG_PACK_16_32; /* Set burst */ dev->dma_data[SNDRV_PCM_STREAM_PLAYBACK].maxburst = 2; diff --git a/sound/soc/soc-generic-dmaengine-pcm.c b/sound/soc/soc-generic-dmaengine-pcm.c index 6fd1906..25552c2 100644 --- a/sound/soc/soc-generic-dmaengine-pcm.c +++ b/sound/soc/soc-generic-dmaengine-pcm.c @@ -173,6 +173,11 @@ static int dmaengine_pcm_set_runtime_hwparams(struct snd_pcm_substream *substrea for (i = 0; i <= SNDRV_PCM_FORMAT_LAST; i++) { int bits = snd_pcm_format_physical_width(i); + if ((bits == 16) && + (dma_data->flags & SND_DMAENGINE_PCM_DAI_FLAG_PACK_16_32)) + /* 16-bit data needs to be packed into 32bit */ + bits = 32; + /* Enable only samples with DMA supported physical widths */ switch (bits) { case 8: _______________________________________________ Alsa-devel mailing list Alsa-devel@xxxxxxxxxxxxxxxx http://mailman.alsa-project.org/mailman/listinfo/alsa-devel