2011-04-26 12:52:42 +04:00
/*
* trimslice . c - TrimSlice machine ASoC driver
*
* Copyright ( C ) 2011 - CompuLab , Ltd .
* Author : Mike Rapoport < mike @ compulab . co . il >
*
* Based on code copyright / by :
* Author : Stephen Warren < swarren @ nvidia . com >
* Copyright ( C ) 2010 - 2011 - NVIDIA , Inc .
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* version 2 as published by the Free Software Foundation .
*
* This program is distributed in the hope that it will be useful , but
* WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
* General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 51 Franklin St , Fifth Floor , Boston , MA
* 02110 - 1301 USA
*
*/
# include <linux/module.h>
2012-04-27 23:34:19 +04:00
# include <linux/of.h>
2011-04-26 12:52:42 +04:00
# include <linux/platform_device.h>
# include <linux/slab.h>
# include <sound/core.h>
# include <sound/jack.h>
# include <sound/pcm.h>
# include <sound/pcm_params.h>
# include <sound/soc.h>
# include "../codecs/tlv320aic23.h"
# include "tegra_asoc_utils.h"
# define DRV_NAME "tegra-snd-trimslice"
struct tegra_trimslice {
struct tegra_asoc_utils_data util_data ;
} ;
static int trimslice_asoc_hw_params ( struct snd_pcm_substream * substream ,
struct snd_pcm_hw_params * params )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
struct snd_soc_dai * codec_dai = rtd - > codec_dai ;
2014-07-18 00:01:06 +04:00
struct snd_soc_card * card = rtd - > card ;
2011-04-26 12:52:42 +04:00
struct tegra_trimslice * trimslice = snd_soc_card_get_drvdata ( card ) ;
int srate , mclk ;
int err ;
srate = params_rate ( params ) ;
mclk = 128 * srate ;
err = tegra_asoc_utils_set_rate ( & trimslice - > util_data , srate , mclk ) ;
if ( err < 0 ) {
dev_err ( card - > dev , " Can't configure clocks \n " ) ;
return err ;
}
err = snd_soc_dai_set_sysclk ( codec_dai , 0 , mclk ,
SND_SOC_CLOCK_IN ) ;
if ( err < 0 ) {
dev_err ( card - > dev , " codec_dai clock not set \n " ) ;
return err ;
}
return 0 ;
}
static struct snd_soc_ops trimslice_asoc_ops = {
. hw_params = trimslice_asoc_hw_params ,
} ;
static const struct snd_soc_dapm_widget trimslice_dapm_widgets [ ] = {
SND_SOC_DAPM_HP ( " Line Out " , NULL ) ,
SND_SOC_DAPM_LINE ( " Line In " , NULL ) ,
} ;
static const struct snd_soc_dapm_route trimslice_audio_map [ ] = {
{ " Line Out " , NULL , " LOUT " } ,
{ " Line Out " , NULL , " ROUT " } ,
{ " LLINEIN " , NULL , " Line In " } ,
{ " RLINEIN " , NULL , " Line In " } ,
} ;
static struct snd_soc_dai_link trimslice_tlv320aic23_dai = {
. name = " TLV320AIC23 " ,
. stream_name = " AIC23 " ,
. codec_dai_name = " tlv320aic23-hifi " ,
. ops = & trimslice_asoc_ops ,
2012-06-07 03:15:48 +04:00
. dai_fmt = SND_SOC_DAIFMT_I2S |
SND_SOC_DAIFMT_NB_NF |
SND_SOC_DAIFMT_CBS_CFS ,
2011-04-26 12:52:42 +04:00
} ;
static struct snd_soc_card snd_soc_trimslice = {
. name = " tegra-trimslice " ,
2011-12-22 17:23:01 +04:00
. owner = THIS_MODULE ,
2011-04-26 12:52:42 +04:00
. dai_link = & trimslice_tlv320aic23_dai ,
. num_links = 1 ,
. dapm_widgets = trimslice_dapm_widgets ,
. num_dapm_widgets = ARRAY_SIZE ( trimslice_dapm_widgets ) ,
. dapm_routes = trimslice_audio_map ,
. num_dapm_routes = ARRAY_SIZE ( trimslice_audio_map ) ,
2011-11-23 23:42:06 +04:00
. fully_routed = true ,
2011-04-26 12:52:42 +04:00
} ;
2012-12-07 18:26:33 +04:00
static int tegra_snd_trimslice_probe ( struct platform_device * pdev )
2011-04-26 12:52:42 +04:00
{
2013-02-16 04:07:34 +04:00
struct device_node * np = pdev - > dev . of_node ;
2011-04-26 12:52:42 +04:00
struct snd_soc_card * card = & snd_soc_trimslice ;
struct tegra_trimslice * trimslice ;
int ret ;
2011-11-23 05:21:21 +04:00
trimslice = devm_kzalloc ( & pdev - > dev , sizeof ( struct tegra_trimslice ) ,
GFP_KERNEL ) ;
2011-04-26 12:52:42 +04:00
if ( ! trimslice ) {
dev_err ( & pdev - > dev , " Can't allocate tegra_trimslice \n " ) ;
2013-02-16 04:07:34 +04:00
return - ENOMEM ;
}
card - > dev = & pdev - > dev ;
platform_set_drvdata ( pdev , card ) ;
snd_soc_card_set_drvdata ( card , trimslice ) ;
trimslice_tlv320aic23_dai . codec_of_node = of_parse_phandle ( np ,
" nvidia,audio-codec " , 0 ) ;
if ( ! trimslice_tlv320aic23_dai . codec_of_node ) {
dev_err ( & pdev - > dev ,
" Property 'nvidia,audio-codec' missing or invalid \n " ) ;
ret = - EINVAL ;
2011-11-23 05:21:21 +04:00
goto err ;
2011-04-26 12:52:42 +04:00
}
2013-02-16 04:07:34 +04:00
trimslice_tlv320aic23_dai . cpu_of_node = of_parse_phandle ( np ,
" nvidia,i2s-controller " , 0 ) ;
if ( ! trimslice_tlv320aic23_dai . cpu_of_node ) {
dev_err ( & pdev - > dev ,
" Property 'nvidia,i2s-controller' missing or invalid \n " ) ;
ret = - EINVAL ;
goto err ;
2012-04-27 23:34:19 +04:00
}
2013-02-16 04:07:34 +04:00
trimslice_tlv320aic23_dai . platform_of_node =
trimslice_tlv320aic23_dai . cpu_of_node ;
2011-04-26 12:52:42 +04:00
ret = tegra_asoc_utils_init ( & trimslice - > util_data , & pdev - > dev ) ;
if ( ret )
2011-11-23 05:21:21 +04:00
goto err ;
2011-04-26 12:52:42 +04:00
ret = snd_soc_register_card ( card ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " snd_soc_register_card failed (%d) \n " ,
ret ) ;
goto err_fini_utils ;
}
return 0 ;
err_fini_utils :
tegra_asoc_utils_fini ( & trimslice - > util_data ) ;
2011-11-23 05:21:21 +04:00
err :
2011-04-26 12:52:42 +04:00
return ret ;
}
2012-12-07 18:26:33 +04:00
static int tegra_snd_trimslice_remove ( struct platform_device * pdev )
2011-04-26 12:52:42 +04:00
{
struct snd_soc_card * card = platform_get_drvdata ( pdev ) ;
struct tegra_trimslice * trimslice = snd_soc_card_get_drvdata ( card ) ;
snd_soc_unregister_card ( card ) ;
tegra_asoc_utils_fini ( & trimslice - > util_data ) ;
return 0 ;
}
2012-11-19 22:25:33 +04:00
static const struct of_device_id trimslice_of_match [ ] = {
2012-04-27 23:34:19 +04:00
{ . compatible = " nvidia,tegra-audio-trimslice " , } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , trimslice_of_match ) ;
2011-04-26 12:52:42 +04:00
static struct platform_driver tegra_snd_trimslice_driver = {
. driver = {
. name = DRV_NAME ,
2012-04-27 23:34:19 +04:00
. of_match_table = trimslice_of_match ,
2011-04-26 12:52:42 +04:00
} ,
. probe = tegra_snd_trimslice_probe ,
2012-12-07 18:26:33 +04:00
. remove = tegra_snd_trimslice_remove ,
2011-04-26 12:52:42 +04:00
} ;
2011-11-23 05:21:21 +04:00
module_platform_driver ( tegra_snd_trimslice_driver ) ;
2011-04-26 12:52:42 +04:00
MODULE_AUTHOR ( " Mike Rapoport <mike@compulab.co.il> " ) ;
MODULE_DESCRIPTION ( " Trimslice machine ASoC driver " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS ( " platform: " DRV_NAME ) ;