2009-11-10 19:08:04 +03:00
/*
* soc - util . c - - ALSA SoC Audio Layer utility functions
*
* Copyright 2009 Wolfson Microelectronics PLC .
*
* Author : Mark Brown < broonie @ opensource . wolfsonmicro . com >
* Liam Girdwood < lrg @ slimlogic . co . uk >
*
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation ; either version 2 of the License , or ( at your
* option ) any later version .
*/
2011-04-27 21:16:32 +04:00
# include <linux/platform_device.h>
2011-09-22 17:34:58 +04:00
# include <linux/export.h>
2009-11-10 19:08:04 +03:00
# include <sound/core.h>
# include <sound/pcm.h>
# include <sound/pcm_params.h>
# include <sound/soc.h>
int snd_soc_calc_frame_size ( int sample_size , int channels , int tdm_slots )
{
return sample_size * channels * tdm_slots ;
}
EXPORT_SYMBOL_GPL ( snd_soc_calc_frame_size ) ;
int snd_soc_params_to_frame_size ( struct snd_pcm_hw_params * params )
{
int sample_size ;
2011-01-31 23:14:38 +03:00
sample_size = snd_pcm_format_width ( params_format ( params ) ) ;
if ( sample_size < 0 )
return sample_size ;
2009-11-10 19:08:04 +03:00
return snd_soc_calc_frame_size ( sample_size , params_channels ( params ) ,
1 ) ;
}
EXPORT_SYMBOL_GPL ( snd_soc_params_to_frame_size ) ;
2009-11-19 14:36:10 +03:00
int snd_soc_calc_bclk ( int fs , int sample_size , int channels , int tdm_slots )
{
return fs * snd_soc_calc_frame_size ( sample_size , channels , tdm_slots ) ;
}
EXPORT_SYMBOL_GPL ( snd_soc_calc_bclk ) ;
2009-11-10 19:08:04 +03:00
int snd_soc_params_to_bclk ( struct snd_pcm_hw_params * params )
{
int ret ;
ret = snd_soc_params_to_frame_size ( params ) ;
if ( ret > 0 )
return ret * params_rate ( params ) ;
else
return ret ;
}
EXPORT_SYMBOL_GPL ( snd_soc_params_to_bclk ) ;
2011-04-27 21:16:32 +04:00
2011-12-06 00:50:45 +04:00
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 )
{
2013-09-13 21:09:47 +04:00
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 ) ;
2011-12-06 00:50:45 +04:00
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 ,
} ;
2011-04-27 21:16:32 +04:00
2012-01-09 16:36:24 +04:00
static struct snd_soc_codec_driver dummy_codec ;
2013-04-17 16:34:06 +04:00
# 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 )
2012-01-09 16:36:24 +04:00
static struct snd_soc_dai_driver dummy_dai = {
. name = " snd-soc-dummy-dai " ,
2013-04-17 16:34:06 +04:00
. 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 ,
} ,
2012-01-09 16:36:24 +04:00
} ;
2014-01-08 14:40:18 +04:00
int snd_soc_dai_is_dummy ( struct snd_soc_dai * dai )
{
if ( dai - > driver = = & dummy_dai )
return 1 ;
return 0 ;
}
2012-12-07 18:26:36 +04:00
static int snd_soc_dummy_probe ( struct platform_device * pdev )
2011-04-27 21:16:32 +04:00
{
2012-01-09 16:36:24 +04:00
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 ;
2011-04-27 21:16:32 +04:00
}
2012-12-07 18:26:36 +04:00
static int snd_soc_dummy_remove ( struct platform_device * pdev )
2011-04-27 21:16:32 +04:00
{
snd_soc_unregister_platform ( & pdev - > dev ) ;
2012-01-09 16:36:24 +04:00
snd_soc_unregister_codec ( & pdev - > dev ) ;
2011-04-27 21:16:32 +04:00
return 0 ;
}
static struct platform_driver soc_dummy_driver = {
. driver = {
. name = " snd-soc-dummy " ,
} ,
. probe = snd_soc_dummy_probe ,
2012-12-07 18:26:36 +04:00
. remove = snd_soc_dummy_remove ,
2011-04-27 21:16:32 +04:00
} ;
static struct platform_device * soc_dummy_dev ;
2011-04-28 13:57:54 +04:00
int __init snd_soc_util_init ( void )
2011-04-27 21:16:32 +04:00
{
int ret ;
2013-05-17 13:26:15 +04:00
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 ) ;
2011-04-27 21:16:32 +04:00
ret = platform_driver_register ( & soc_dummy_driver ) ;
if ( ret ! = 0 )
platform_device_unregister ( soc_dummy_dev ) ;
return ret ;
}
2011-04-28 13:57:54 +04:00
void __exit snd_soc_util_exit ( void )
2011-04-27 21:16:32 +04:00
{
platform_device_unregister ( soc_dummy_dev ) ;
platform_driver_unregister ( & soc_dummy_driver ) ;
}