2019-05-29 16:57:50 -07:00
// SPDX-License-Identifier: GPL-2.0-only
2011-01-07 22:36:14 -07:00
/*
2012-04-05 15:54:53 -06:00
* tegra20_i2s . c - Tegra20 I2S driver
2011-01-07 22:36:14 -07:00
*
* Author : Stephen Warren < swarren @ nvidia . com >
2012-03-20 14:55:49 -06:00
* Copyright ( C ) 2010 , 2012 - NVIDIA , Inc .
2011-01-07 22:36:14 -07:00
*
* 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 >
*/
# include <linux/clk.h>
# include <linux/device.h>
2012-04-06 11:12:25 -06:00
# include <linux/io.h>
# include <linux/module.h>
# include <linux/of.h>
2011-01-07 22:36:14 -07:00
# include <linux/platform_device.h>
2012-04-09 09:52:22 -06:00
# include <linux/pm_runtime.h>
2012-04-13 12:14:06 -06:00
# include <linux/regmap.h>
2021-03-14 18:44:44 +03:00
# include <linux/reset.h>
2011-01-07 22:36:14 -07:00
# include <linux/slab.h>
# include <sound/core.h>
# include <sound/pcm.h>
# include <sound/pcm_params.h>
# include <sound/soc.h>
2013-04-03 11:06:03 +02:00
# include <sound/dmaengine_pcm.h>
2011-01-07 22:36:14 -07:00
2012-04-05 15:54:53 -06:00
# include "tegra20_i2s.h"
2011-01-07 22:36:14 -07:00
2012-04-06 10:30:52 -06:00
# define DRV_NAME "tegra20-i2s"
2011-01-07 22:36:14 -07:00
2021-04-22 15:34:00 +02:00
static __maybe_unused int tegra20_i2s_runtime_suspend ( struct device * dev )
2012-04-09 09:52:22 -06:00
{
struct tegra20_i2s * i2s = dev_get_drvdata ( dev ) ;
2021-03-14 18:44:44 +03:00
regcache_cache_only ( i2s - > regmap , true ) ;
2012-06-05 09:59:42 +05:30
clk_disable_unprepare ( i2s - > clk_i2s ) ;
2012-04-09 09:52:22 -06:00
return 0 ;
}
2021-04-22 15:34:00 +02:00
static __maybe_unused int tegra20_i2s_runtime_resume ( struct device * dev )
2012-04-09 09:52:22 -06:00
{
struct tegra20_i2s * i2s = dev_get_drvdata ( dev ) ;
int ret ;
2021-03-14 18:44:44 +03:00
ret = reset_control_assert ( i2s - > reset ) ;
if ( ret )
return ret ;
2012-06-05 09:59:42 +05:30
ret = clk_prepare_enable ( i2s - > clk_i2s ) ;
2012-04-09 09:52:22 -06:00
if ( ret ) {
dev_err ( dev , " clk_enable failed: %d \n " , ret ) ;
return ret ;
}
2021-03-14 18:44:44 +03:00
usleep_range ( 10 , 100 ) ;
ret = reset_control_deassert ( i2s - > reset ) ;
if ( ret )
goto disable_clocks ;
regcache_cache_only ( i2s - > regmap , false ) ;
regcache_mark_dirty ( i2s - > regmap ) ;
ret = regcache_sync ( i2s - > regmap ) ;
if ( ret )
goto disable_clocks ;
2012-04-09 09:52:22 -06:00
return 0 ;
2021-03-14 18:44:44 +03:00
disable_clocks :
clk_disable_unprepare ( i2s - > clk_i2s ) ;
return ret ;
2012-04-09 09:52:22 -06:00
}
2012-04-06 10:30:52 -06:00
static int tegra20_i2s_set_fmt ( struct snd_soc_dai * dai ,
2011-01-07 22:36:14 -07:00
unsigned int fmt )
{
2012-04-06 10:30:52 -06:00
struct tegra20_i2s * i2s = snd_soc_dai_get_drvdata ( dai ) ;
2013-12-06 13:34:50 -07:00
unsigned int mask = 0 , val = 0 ;
2011-01-07 22:36:14 -07:00
switch ( fmt & SND_SOC_DAIFMT_INV_MASK ) {
case SND_SOC_DAIFMT_NB_NF :
break ;
default :
return - EINVAL ;
}
2013-12-06 13:34:50 -07:00
mask | = TEGRA20_I2S_CTRL_MASTER_ENABLE ;
2022-05-19 16:42:45 +01:00
switch ( fmt & SND_SOC_DAIFMT_CLOCK_PROVIDER_MASK ) {
case SND_SOC_DAIFMT_BP_FP :
2013-12-06 13:34:50 -07:00
val | = TEGRA20_I2S_CTRL_MASTER_ENABLE ;
2011-01-07 22:36:14 -07:00
break ;
2022-05-19 16:42:45 +01:00
case SND_SOC_DAIFMT_BC_FC :
2011-01-07 22:36:14 -07:00
break ;
default :
return - EINVAL ;
}
2012-06-06 17:15:06 -06:00
mask | = TEGRA20_I2S_CTRL_BIT_FORMAT_MASK |
TEGRA20_I2S_CTRL_LRCK_MASK ;
2011-01-07 22:36:14 -07:00
switch ( fmt & SND_SOC_DAIFMT_FORMAT_MASK ) {
case SND_SOC_DAIFMT_DSP_A :
2012-06-06 17:15:06 -06:00
val | = TEGRA20_I2S_CTRL_BIT_FORMAT_DSP ;
val | = TEGRA20_I2S_CTRL_LRCK_L_LOW ;
2011-01-07 22:36:14 -07:00
break ;
case SND_SOC_DAIFMT_DSP_B :
2012-06-06 17:15:06 -06:00
val | = TEGRA20_I2S_CTRL_BIT_FORMAT_DSP ;
val | = TEGRA20_I2S_CTRL_LRCK_R_LOW ;
2011-01-07 22:36:14 -07:00
break ;
case SND_SOC_DAIFMT_I2S :
2012-06-06 17:15:06 -06:00
val | = TEGRA20_I2S_CTRL_BIT_FORMAT_I2S ;
val | = TEGRA20_I2S_CTRL_LRCK_L_LOW ;
2011-01-07 22:36:14 -07:00
break ;
case SND_SOC_DAIFMT_RIGHT_J :
2012-06-06 17:15:06 -06:00
val | = TEGRA20_I2S_CTRL_BIT_FORMAT_RJM ;
val | = TEGRA20_I2S_CTRL_LRCK_L_LOW ;
2011-01-07 22:36:14 -07:00
break ;
case SND_SOC_DAIFMT_LEFT_J :
2012-06-06 17:15:06 -06:00
val | = TEGRA20_I2S_CTRL_BIT_FORMAT_LJM ;
val | = TEGRA20_I2S_CTRL_LRCK_L_LOW ;
2011-01-07 22:36:14 -07:00
break ;
default :
return - EINVAL ;
}
2012-06-06 17:15:06 -06:00
regmap_update_bits ( i2s - > regmap , TEGRA20_I2S_CTRL , mask , val ) ;
2011-01-07 22:36:14 -07:00
return 0 ;
}
2012-04-06 10:30:52 -06:00
static int tegra20_i2s_hw_params ( struct snd_pcm_substream * substream ,
struct snd_pcm_hw_params * params ,
struct snd_soc_dai * dai )
2011-01-07 22:36:14 -07:00
{
2012-06-06 17:15:05 -06:00
struct device * dev = dai - > dev ;
2012-04-06 10:30:52 -06:00
struct tegra20_i2s * i2s = snd_soc_dai_get_drvdata ( dai ) ;
2012-06-06 17:15:06 -06:00
unsigned int mask , val ;
2011-01-07 22:36:14 -07:00
int ret , sample_size , srate , i2sclock , bitcnt ;
2012-06-06 17:15:06 -06:00
mask = TEGRA20_I2S_CTRL_BIT_SIZE_MASK ;
2011-01-07 22:36:14 -07:00
switch ( params_format ( params ) ) {
case SNDRV_PCM_FORMAT_S16_LE :
2012-06-06 17:15:06 -06:00
val = TEGRA20_I2S_CTRL_BIT_SIZE_16 ;
2011-01-07 22:36:14 -07:00
sample_size = 16 ;
break ;
case SNDRV_PCM_FORMAT_S24_LE :
2012-06-06 17:15:06 -06:00
val = TEGRA20_I2S_CTRL_BIT_SIZE_24 ;
2011-01-07 22:36:14 -07:00
sample_size = 24 ;
break ;
case SNDRV_PCM_FORMAT_S32_LE :
2012-06-06 17:15:06 -06:00
val = TEGRA20_I2S_CTRL_BIT_SIZE_32 ;
2011-01-07 22:36:14 -07:00
sample_size = 32 ;
break ;
default :
return - EINVAL ;
}
2012-06-06 17:15:06 -06:00
mask | = TEGRA20_I2S_CTRL_FIFO_FORMAT_MASK ;
val | = TEGRA20_I2S_CTRL_FIFO_FORMAT_PACKED ;
regmap_update_bits ( i2s - > regmap , TEGRA20_I2S_CTRL , mask , val ) ;
2011-01-07 22:36:14 -07:00
srate = params_rate ( params ) ;
/* Final "* 2" required by Tegra hardware */
i2sclock = srate * params_channels ( params ) * sample_size * 2 ;
ret = clk_set_rate ( i2s - > clk_i2s , i2sclock ) ;
if ( ret ) {
dev_err ( dev , " Can't set I2S clock rate: %d \n " , ret ) ;
return ret ;
}
bitcnt = ( i2sclock / ( 2 * srate ) ) - 1 ;
2012-04-06 10:30:52 -06:00
if ( bitcnt < 0 | | bitcnt > TEGRA20_I2S_TIMING_CHANNEL_BIT_COUNT_MASK_US )
2011-01-07 22:36:14 -07:00
return - EINVAL ;
2012-06-06 17:15:06 -06:00
val = bitcnt < < TEGRA20_I2S_TIMING_CHANNEL_BIT_COUNT_SHIFT ;
2011-01-07 22:36:14 -07:00
if ( i2sclock % ( 2 * srate ) )
2012-06-06 17:15:06 -06:00
val | = TEGRA20_I2S_TIMING_NON_SYM_ENABLE ;
2011-01-07 22:36:14 -07:00
2012-06-06 17:15:06 -06:00
regmap_write ( i2s - > regmap , TEGRA20_I2S_TIMING , val ) ;
2011-01-07 22:36:14 -07:00
2012-06-06 17:15:06 -06:00
regmap_write ( i2s - > regmap , TEGRA20_I2S_FIFO_SCR ,
TEGRA20_I2S_FIFO_SCR_FIFO2_ATN_LVL_FOUR_SLOTS |
TEGRA20_I2S_FIFO_SCR_FIFO1_ATN_LVL_FOUR_SLOTS ) ;
2011-01-07 22:36:14 -07:00
return 0 ;
}
2012-04-06 10:30:52 -06:00
static void tegra20_i2s_start_playback ( struct tegra20_i2s * i2s )
2011-01-07 22:36:14 -07:00
{
2012-06-06 17:15:06 -06:00
regmap_update_bits ( i2s - > regmap , TEGRA20_I2S_CTRL ,
TEGRA20_I2S_CTRL_FIFO1_ENABLE ,
TEGRA20_I2S_CTRL_FIFO1_ENABLE ) ;
2011-01-07 22:36:14 -07:00
}
2012-04-06 10:30:52 -06:00
static void tegra20_i2s_stop_playback ( struct tegra20_i2s * i2s )
2011-01-07 22:36:14 -07:00
{
2012-06-06 17:15:06 -06:00
regmap_update_bits ( i2s - > regmap , TEGRA20_I2S_CTRL ,
TEGRA20_I2S_CTRL_FIFO1_ENABLE , 0 ) ;
2011-01-07 22:36:14 -07:00
}
2012-04-06 10:30:52 -06:00
static void tegra20_i2s_start_capture ( struct tegra20_i2s * i2s )
2011-01-07 22:36:14 -07:00
{
2012-06-06 17:15:06 -06:00
regmap_update_bits ( i2s - > regmap , TEGRA20_I2S_CTRL ,
TEGRA20_I2S_CTRL_FIFO2_ENABLE ,
TEGRA20_I2S_CTRL_FIFO2_ENABLE ) ;
2011-01-07 22:36:14 -07:00
}
2012-04-06 10:30:52 -06:00
static void tegra20_i2s_stop_capture ( struct tegra20_i2s * i2s )
2011-01-07 22:36:14 -07:00
{
2012-06-06 17:15:06 -06:00
regmap_update_bits ( i2s - > regmap , TEGRA20_I2S_CTRL ,
TEGRA20_I2S_CTRL_FIFO2_ENABLE , 0 ) ;
2011-01-07 22:36:14 -07:00
}
2012-04-06 10:30:52 -06:00
static int tegra20_i2s_trigger ( struct snd_pcm_substream * substream , int cmd ,
struct snd_soc_dai * dai )
2011-01-07 22:36:14 -07:00
{
2012-04-06 10:30:52 -06:00
struct tegra20_i2s * i2s = snd_soc_dai_get_drvdata ( dai ) ;
2011-01-07 22:36:14 -07:00
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 )
2012-04-06 10:30:52 -06:00
tegra20_i2s_start_playback ( i2s ) ;
2011-01-07 22:36:14 -07:00
else
2012-04-06 10:30:52 -06:00
tegra20_i2s_start_capture ( i2s ) ;
2011-01-07 22:36:14 -07:00
break ;
case SNDRV_PCM_TRIGGER_STOP :
case SNDRV_PCM_TRIGGER_PAUSE_PUSH :
case SNDRV_PCM_TRIGGER_SUSPEND :
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK )
2012-04-06 10:30:52 -06:00
tegra20_i2s_stop_playback ( i2s ) ;
2011-01-07 22:36:14 -07:00
else
2012-04-06 10:30:52 -06:00
tegra20_i2s_stop_capture ( i2s ) ;
2011-01-07 22:36:14 -07:00
break ;
default :
return - EINVAL ;
}
return 0 ;
}
2012-04-06 10:30:52 -06:00
static int tegra20_i2s_probe ( struct snd_soc_dai * dai )
2011-01-07 22:36:14 -07:00
{
2012-04-06 10:30:52 -06:00
struct tegra20_i2s * i2s = snd_soc_dai_get_drvdata ( dai ) ;
2011-01-07 22:36:14 -07:00
dai - > capture_dma_data = & i2s - > capture_dma_data ;
dai - > playback_dma_data = & i2s - > playback_dma_data ;
return 0 ;
}
2021-12-04 17:37:16 +03:00
static const unsigned int tegra20_i2s_rates [ ] = {
8000 , 11025 , 16000 , 22050 , 32000 , 44100 , 48000 , 64000 , 88200 , 96000
} ;
static int tegra20_i2s_filter_rates ( struct snd_pcm_hw_params * params ,
struct snd_pcm_hw_rule * rule )
{
struct snd_interval * r = hw_param_interval ( params , rule - > var ) ;
struct snd_soc_dai * dai = rule - > private ;
struct tegra20_i2s * i2s = dev_get_drvdata ( dai - > dev ) ;
struct clk * parent = clk_get_parent ( i2s - > clk_i2s ) ;
long i , parent_rate , valid_rates = 0 ;
parent_rate = clk_get_rate ( parent ) ;
if ( parent_rate < = 0 ) {
dev_err ( dai - > dev , " Can't get parent clock rate: %ld \n " ,
parent_rate ) ;
return parent_rate ? : - EINVAL ;
}
for ( i = 0 ; i < ARRAY_SIZE ( tegra20_i2s_rates ) ; i + + ) {
if ( parent_rate % ( tegra20_i2s_rates [ i ] * 128 ) = = 0 )
valid_rates | = BIT ( i ) ;
}
/*
* At least one rate must be valid , otherwise the parent clock isn ' t
* audio PLL . Nothing should be filtered in this case .
*/
if ( ! valid_rates )
valid_rates = BIT ( ARRAY_SIZE ( tegra20_i2s_rates ) ) - 1 ;
return snd_interval_list ( r , ARRAY_SIZE ( tegra20_i2s_rates ) ,
tegra20_i2s_rates , valid_rates ) ;
}
static int tegra20_i2s_startup ( struct snd_pcm_substream * substream ,
struct snd_soc_dai * dai )
{
if ( ! device_property_read_bool ( dai - > dev , " nvidia,fixed-parent-rate " ) )
return 0 ;
return snd_pcm_hw_rule_add ( substream - > runtime , 0 ,
SNDRV_PCM_HW_PARAM_RATE ,
tegra20_i2s_filter_rates , dai ,
SNDRV_PCM_HW_PARAM_RATE , - 1 ) ;
}
2012-04-06 10:30:52 -06:00
static const struct snd_soc_dai_ops tegra20_i2s_dai_ops = {
2022-05-19 16:43:12 +01:00
. set_fmt = tegra20_i2s_set_fmt ,
2012-04-06 10:30:52 -06:00
. hw_params = tegra20_i2s_hw_params ,
. trigger = tegra20_i2s_trigger ,
2021-12-04 17:37:16 +03:00
. startup = tegra20_i2s_startup ,
2011-01-07 22:36:14 -07:00
} ;
2012-04-06 10:30:52 -06:00
static const struct snd_soc_dai_driver tegra20_i2s_dai_template = {
. probe = tegra20_i2s_probe ,
2011-11-23 13:33:25 -07:00
. playback = {
2012-06-06 17:15:07 -06:00
. stream_name = " Playback " ,
2011-11-23 13:33:25 -07:00
. channels_min = 2 ,
. channels_max = 2 ,
. rates = SNDRV_PCM_RATE_8000_96000 ,
. formats = SNDRV_PCM_FMTBIT_S16_LE ,
2011-01-07 22:36:14 -07:00
} ,
2011-11-23 13:33:25 -07:00
. capture = {
2012-06-06 17:15:07 -06:00
. stream_name = " Capture " ,
2011-11-23 13:33:25 -07:00
. channels_min = 2 ,
. channels_max = 2 ,
. rates = SNDRV_PCM_RATE_8000_96000 ,
. formats = SNDRV_PCM_FMTBIT_S16_LE ,
2011-01-07 22:36:14 -07:00
} ,
2012-04-06 10:30:52 -06:00
. ops = & tegra20_i2s_dai_ops ,
2021-01-15 13:53:33 +09:00
. symmetric_rate = 1 ,
2011-01-07 22:36:14 -07:00
} ;
2013-03-21 03:37:55 -07:00
static const struct snd_soc_component_driver tegra20_i2s_component = {
2022-06-23 13:51:25 +01:00
. name = DRV_NAME ,
. legacy_dai_naming = 1 ,
2013-03-21 03:37:55 -07:00
} ;
2012-04-13 12:14:06 -06:00
static bool tegra20_i2s_wr_rd_reg ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case TEGRA20_I2S_CTRL :
case TEGRA20_I2S_STATUS :
case TEGRA20_I2S_TIMING :
case TEGRA20_I2S_FIFO_SCR :
case TEGRA20_I2S_PCM_CTRL :
case TEGRA20_I2S_NW_CTRL :
case TEGRA20_I2S_TDM_CTRL :
case TEGRA20_I2S_TDM_TX_RX_CTRL :
case TEGRA20_I2S_FIFO1 :
case TEGRA20_I2S_FIFO2 :
return true ;
default :
return false ;
2013-10-08 15:55:45 -07:00
}
2012-04-13 12:14:06 -06:00
}
static bool tegra20_i2s_volatile_reg ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case TEGRA20_I2S_STATUS :
case TEGRA20_I2S_FIFO_SCR :
case TEGRA20_I2S_FIFO1 :
case TEGRA20_I2S_FIFO2 :
return true ;
default :
return false ;
2013-10-08 15:55:45 -07:00
}
2012-04-13 12:14:06 -06:00
}
static bool tegra20_i2s_precious_reg ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case TEGRA20_I2S_FIFO1 :
case TEGRA20_I2S_FIFO2 :
return true ;
default :
return false ;
2013-10-08 15:55:45 -07:00
}
2012-04-13 12:14:06 -06:00
}
static const struct regmap_config tegra20_i2s_regmap_config = {
. reg_bits = 32 ,
. reg_stride = 4 ,
. val_bits = 32 ,
. max_register = TEGRA20_I2S_FIFO2 ,
. writeable_reg = tegra20_i2s_wr_rd_reg ,
. readable_reg = tegra20_i2s_wr_rd_reg ,
. volatile_reg = tegra20_i2s_volatile_reg ,
. precious_reg = tegra20_i2s_precious_reg ,
2014-03-17 22:08:49 -07:00
. cache_type = REGCACHE_FLAT ,
2012-04-13 12:14:06 -06:00
} ;
2012-12-07 09:26:33 -05:00
static int tegra20_i2s_platform_probe ( struct platform_device * pdev )
2011-01-07 22:36:14 -07:00
{
2012-04-06 10:30:52 -06:00
struct tegra20_i2s * i2s ;
2015-08-23 23:32:14 +08:00
struct resource * mem ;
2012-04-13 12:14:06 -06:00
void __iomem * regs ;
2011-01-07 22:36:14 -07:00
int ret ;
2012-04-06 10:30:52 -06:00
i2s = devm_kzalloc ( & pdev - > dev , sizeof ( struct tegra20_i2s ) , GFP_KERNEL ) ;
2011-01-07 22:36:14 -07:00
if ( ! i2s ) {
ret = - ENOMEM ;
2011-11-22 18:21:16 -07:00
goto err ;
2011-01-07 22:36:14 -07:00
}
dev_set_drvdata ( & pdev - > dev , i2s ) ;
2012-04-06 10:30:52 -06:00
i2s - > dai = tegra20_i2s_dai_template ;
2011-11-23 13:33:25 -07:00
i2s - > dai . name = dev_name ( & pdev - > dev ) ;
2021-03-14 18:44:44 +03:00
i2s - > reset = devm_reset_control_get_exclusive ( & pdev - > dev , " i2s " ) ;
if ( IS_ERR ( i2s - > reset ) ) {
dev_err ( & pdev - > dev , " Can't retrieve i2s reset \n " ) ;
return PTR_ERR ( i2s - > reset ) ;
}
2021-03-14 18:44:52 +03:00
i2s - > clk_i2s = devm_clk_get ( & pdev - > dev , NULL ) ;
2011-01-11 12:48:53 -07:00
if ( IS_ERR ( i2s - > clk_i2s ) ) {
2011-01-28 14:26:41 -07:00
dev_err ( & pdev - > dev , " Can't retrieve i2s clock \n " ) ;
2011-01-07 22:36:14 -07:00
ret = PTR_ERR ( i2s - > clk_i2s ) ;
2011-11-22 18:21:16 -07:00
goto err ;
2011-01-07 22:36:14 -07:00
}
2021-06-18 10:47:19 +08:00
regs = devm_platform_get_and_ioremap_resource ( pdev , 0 , & mem ) ;
2015-08-23 23:32:14 +08:00
if ( IS_ERR ( regs ) ) {
ret = PTR_ERR ( regs ) ;
2021-03-14 18:44:52 +03:00
goto err ;
2011-01-07 22:36:14 -07:00
}
2012-04-13 12:14:06 -06:00
i2s - > regmap = devm_regmap_init_mmio ( & pdev - > dev , regs ,
& tegra20_i2s_regmap_config ) ;
if ( IS_ERR ( i2s - > regmap ) ) {
dev_err ( & pdev - > dev , " regmap init failed \n " ) ;
ret = PTR_ERR ( i2s - > regmap ) ;
2021-03-14 18:44:52 +03:00
goto err ;
2012-04-13 12:14:06 -06:00
}
2012-04-06 10:30:52 -06:00
i2s - > capture_dma_data . addr = mem - > start + TEGRA20_I2S_FIFO2 ;
2013-04-03 11:06:03 +02:00
i2s - > capture_dma_data . addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES ;
i2s - > capture_dma_data . maxburst = 4 ;
2011-01-07 22:36:14 -07:00
2012-04-06 10:30:52 -06:00
i2s - > playback_dma_data . addr = mem - > start + TEGRA20_I2S_FIFO1 ;
2013-04-03 11:06:03 +02:00
i2s - > playback_dma_data . addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES ;
i2s - > playback_dma_data . maxburst = 4 ;
2011-01-07 22:36:14 -07:00
2012-04-09 09:52:22 -06:00
pm_runtime_enable ( & pdev - > dev ) ;
2013-03-21 03:37:55 -07:00
ret = snd_soc_register_component ( & pdev - > dev , & tegra20_i2s_component ,
& i2s - > dai , 1 ) ;
2011-01-07 22:36:14 -07:00
if ( ret ) {
dev_err ( & pdev - > dev , " Could not register DAI: %d \n " , ret ) ;
ret = - ENOMEM ;
2021-03-14 18:44:53 +03:00
goto err_pm_disable ;
2011-01-07 22:36:14 -07:00
}
2012-03-20 14:55:49 -06:00
ret = tegra_pcm_platform_register ( & pdev - > dev ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " Could not register PCM: %d \n " , ret ) ;
2013-03-21 03:37:55 -07:00
goto err_unregister_component ;
2012-03-20 14:55:49 -06:00
}
2011-01-07 22:36:14 -07:00
return 0 ;
2013-03-21 03:37:55 -07:00
err_unregister_component :
snd_soc_unregister_component ( & pdev - > dev ) ;
2012-04-09 09:52:22 -06:00
err_pm_disable :
pm_runtime_disable ( & pdev - > dev ) ;
2011-11-22 18:21:16 -07:00
err :
2011-01-07 22:36:14 -07:00
return ret ;
}
2012-12-07 09:26:33 -05:00
static int tegra20_i2s_platform_remove ( struct platform_device * pdev )
2011-01-07 22:36:14 -07:00
{
2021-03-14 18:44:51 +03:00
tegra_pcm_platform_unregister ( & pdev - > dev ) ;
snd_soc_unregister_component ( & pdev - > dev ) ;
2012-04-09 09:52:22 -06:00
pm_runtime_disable ( & pdev - > dev ) ;
2011-01-07 22:36:14 -07:00
return 0 ;
}
2012-11-19 13:25:33 -05:00
static const struct of_device_id tegra20_i2s_of_match [ ] = {
2011-11-29 18:36:48 -07:00
{ . compatible = " nvidia,tegra20-i2s " , } ,
{ } ,
} ;
2012-11-19 13:25:33 -05:00
static const struct dev_pm_ops tegra20_i2s_pm_ops = {
2012-04-09 09:52:22 -06:00
SET_RUNTIME_PM_OPS ( tegra20_i2s_runtime_suspend ,
tegra20_i2s_runtime_resume , NULL )
2021-03-14 18:44:50 +03:00
SET_SYSTEM_SLEEP_PM_OPS ( pm_runtime_force_suspend ,
pm_runtime_force_resume )
2012-04-09 09:52:22 -06:00
} ;
2012-04-06 10:30:52 -06:00
static struct platform_driver tegra20_i2s_driver = {
2011-01-07 22:36:14 -07:00
. driver = {
. name = DRV_NAME ,
2012-04-06 10:30:52 -06:00
. of_match_table = tegra20_i2s_of_match ,
2012-04-09 09:52:22 -06:00
. pm = & tegra20_i2s_pm_ops ,
2011-01-07 22:36:14 -07:00
} ,
2012-04-06 10:30:52 -06:00
. probe = tegra20_i2s_platform_probe ,
2012-12-07 09:26:33 -05:00
. remove = tegra20_i2s_platform_remove ,
2011-01-07 22:36:14 -07:00
} ;
2012-04-06 10:30:52 -06:00
module_platform_driver ( tegra20_i2s_driver ) ;
2011-01-07 22:36:14 -07:00
MODULE_AUTHOR ( " Stephen Warren <swarren@nvidia.com> " ) ;
2012-04-06 10:30:52 -06:00
MODULE_DESCRIPTION ( " Tegra20 I2S ASoC driver " ) ;
2011-01-07 22:36:14 -07:00
MODULE_LICENSE ( " GPL " ) ;
2011-02-10 15:37:19 -07:00
MODULE_ALIAS ( " platform: " DRV_NAME ) ;
2012-04-06 10:30:52 -06:00
MODULE_DEVICE_TABLE ( of , tegra20_i2s_of_match ) ;