2012-04-11 02:32:00 +04:00
/*
* tegra30_i2s . c - Tegra30 I2S driver
*
* Author : Stephen Warren < swarren @ nvidia . com >
* Copyright ( c ) 2010 - 2012 , NVIDIA CORPORATION . All rights reserved .
*
* Based on code copyright / by :
*
* Copyright ( c ) 2009 - 2010 , NVIDIA Corporation .
* Scott Peterson < speterson @ nvidia . com >
*
* Copyright ( C ) 2010 Google , Inc .
* Iliyan Malchev < malchev @ google . com >
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms and conditions of the GNU General Public License ,
* version 2 , as published by the Free Software Foundation .
*
* This program is distributed in the hope 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 , see < http : //www.gnu.org/licenses/>.
*/
# include <linux/clk.h>
# include <linux/device.h>
# include <linux/io.h>
# include <linux/module.h>
# include <linux/of.h>
2013-10-12 01:43:17 +04:00
# include <linux/of_device.h>
2012-04-11 02:32:00 +04:00
# include <linux/platform_device.h>
# include <linux/pm_runtime.h>
# include <linux/regmap.h>
# include <linux/slab.h>
# include <sound/core.h>
# include <sound/pcm.h>
# include <sound/pcm_params.h>
# include <sound/soc.h>
2013-04-03 13:06:03 +04:00
# include <sound/dmaengine_pcm.h>
2012-04-11 02:32:00 +04:00
# include "tegra30_ahub.h"
# include "tegra30_i2s.h"
# define DRV_NAME "tegra30-i2s"
static int tegra30_i2s_runtime_suspend ( struct device * dev )
{
struct tegra30_i2s * i2s = dev_get_drvdata ( dev ) ;
regcache_cache_only ( i2s - > regmap , true ) ;
2012-06-05 08:29:42 +04:00
clk_disable_unprepare ( i2s - > clk_i2s ) ;
2012-04-11 02:32:00 +04:00
return 0 ;
}
static int tegra30_i2s_runtime_resume ( struct device * dev )
{
struct tegra30_i2s * i2s = dev_get_drvdata ( dev ) ;
int ret ;
2012-06-05 08:29:42 +04:00
ret = clk_prepare_enable ( i2s - > clk_i2s ) ;
2012-04-11 02:32:00 +04:00
if ( ret ) {
dev_err ( dev , " clk_enable failed: %d \n " , ret ) ;
return ret ;
}
regcache_cache_only ( i2s - > regmap , false ) ;
return 0 ;
}
static int tegra30_i2s_set_fmt ( struct snd_soc_dai * dai ,
unsigned int fmt )
{
struct tegra30_i2s * i2s = snd_soc_dai_get_drvdata ( dai ) ;
2013-12-07 00:34:50 +04:00
unsigned int mask = 0 , val = 0 ;
2012-04-11 02:32:00 +04:00
switch ( fmt & SND_SOC_DAIFMT_INV_MASK ) {
case SND_SOC_DAIFMT_NB_NF :
break ;
default :
return - EINVAL ;
}
2013-12-07 00:34:50 +04:00
mask | = TEGRA30_I2S_CTRL_MASTER_ENABLE ;
2012-04-11 02:32:00 +04:00
switch ( fmt & SND_SOC_DAIFMT_MASTER_MASK ) {
case SND_SOC_DAIFMT_CBS_CFS :
2013-12-07 00:34:50 +04:00
val | = TEGRA30_I2S_CTRL_MASTER_ENABLE ;
2012-04-11 02:32:00 +04:00
break ;
case SND_SOC_DAIFMT_CBM_CFM :
break ;
default :
return - EINVAL ;
}
2012-06-07 03:15:06 +04:00
mask | = TEGRA30_I2S_CTRL_FRAME_FORMAT_MASK |
TEGRA30_I2S_CTRL_LRCK_MASK ;
2012-04-11 02:32:00 +04:00
switch ( fmt & SND_SOC_DAIFMT_FORMAT_MASK ) {
case SND_SOC_DAIFMT_DSP_A :
2012-06-07 03:15:06 +04:00
val | = TEGRA30_I2S_CTRL_FRAME_FORMAT_FSYNC ;
val | = TEGRA30_I2S_CTRL_LRCK_L_LOW ;
2012-04-11 02:32:00 +04:00
break ;
case SND_SOC_DAIFMT_DSP_B :
2012-06-07 03:15:06 +04:00
val | = TEGRA30_I2S_CTRL_FRAME_FORMAT_FSYNC ;
val | = TEGRA30_I2S_CTRL_LRCK_R_LOW ;
2012-04-11 02:32:00 +04:00
break ;
case SND_SOC_DAIFMT_I2S :
2012-06-07 03:15:06 +04:00
val | = TEGRA30_I2S_CTRL_FRAME_FORMAT_LRCK ;
val | = TEGRA30_I2S_CTRL_LRCK_L_LOW ;
2012-04-11 02:32:00 +04:00
break ;
case SND_SOC_DAIFMT_RIGHT_J :
2012-06-07 03:15:06 +04:00
val | = TEGRA30_I2S_CTRL_FRAME_FORMAT_LRCK ;
val | = TEGRA30_I2S_CTRL_LRCK_L_LOW ;
2012-04-11 02:32:00 +04:00
break ;
case SND_SOC_DAIFMT_LEFT_J :
2012-06-07 03:15:06 +04:00
val | = TEGRA30_I2S_CTRL_FRAME_FORMAT_LRCK ;
val | = TEGRA30_I2S_CTRL_LRCK_L_LOW ;
2012-04-11 02:32:00 +04:00
break ;
default :
return - EINVAL ;
}
2012-06-07 03:15:06 +04:00
pm_runtime_get_sync ( dai - > dev ) ;
regmap_update_bits ( i2s - > regmap , TEGRA30_I2S_CTRL , mask , val ) ;
pm_runtime_put ( dai - > dev ) ;
2012-04-11 02:32:00 +04:00
return 0 ;
}
static int tegra30_i2s_hw_params ( struct snd_pcm_substream * substream ,
struct snd_pcm_hw_params * params ,
struct snd_soc_dai * dai )
{
2012-06-07 03:15:05 +04:00
struct device * dev = dai - > dev ;
2012-04-11 02:32:00 +04:00
struct tegra30_i2s * i2s = snd_soc_dai_get_drvdata ( dai ) ;
2012-06-07 03:15:06 +04:00
unsigned int mask , val , reg ;
2012-04-11 02:32:00 +04:00
int ret , sample_size , srate , i2sclock , bitcnt ;
2013-10-12 01:43:17 +04:00
struct tegra30_ahub_cif_conf cif_conf ;
2012-04-11 02:32:00 +04:00
if ( params_channels ( params ) ! = 2 )
return - EINVAL ;
2012-06-07 03:15:06 +04:00
mask = TEGRA30_I2S_CTRL_BIT_SIZE_MASK ;
2012-04-11 02:32:00 +04:00
switch ( params_format ( params ) ) {
case SNDRV_PCM_FORMAT_S16_LE :
2012-06-07 03:15:06 +04:00
val = TEGRA30_I2S_CTRL_BIT_SIZE_16 ;
2012-04-11 02:32:00 +04:00
sample_size = 16 ;
break ;
default :
return - EINVAL ;
}
2012-06-07 03:15:06 +04:00
regmap_update_bits ( i2s - > regmap , TEGRA30_I2S_CTRL , mask , val ) ;
2012-04-11 02:32:00 +04:00
srate = params_rate ( params ) ;
/* Final "* 2" required by Tegra hardware */
i2sclock = srate * params_channels ( params ) * sample_size * 2 ;
bitcnt = ( i2sclock / ( 2 * srate ) ) - 1 ;
if ( bitcnt < 0 | | bitcnt > TEGRA30_I2S_TIMING_CHANNEL_BIT_COUNT_MASK_US )
return - EINVAL ;
ret = clk_set_rate ( i2s - > clk_i2s , i2sclock ) ;
if ( ret ) {
dev_err ( dev , " Can't set I2S clock rate: %d \n " , ret ) ;
return ret ;
}
val = bitcnt < < TEGRA30_I2S_TIMING_CHANNEL_BIT_COUNT_SHIFT ;
if ( i2sclock % ( 2 * srate ) )
val | = TEGRA30_I2S_TIMING_NON_SYM_ENABLE ;
2012-06-07 03:15:06 +04:00
regmap_write ( i2s - > regmap , TEGRA30_I2S_TIMING , val ) ;
2012-04-11 02:32:00 +04:00
2013-10-12 01:43:17 +04:00
cif_conf . threshold = 0 ;
cif_conf . audio_channels = 2 ;
cif_conf . client_channels = 2 ;
cif_conf . audio_bits = TEGRA30_AUDIOCIF_BITS_16 ;
cif_conf . client_bits = TEGRA30_AUDIOCIF_BITS_16 ;
cif_conf . expand = 0 ;
cif_conf . stereo_conv = 0 ;
cif_conf . replicate = 0 ;
cif_conf . truncate = 0 ;
cif_conf . mono_conv = 0 ;
2012-04-11 02:32:00 +04:00
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK ) {
2013-10-12 01:43:17 +04:00
cif_conf . direction = TEGRA30_AUDIOCIF_DIRECTION_RX ;
2012-06-07 03:15:06 +04:00
reg = TEGRA30_I2S_CIF_RX_CTRL ;
2012-04-11 02:32:00 +04:00
} else {
2013-10-12 01:43:17 +04:00
cif_conf . direction = TEGRA30_AUDIOCIF_DIRECTION_TX ;
2013-08-15 00:24:16 +04:00
reg = TEGRA30_I2S_CIF_TX_CTRL ;
2012-04-11 02:32:00 +04:00
}
2013-10-12 01:43:17 +04:00
i2s - > soc_data - > set_audio_cif ( i2s - > regmap , reg , & cif_conf ) ;
2012-06-07 03:15:06 +04:00
2012-04-11 02:32:00 +04:00
val = ( 1 < < TEGRA30_I2S_OFFSET_RX_DATA_OFFSET_SHIFT ) |
( 1 < < TEGRA30_I2S_OFFSET_TX_DATA_OFFSET_SHIFT ) ;
2012-06-07 03:15:06 +04:00
regmap_write ( i2s - > regmap , TEGRA30_I2S_OFFSET , val ) ;
2012-04-11 02:32:00 +04:00
return 0 ;
}
static void tegra30_i2s_start_playback ( struct tegra30_i2s * i2s )
{
tegra30_ahub_enable_tx_fifo ( i2s - > playback_fifo_cif ) ;
2012-06-07 03:15:06 +04:00
regmap_update_bits ( i2s - > regmap , TEGRA30_I2S_CTRL ,
TEGRA30_I2S_CTRL_XFER_EN_TX ,
TEGRA30_I2S_CTRL_XFER_EN_TX ) ;
2012-04-11 02:32:00 +04:00
}
static void tegra30_i2s_stop_playback ( struct tegra30_i2s * i2s )
{
tegra30_ahub_disable_tx_fifo ( i2s - > playback_fifo_cif ) ;
2012-06-07 03:15:06 +04:00
regmap_update_bits ( i2s - > regmap , TEGRA30_I2S_CTRL ,
TEGRA30_I2S_CTRL_XFER_EN_TX , 0 ) ;
2012-04-11 02:32:00 +04:00
}
static void tegra30_i2s_start_capture ( struct tegra30_i2s * i2s )
{
tegra30_ahub_enable_rx_fifo ( i2s - > capture_fifo_cif ) ;
2012-06-07 03:15:06 +04:00
regmap_update_bits ( i2s - > regmap , TEGRA30_I2S_CTRL ,
TEGRA30_I2S_CTRL_XFER_EN_RX ,
TEGRA30_I2S_CTRL_XFER_EN_RX ) ;
2012-04-11 02:32:00 +04:00
}
static void tegra30_i2s_stop_capture ( struct tegra30_i2s * i2s )
{
tegra30_ahub_disable_rx_fifo ( i2s - > capture_fifo_cif ) ;
2012-06-07 03:15:06 +04:00
regmap_update_bits ( i2s - > regmap , TEGRA30_I2S_CTRL ,
TEGRA30_I2S_CTRL_XFER_EN_RX , 0 ) ;
2012-04-11 02:32:00 +04:00
}
static int tegra30_i2s_trigger ( struct snd_pcm_substream * substream , int cmd ,
struct snd_soc_dai * dai )
{
struct tegra30_i2s * i2s = snd_soc_dai_get_drvdata ( dai ) ;
switch ( cmd ) {
case SNDRV_PCM_TRIGGER_START :
case SNDRV_PCM_TRIGGER_PAUSE_RELEASE :
case SNDRV_PCM_TRIGGER_RESUME :
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK )
tegra30_i2s_start_playback ( i2s ) ;
else
tegra30_i2s_start_capture ( i2s ) ;
break ;
case SNDRV_PCM_TRIGGER_STOP :
case SNDRV_PCM_TRIGGER_PAUSE_PUSH :
case SNDRV_PCM_TRIGGER_SUSPEND :
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK )
tegra30_i2s_stop_playback ( i2s ) ;
else
tegra30_i2s_stop_capture ( i2s ) ;
break ;
default :
return - EINVAL ;
}
return 0 ;
}
static int tegra30_i2s_probe ( struct snd_soc_dai * dai )
{
struct tegra30_i2s * i2s = snd_soc_dai_get_drvdata ( dai ) ;
dai - > capture_dma_data = & i2s - > capture_dma_data ;
dai - > playback_dma_data = & i2s - > playback_dma_data ;
return 0 ;
}
static struct snd_soc_dai_ops tegra30_i2s_dai_ops = {
. set_fmt = tegra30_i2s_set_fmt ,
. hw_params = tegra30_i2s_hw_params ,
. trigger = tegra30_i2s_trigger ,
} ;
static const struct snd_soc_dai_driver tegra30_i2s_dai_template = {
. probe = tegra30_i2s_probe ,
. playback = {
2012-06-07 03:15:07 +04:00
. stream_name = " Playback " ,
2012-04-11 02:32:00 +04:00
. channels_min = 2 ,
. channels_max = 2 ,
. rates = SNDRV_PCM_RATE_8000_96000 ,
. formats = SNDRV_PCM_FMTBIT_S16_LE ,
} ,
. capture = {
2012-06-07 03:15:07 +04:00
. stream_name = " Capture " ,
2012-04-11 02:32:00 +04:00
. channels_min = 2 ,
. channels_max = 2 ,
. rates = SNDRV_PCM_RATE_8000_96000 ,
. formats = SNDRV_PCM_FMTBIT_S16_LE ,
} ,
. ops = & tegra30_i2s_dai_ops ,
. symmetric_rates = 1 ,
} ;
2013-03-21 14:37:22 +04:00
static const struct snd_soc_component_driver tegra30_i2s_component = {
. name = DRV_NAME ,
} ;
2012-04-11 02:32:00 +04:00
static bool tegra30_i2s_wr_rd_reg ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case TEGRA30_I2S_CTRL :
case TEGRA30_I2S_TIMING :
case TEGRA30_I2S_OFFSET :
case TEGRA30_I2S_CH_CTRL :
case TEGRA30_I2S_SLOT_CTRL :
case TEGRA30_I2S_CIF_RX_CTRL :
case TEGRA30_I2S_CIF_TX_CTRL :
case TEGRA30_I2S_FLOWCTL :
case TEGRA30_I2S_TX_STEP :
case TEGRA30_I2S_FLOW_STATUS :
case TEGRA30_I2S_FLOW_TOTAL :
case TEGRA30_I2S_FLOW_OVER :
case TEGRA30_I2S_FLOW_UNDER :
case TEGRA30_I2S_LCOEF_1_4_0 :
case TEGRA30_I2S_LCOEF_1_4_1 :
case TEGRA30_I2S_LCOEF_1_4_2 :
case TEGRA30_I2S_LCOEF_1_4_3 :
case TEGRA30_I2S_LCOEF_1_4_4 :
case TEGRA30_I2S_LCOEF_1_4_5 :
case TEGRA30_I2S_LCOEF_2_4_0 :
case TEGRA30_I2S_LCOEF_2_4_1 :
case TEGRA30_I2S_LCOEF_2_4_2 :
return true ;
default :
return false ;
2013-10-09 02:55:45 +04:00
}
2012-04-11 02:32:00 +04:00
}
static bool tegra30_i2s_volatile_reg ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case TEGRA30_I2S_FLOW_STATUS :
case TEGRA30_I2S_FLOW_TOTAL :
case TEGRA30_I2S_FLOW_OVER :
case TEGRA30_I2S_FLOW_UNDER :
return true ;
default :
return false ;
2013-10-09 02:55:45 +04:00
}
2012-04-11 02:32:00 +04:00
}
static const struct regmap_config tegra30_i2s_regmap_config = {
. reg_bits = 32 ,
. reg_stride = 4 ,
. val_bits = 32 ,
. max_register = TEGRA30_I2S_LCOEF_2_4_2 ,
. writeable_reg = tegra30_i2s_wr_rd_reg ,
. readable_reg = tegra30_i2s_wr_rd_reg ,
. volatile_reg = tegra30_i2s_volatile_reg ,
. cache_type = REGCACHE_RBTREE ,
} ;
2013-10-12 01:43:17 +04:00
static const struct tegra30_i2s_soc_data tegra30_i2s_config = {
. set_audio_cif = tegra30_ahub_set_cif ,
} ;
static const struct tegra30_i2s_soc_data tegra124_i2s_config = {
. set_audio_cif = tegra124_ahub_set_cif ,
} ;
static const struct of_device_id tegra30_i2s_of_match [ ] = {
{ . compatible = " nvidia,tegra124-i2s " , . data = & tegra124_i2s_config } ,
{ . compatible = " nvidia,tegra30-i2s " , . data = & tegra30_i2s_config } ,
{ } ,
} ;
2012-12-07 18:26:33 +04:00
static int tegra30_i2s_platform_probe ( struct platform_device * pdev )
2012-04-11 02:32:00 +04:00
{
struct tegra30_i2s * i2s ;
2013-10-12 01:43:17 +04:00
const struct of_device_id * match ;
2012-04-11 02:32:00 +04:00
u32 cif_ids [ 2 ] ;
struct resource * mem , * memregion ;
void __iomem * regs ;
int ret ;
i2s = devm_kzalloc ( & pdev - > dev , sizeof ( struct tegra30_i2s ) , GFP_KERNEL ) ;
if ( ! i2s ) {
dev_err ( & pdev - > dev , " Can't allocate tegra30_i2s \n " ) ;
ret = - ENOMEM ;
goto err ;
}
dev_set_drvdata ( & pdev - > dev , i2s ) ;
2013-10-12 01:43:17 +04:00
match = of_match_device ( tegra30_i2s_of_match , & pdev - > dev ) ;
if ( ! match ) {
dev_err ( & pdev - > dev , " Error: No device match found \n " ) ;
ret = - ENODEV ;
goto err ;
}
i2s - > soc_data = ( struct tegra30_i2s_soc_data * ) match - > data ;
2012-04-11 02:32:00 +04:00
i2s - > dai = tegra30_i2s_dai_template ;
i2s - > dai . name = dev_name ( & pdev - > dev ) ;
ret = of_property_read_u32_array ( pdev - > dev . of_node ,
" nvidia,ahub-cif-ids " , cif_ids ,
ARRAY_SIZE ( cif_ids ) ) ;
if ( ret < 0 )
goto err ;
i2s - > playback_i2s_cif = cif_ids [ 0 ] ;
i2s - > capture_i2s_cif = cif_ids [ 1 ] ;
i2s - > clk_i2s = clk_get ( & pdev - > dev , NULL ) ;
if ( IS_ERR ( i2s - > clk_i2s ) ) {
dev_err ( & pdev - > dev , " Can't retrieve i2s clock \n " ) ;
ret = PTR_ERR ( i2s - > clk_i2s ) ;
goto err ;
}
mem = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
if ( ! mem ) {
dev_err ( & pdev - > dev , " No memory resource \n " ) ;
ret = - ENODEV ;
goto err_clk_put ;
}
memregion = devm_request_mem_region ( & pdev - > dev , mem - > start ,
resource_size ( mem ) , DRV_NAME ) ;
if ( ! memregion ) {
dev_err ( & pdev - > dev , " Memory region already claimed \n " ) ;
ret = - EBUSY ;
goto err_clk_put ;
}
regs = devm_ioremap ( & pdev - > dev , mem - > start , resource_size ( mem ) ) ;
if ( ! regs ) {
dev_err ( & pdev - > dev , " ioremap failed \n " ) ;
ret = - ENOMEM ;
goto err_clk_put ;
}
i2s - > regmap = devm_regmap_init_mmio ( & pdev - > dev , regs ,
& tegra30_i2s_regmap_config ) ;
if ( IS_ERR ( i2s - > regmap ) ) {
dev_err ( & pdev - > dev , " regmap init failed \n " ) ;
ret = PTR_ERR ( i2s - > regmap ) ;
goto err_clk_put ;
}
regcache_cache_only ( i2s - > regmap , true ) ;
pm_runtime_enable ( & pdev - > dev ) ;
if ( ! pm_runtime_enabled ( & pdev - > dev ) ) {
ret = tegra30_i2s_runtime_resume ( & pdev - > dev ) ;
if ( ret )
goto err_pm_disable ;
}
2013-11-15 22:48:47 +04:00
i2s - > playback_dma_data . addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES ;
i2s - > playback_dma_data . maxburst = 4 ;
ret = tegra30_ahub_allocate_tx_fifo ( & i2s - > playback_fifo_cif ,
2013-11-12 02:21:01 +04:00
i2s - > playback_dma_chan ,
sizeof ( i2s - > playback_dma_chan ) ,
& i2s - > playback_dma_data . addr ) ;
2013-11-15 22:48:47 +04:00
if ( ret ) {
dev_err ( & pdev - > dev , " Could not alloc TX FIFO: %d \n " , ret ) ;
goto err_suspend ;
}
ret = tegra30_ahub_set_rx_cif_source ( i2s - > playback_i2s_cif ,
i2s - > playback_fifo_cif ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " Could not route TX FIFO: %d \n " , ret ) ;
goto err_free_tx_fifo ;
}
i2s - > capture_dma_data . addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES ;
i2s - > capture_dma_data . maxburst = 4 ;
ret = tegra30_ahub_allocate_rx_fifo ( & i2s - > capture_fifo_cif ,
2013-11-12 02:21:01 +04:00
i2s - > capture_dma_chan ,
sizeof ( i2s - > capture_dma_chan ) ,
& i2s - > capture_dma_data . addr ) ;
2013-11-15 22:48:47 +04:00
if ( ret ) {
dev_err ( & pdev - > dev , " Could not alloc RX FIFO: %d \n " , ret ) ;
goto err_unroute_tx_fifo ;
}
ret = tegra30_ahub_set_rx_cif_source ( i2s - > capture_fifo_cif ,
i2s - > capture_i2s_cif ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " Could not route TX FIFO: %d \n " , ret ) ;
goto err_free_rx_fifo ;
}
2013-03-21 14:37:22 +04:00
ret = snd_soc_register_component ( & pdev - > dev , & tegra30_i2s_component ,
& i2s - > dai , 1 ) ;
2012-04-11 02:32:00 +04:00
if ( ret ) {
dev_err ( & pdev - > dev , " Could not register DAI: %d \n " , ret ) ;
ret = - ENOMEM ;
2013-11-15 22:48:47 +04:00
goto err_unroute_rx_fifo ;
2012-04-11 02:32:00 +04:00
}
2013-11-12 02:21:01 +04:00
ret = tegra_pcm_platform_register_with_chan_names ( & pdev - > dev ,
& i2s - > dma_config , i2s - > playback_dma_chan ,
i2s - > capture_dma_chan ) ;
2012-04-11 02:32:00 +04:00
if ( ret ) {
dev_err ( & pdev - > dev , " Could not register PCM: %d \n " , ret ) ;
2013-03-21 14:37:22 +04:00
goto err_unregister_component ;
2012-04-11 02:32:00 +04:00
}
return 0 ;
2013-03-21 14:37:22 +04:00
err_unregister_component :
snd_soc_unregister_component ( & pdev - > dev ) ;
2013-11-15 22:48:47 +04:00
err_unroute_rx_fifo :
tegra30_ahub_unset_rx_cif_source ( i2s - > capture_fifo_cif ) ;
err_free_rx_fifo :
tegra30_ahub_free_rx_fifo ( i2s - > capture_fifo_cif ) ;
err_unroute_tx_fifo :
tegra30_ahub_unset_rx_cif_source ( i2s - > playback_i2s_cif ) ;
err_free_tx_fifo :
tegra30_ahub_free_tx_fifo ( i2s - > playback_fifo_cif ) ;
2012-04-11 02:32:00 +04:00
err_suspend :
if ( ! pm_runtime_status_suspended ( & pdev - > dev ) )
tegra30_i2s_runtime_suspend ( & pdev - > dev ) ;
err_pm_disable :
pm_runtime_disable ( & pdev - > dev ) ;
err_clk_put :
clk_put ( i2s - > clk_i2s ) ;
err :
return ret ;
}
2012-12-07 18:26:33 +04:00
static int tegra30_i2s_platform_remove ( struct platform_device * pdev )
2012-04-11 02:32:00 +04:00
{
struct tegra30_i2s * i2s = dev_get_drvdata ( & pdev - > dev ) ;
pm_runtime_disable ( & pdev - > dev ) ;
if ( ! pm_runtime_status_suspended ( & pdev - > dev ) )
tegra30_i2s_runtime_suspend ( & pdev - > dev ) ;
tegra_pcm_platform_unregister ( & pdev - > dev ) ;
2013-03-21 14:37:22 +04:00
snd_soc_unregister_component ( & pdev - > dev ) ;
2012-04-11 02:32:00 +04:00
2013-11-15 22:48:47 +04:00
tegra30_ahub_unset_rx_cif_source ( i2s - > capture_fifo_cif ) ;
tegra30_ahub_free_rx_fifo ( i2s - > capture_fifo_cif ) ;
tegra30_ahub_unset_rx_cif_source ( i2s - > playback_i2s_cif ) ;
tegra30_ahub_free_tx_fifo ( i2s - > playback_fifo_cif ) ;
2012-04-11 02:32:00 +04:00
clk_put ( i2s - > clk_i2s ) ;
return 0 ;
}
2013-06-03 21:37:42 +04:00
# ifdef CONFIG_PM_SLEEP
static int tegra30_i2s_suspend ( struct device * dev )
{
struct tegra30_i2s * i2s = dev_get_drvdata ( dev ) ;
regcache_mark_dirty ( i2s - > regmap ) ;
return 0 ;
}
static int tegra30_i2s_resume ( struct device * dev )
{
struct tegra30_i2s * i2s = dev_get_drvdata ( dev ) ;
2013-06-04 22:58:14 +04:00
int ret ;
2013-06-03 21:37:42 +04:00
2013-06-04 22:58:14 +04:00
ret = pm_runtime_get_sync ( dev ) ;
if ( ret < 0 )
return ret ;
ret = regcache_sync ( i2s - > regmap ) ;
pm_runtime_put ( dev ) ;
return ret ;
2013-06-03 21:37:42 +04:00
}
# endif
2012-11-19 22:25:33 +04:00
static const struct dev_pm_ops tegra30_i2s_pm_ops = {
2012-04-11 02:32:00 +04:00
SET_RUNTIME_PM_OPS ( tegra30_i2s_runtime_suspend ,
tegra30_i2s_runtime_resume , NULL )
2013-06-03 21:37:42 +04:00
SET_SYSTEM_SLEEP_PM_OPS ( tegra30_i2s_suspend , tegra30_i2s_resume )
2012-04-11 02:32:00 +04:00
} ;
static struct platform_driver tegra30_i2s_driver = {
. driver = {
. name = DRV_NAME ,
. owner = THIS_MODULE ,
. of_match_table = tegra30_i2s_of_match ,
. pm = & tegra30_i2s_pm_ops ,
} ,
. probe = tegra30_i2s_platform_probe ,
2012-12-07 18:26:33 +04:00
. remove = tegra30_i2s_platform_remove ,
2012-04-11 02:32:00 +04:00
} ;
module_platform_driver ( tegra30_i2s_driver ) ;
MODULE_AUTHOR ( " Stephen Warren <swarren@nvidia.com> " ) ;
MODULE_DESCRIPTION ( " Tegra30 I2S ASoC driver " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS ( " platform: " DRV_NAME ) ;
MODULE_DEVICE_TABLE ( of , tegra30_i2s_of_match ) ;