diff options
Diffstat (limited to 'sound/soc/soc-utils.c')
| -rw-r--r-- | sound/soc/soc-utils.c | 151 | 
1 files changed, 131 insertions, 20 deletions
diff --git a/sound/soc/soc-utils.c b/sound/soc/soc-utils.c index 1d07b931f3d..7f22ca35a41 100644 --- a/sound/soc/soc-utils.c +++ b/sound/soc/soc-utils.c @@ -13,6 +13,8 @@   *  option) any later version.   */ +#include <linux/platform_device.h> +#include <linux/export.h>  #include <sound/core.h>  #include <sound/pcm.h>  #include <sound/pcm_params.h> @@ -28,26 +30,9 @@ int snd_soc_params_to_frame_size(struct snd_pcm_hw_params *params)  {  	int sample_size; -	switch (params_format(params)) { -	case SNDRV_PCM_FORMAT_S16_LE: -	case SNDRV_PCM_FORMAT_S16_BE: -		sample_size = 16; -		break; -	case SNDRV_PCM_FORMAT_S20_3LE: -	case SNDRV_PCM_FORMAT_S20_3BE: -		sample_size = 20; -		break; -	case SNDRV_PCM_FORMAT_S24_LE: -	case SNDRV_PCM_FORMAT_S24_BE: -		sample_size = 24; -		break; -	case SNDRV_PCM_FORMAT_S32_LE: -	case SNDRV_PCM_FORMAT_S32_BE: -		sample_size = 32; -		break; -	default: -		return -ENOTSUPP; -	} +	sample_size = snd_pcm_format_width(params_format(params)); +	if (sample_size < 0) +		return sample_size;  	return snd_soc_calc_frame_size(sample_size, params_channels(params),  				       1); @@ -72,3 +57,129 @@ int snd_soc_params_to_bclk(struct snd_pcm_hw_params *params)  		return ret;  }  EXPORT_SYMBOL_GPL(snd_soc_params_to_bclk); + +static const struct snd_pcm_hardware dummy_dma_hardware = { +	/* Random values to keep userspace happy when checking constraints */ +	.info			= SNDRV_PCM_INFO_INTERLEAVED | +				  SNDRV_PCM_INFO_BLOCK_TRANSFER, +	.buffer_bytes_max	= 128*1024, +	.period_bytes_min	= PAGE_SIZE, +	.period_bytes_max	= PAGE_SIZE*2, +	.periods_min		= 2, +	.periods_max		= 128, +}; + +static int dummy_dma_open(struct snd_pcm_substream *substream) +{ +	struct snd_soc_pcm_runtime *rtd = substream->private_data; + +	/* BE's dont need dummy params */ +	if (!rtd->dai_link->no_pcm) +		snd_soc_set_runtime_hwparams(substream, &dummy_dma_hardware); + +	return 0; +} + +static struct snd_pcm_ops dummy_dma_ops = { +	.open		= dummy_dma_open, +	.ioctl		= snd_pcm_lib_ioctl, +}; + +static struct snd_soc_platform_driver dummy_platform = { +	.ops = &dummy_dma_ops, +}; + +static struct snd_soc_codec_driver dummy_codec; + +#define STUB_RATES	SNDRV_PCM_RATE_8000_192000 +#define STUB_FORMATS	(SNDRV_PCM_FMTBIT_S8 | \ +			SNDRV_PCM_FMTBIT_U8 | \ +			SNDRV_PCM_FMTBIT_S16_LE | \ +			SNDRV_PCM_FMTBIT_U16_LE | \ +			SNDRV_PCM_FMTBIT_S24_LE | \ +			SNDRV_PCM_FMTBIT_U24_LE | \ +			SNDRV_PCM_FMTBIT_S32_LE | \ +			SNDRV_PCM_FMTBIT_U32_LE | \ +			SNDRV_PCM_FMTBIT_IEC958_SUBFRAME_LE) +static struct snd_soc_dai_driver dummy_dai = { +	.name = "snd-soc-dummy-dai", +	.playback = { +		.stream_name	= "Playback", +		.channels_min	= 1, +		.channels_max	= 384, +		.rates		= STUB_RATES, +		.formats	= STUB_FORMATS, +	}, +	.capture = { +		.stream_name	= "Capture", +		.channels_min	= 1, +		.channels_max	= 384, +		.rates = STUB_RATES, +		.formats = STUB_FORMATS, +	 }, +}; + +int snd_soc_dai_is_dummy(struct snd_soc_dai *dai) +{ +	if (dai->driver == &dummy_dai) +		return 1; +	return 0; +} + +static int snd_soc_dummy_probe(struct platform_device *pdev) +{ +	int ret; + +	ret = snd_soc_register_codec(&pdev->dev, &dummy_codec, &dummy_dai, 1); +	if (ret < 0) +		return ret; + +	ret = snd_soc_register_platform(&pdev->dev, &dummy_platform); +	if (ret < 0) { +		snd_soc_unregister_codec(&pdev->dev); +		return ret; +	} + +	return ret; +} + +static int snd_soc_dummy_remove(struct platform_device *pdev) +{ +	snd_soc_unregister_platform(&pdev->dev); +	snd_soc_unregister_codec(&pdev->dev); + +	return 0; +} + +static struct platform_driver soc_dummy_driver = { +	.driver = { +		.name = "snd-soc-dummy", +		.owner = THIS_MODULE, +	}, +	.probe = snd_soc_dummy_probe, +	.remove = snd_soc_dummy_remove, +}; + +static struct platform_device *soc_dummy_dev; + +int __init snd_soc_util_init(void) +{ +	int ret; + +	soc_dummy_dev = +		platform_device_register_simple("snd-soc-dummy", -1, NULL, 0); +	if (IS_ERR(soc_dummy_dev)) +		return PTR_ERR(soc_dummy_dev); + +	ret = platform_driver_register(&soc_dummy_driver); +	if (ret != 0) +		platform_device_unregister(soc_dummy_dev); + +	return ret; +} + +void __exit snd_soc_util_exit(void) +{ +	platform_device_unregister(soc_dummy_dev); +	platform_driver_unregister(&soc_dummy_driver); +}  | 
