2019-05-29 16:57:50 -07:00
// SPDX-License-Identifier: GPL-2.0-only
2011-07-05 10:55:27 -06:00
/*
2012-04-05 15:54:53 -06:00
* tegra20_spdif . c - Tegra20 SPDIF driver
2011-07-05 10:55:27 -06:00
*
* Author : Stephen Warren < swarren @ nvidia . com >
2012-03-20 14:55:49 -06:00
* Copyright ( C ) 2011 - 2012 - NVIDIA , Inc .
2011-07-05 10:55:27 -06:00
*/
# include <linux/clk.h>
# include <linux/device.h>
2012-04-06 11:12:25 -06:00
# include <linux/io.h>
# include <linux/module.h>
2011-07-05 10:55:27 -06:00
# include <linux/platform_device.h>
2012-04-09 09:52:22 -06:00
# include <linux/pm_runtime.h>
2012-04-13 12:14:07 -06:00
# include <linux/regmap.h>
2011-07-05 10:55:27 -06: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-07-05 10:55:27 -06:00
2012-04-05 15:54:53 -06:00
# include "tegra20_spdif.h"
2011-07-05 10:55:27 -06:00
2012-04-06 10:30:52 -06:00
# define DRV_NAME "tegra20-spdif"
2011-07-05 10:55:27 -06:00
2012-04-09 09:52:22 -06:00
static int tegra20_spdif_runtime_suspend ( struct device * dev )
{
struct tegra20_spdif * spdif = dev_get_drvdata ( dev ) ;
2012-06-05 09:59:42 +05:30
clk_disable_unprepare ( spdif - > clk_spdif_out ) ;
2012-04-09 09:52:22 -06:00
return 0 ;
}
static int tegra20_spdif_runtime_resume ( struct device * dev )
{
struct tegra20_spdif * spdif = dev_get_drvdata ( dev ) ;
int ret ;
2012-06-05 09:59:42 +05:30
ret = clk_prepare_enable ( spdif - > clk_spdif_out ) ;
2012-04-09 09:52:22 -06:00
if ( ret ) {
dev_err ( dev , " clk_enable failed: %d \n " , ret ) ;
return ret ;
}
return 0 ;
}
2012-04-06 10:30:52 -06:00
static int tegra20_spdif_hw_params ( struct snd_pcm_substream * substream ,
2011-07-05 10:55:27 -06:00
struct snd_pcm_hw_params * params ,
struct snd_soc_dai * dai )
{
2012-06-06 17:15:05 -06:00
struct device * dev = dai - > dev ;
2012-04-06 10:30:52 -06:00
struct tegra20_spdif * spdif = snd_soc_dai_get_drvdata ( dai ) ;
2013-12-06 13:34:50 -07:00
unsigned int mask = 0 , val = 0 ;
2011-10-02 21:07:02 +08:00
int ret , spdifclock ;
2011-07-05 10:55:27 -06:00
2013-12-06 13:34:50 -07:00
mask | = TEGRA20_SPDIF_CTRL_PACK |
TEGRA20_SPDIF_CTRL_BIT_MODE_MASK ;
2011-07-05 10:55:27 -06:00
switch ( params_format ( params ) ) {
case SNDRV_PCM_FORMAT_S16_LE :
2013-12-06 13:34:50 -07:00
val | = TEGRA20_SPDIF_CTRL_PACK |
TEGRA20_SPDIF_CTRL_BIT_MODE_16BIT ;
2011-07-05 10:55:27 -06:00
break ;
default :
return - EINVAL ;
}
2012-06-06 17:15:06 -06:00
regmap_update_bits ( spdif - > regmap , TEGRA20_SPDIF_CTRL , mask , val ) ;
2011-07-05 10:55:27 -06:00
switch ( params_rate ( params ) ) {
case 32000 :
spdifclock = 4096000 ;
break ;
case 44100 :
spdifclock = 5644800 ;
break ;
case 48000 :
spdifclock = 6144000 ;
break ;
case 88200 :
spdifclock = 11289600 ;
break ;
case 96000 :
spdifclock = 12288000 ;
break ;
case 176400 :
spdifclock = 22579200 ;
break ;
case 192000 :
spdifclock = 24576000 ;
break ;
default :
return - EINVAL ;
}
ret = clk_set_rate ( spdif - > clk_spdif_out , spdifclock ) ;
if ( ret ) {
dev_err ( dev , " Can't set SPDIF clock rate: %d \n " , ret ) ;
return ret ;
}
return 0 ;
}
2012-04-06 10:30:52 -06:00
static void tegra20_spdif_start_playback ( struct tegra20_spdif * spdif )
2011-07-05 10:55:27 -06:00
{
2012-06-06 17:15:06 -06:00
regmap_update_bits ( spdif - > regmap , TEGRA20_SPDIF_CTRL ,
TEGRA20_SPDIF_CTRL_TX_EN ,
TEGRA20_SPDIF_CTRL_TX_EN ) ;
2011-07-05 10:55:27 -06:00
}
2012-04-06 10:30:52 -06:00
static void tegra20_spdif_stop_playback ( struct tegra20_spdif * spdif )
2011-07-05 10:55:27 -06:00
{
2012-06-06 17:15:06 -06:00
regmap_update_bits ( spdif - > regmap , TEGRA20_SPDIF_CTRL ,
TEGRA20_SPDIF_CTRL_TX_EN , 0 ) ;
2011-07-05 10:55:27 -06:00
}
2012-04-06 10:30:52 -06:00
static int tegra20_spdif_trigger ( struct snd_pcm_substream * substream , int cmd ,
2011-07-05 10:55:27 -06:00
struct snd_soc_dai * dai )
{
2012-04-06 10:30:52 -06:00
struct tegra20_spdif * spdif = snd_soc_dai_get_drvdata ( dai ) ;
2011-07-05 10:55:27 -06:00
switch ( cmd ) {
case SNDRV_PCM_TRIGGER_START :
case SNDRV_PCM_TRIGGER_PAUSE_RELEASE :
case SNDRV_PCM_TRIGGER_RESUME :
2012-04-06 10:30:52 -06:00
tegra20_spdif_start_playback ( spdif ) ;
2011-07-05 10:55:27 -06:00
break ;
case SNDRV_PCM_TRIGGER_STOP :
case SNDRV_PCM_TRIGGER_PAUSE_PUSH :
case SNDRV_PCM_TRIGGER_SUSPEND :
2012-04-06 10:30:52 -06:00
tegra20_spdif_stop_playback ( spdif ) ;
2011-07-05 10:55:27 -06:00
break ;
default :
return - EINVAL ;
}
return 0 ;
}
2012-04-06 10:30:52 -06:00
static int tegra20_spdif_probe ( struct snd_soc_dai * dai )
2011-07-05 10:55:27 -06:00
{
2012-04-06 10:30:52 -06:00
struct tegra20_spdif * spdif = snd_soc_dai_get_drvdata ( dai ) ;
2011-07-05 10:55:27 -06:00
dai - > capture_dma_data = NULL ;
dai - > playback_dma_data = & spdif - > playback_dma_data ;
return 0 ;
}
2012-04-06 10:30:52 -06:00
static const struct snd_soc_dai_ops tegra20_spdif_dai_ops = {
. hw_params = tegra20_spdif_hw_params ,
. trigger = tegra20_spdif_trigger ,
2011-07-05 10:55:27 -06:00
} ;
2012-04-06 10:30:52 -06:00
static struct snd_soc_dai_driver tegra20_spdif_dai = {
2011-07-05 10:55:27 -06:00
. name = DRV_NAME ,
2012-04-06 10:30:52 -06:00
. probe = tegra20_spdif_probe ,
2011-07-05 10:55:27 -06:00
. playback = {
2012-06-06 17:15:07 -06:00
. stream_name = " Playback " ,
2011-07-05 10:55:27 -06:00
. channels_min = 2 ,
. channels_max = 2 ,
. rates = SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 |
SNDRV_PCM_RATE_48000 ,
. formats = SNDRV_PCM_FMTBIT_S16_LE ,
} ,
2012-04-06 10:30:52 -06:00
. ops = & tegra20_spdif_dai_ops ,
2011-07-05 10:55:27 -06:00
} ;
2013-03-21 03:37:33 -07:00
static const struct snd_soc_component_driver tegra20_spdif_component = {
. name = DRV_NAME ,
} ;
2012-04-13 12:14:07 -06:00
static bool tegra20_spdif_wr_rd_reg ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case TEGRA20_SPDIF_CTRL :
case TEGRA20_SPDIF_STATUS :
case TEGRA20_SPDIF_STROBE_CTRL :
case TEGRA20_SPDIF_DATA_FIFO_CSR :
case TEGRA20_SPDIF_DATA_OUT :
case TEGRA20_SPDIF_DATA_IN :
case TEGRA20_SPDIF_CH_STA_RX_A :
case TEGRA20_SPDIF_CH_STA_RX_B :
case TEGRA20_SPDIF_CH_STA_RX_C :
case TEGRA20_SPDIF_CH_STA_RX_D :
case TEGRA20_SPDIF_CH_STA_RX_E :
case TEGRA20_SPDIF_CH_STA_RX_F :
case TEGRA20_SPDIF_CH_STA_TX_A :
case TEGRA20_SPDIF_CH_STA_TX_B :
case TEGRA20_SPDIF_CH_STA_TX_C :
case TEGRA20_SPDIF_CH_STA_TX_D :
case TEGRA20_SPDIF_CH_STA_TX_E :
case TEGRA20_SPDIF_CH_STA_TX_F :
case TEGRA20_SPDIF_USR_STA_RX_A :
case TEGRA20_SPDIF_USR_DAT_TX_A :
return true ;
default :
return false ;
2013-10-08 15:55:45 -07:00
}
2012-04-13 12:14:07 -06:00
}
static bool tegra20_spdif_volatile_reg ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case TEGRA20_SPDIF_STATUS :
case TEGRA20_SPDIF_DATA_FIFO_CSR :
case TEGRA20_SPDIF_DATA_OUT :
case TEGRA20_SPDIF_DATA_IN :
case TEGRA20_SPDIF_CH_STA_RX_A :
case TEGRA20_SPDIF_CH_STA_RX_B :
case TEGRA20_SPDIF_CH_STA_RX_C :
case TEGRA20_SPDIF_CH_STA_RX_D :
case TEGRA20_SPDIF_CH_STA_RX_E :
case TEGRA20_SPDIF_CH_STA_RX_F :
case TEGRA20_SPDIF_USR_STA_RX_A :
case TEGRA20_SPDIF_USR_DAT_TX_A :
return true ;
default :
return false ;
2013-10-08 15:55:45 -07:00
}
2012-04-13 12:14:07 -06:00
}
static bool tegra20_spdif_precious_reg ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case TEGRA20_SPDIF_DATA_OUT :
case TEGRA20_SPDIF_DATA_IN :
case TEGRA20_SPDIF_USR_STA_RX_A :
case TEGRA20_SPDIF_USR_DAT_TX_A :
return true ;
default :
return false ;
2013-10-08 15:55:45 -07:00
}
2012-04-13 12:14:07 -06:00
}
static const struct regmap_config tegra20_spdif_regmap_config = {
. reg_bits = 32 ,
. reg_stride = 4 ,
. val_bits = 32 ,
. max_register = TEGRA20_SPDIF_USR_DAT_TX_A ,
. writeable_reg = tegra20_spdif_wr_rd_reg ,
. readable_reg = tegra20_spdif_wr_rd_reg ,
. volatile_reg = tegra20_spdif_volatile_reg ,
. precious_reg = tegra20_spdif_precious_reg ,
2014-03-17 22:08:49 -07:00
. cache_type = REGCACHE_FLAT ,
2012-04-13 12:14:07 -06:00
} ;
2012-12-07 09:26:33 -05:00
static int tegra20_spdif_platform_probe ( struct platform_device * pdev )
2011-07-05 10:55:27 -06:00
{
2012-04-06 10:30:52 -06:00
struct tegra20_spdif * spdif ;
2015-08-23 23:32:14 +08:00
struct resource * mem , * dmareq ;
2012-04-13 12:14:07 -06:00
void __iomem * regs ;
2011-07-05 10:55:27 -06:00
int ret ;
2012-04-06 11:14:04 -06:00
spdif = devm_kzalloc ( & pdev - > dev , sizeof ( struct tegra20_spdif ) ,
GFP_KERNEL ) ;
2017-02-25 13:18:08 +02:00
if ( ! spdif )
2015-08-13 12:59:21 +05:30
return - ENOMEM ;
2017-02-25 13:18:08 +02:00
2011-07-05 10:55:27 -06:00
dev_set_drvdata ( & pdev - > dev , spdif ) ;
2015-08-13 12:59:21 +05:30
spdif - > clk_spdif_out = devm_clk_get ( & pdev - > dev , " spdif_out " ) ;
2011-07-05 10:55:27 -06:00
if ( IS_ERR ( spdif - > clk_spdif_out ) ) {
pr_err ( " Can't retrieve spdif clock \n " ) ;
ret = PTR_ERR ( spdif - > clk_spdif_out ) ;
2015-08-13 12:59:21 +05:30
return ret ;
2011-07-05 10:55:27 -06:00
}
mem = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
2015-08-23 23:32:14 +08:00
regs = devm_ioremap_resource ( & pdev - > dev , mem ) ;
if ( IS_ERR ( regs ) )
return PTR_ERR ( regs ) ;
2011-07-05 10:55:27 -06:00
dmareq = platform_get_resource ( pdev , IORESOURCE_DMA , 0 ) ;
if ( ! dmareq ) {
dev_err ( & pdev - > dev , " No DMA resource \n " ) ;
2015-08-13 12:59:21 +05:30
return - ENODEV ;
2011-07-05 10:55:27 -06:00
}
2012-04-13 12:14:07 -06:00
spdif - > regmap = devm_regmap_init_mmio ( & pdev - > dev , regs ,
& tegra20_spdif_regmap_config ) ;
if ( IS_ERR ( spdif - > regmap ) ) {
dev_err ( & pdev - > dev , " regmap init failed \n " ) ;
ret = PTR_ERR ( spdif - > regmap ) ;
2015-08-13 12:59:21 +05:30
return ret ;
2012-04-13 12:14:07 -06:00
}
2012-04-06 10:30:52 -06:00
spdif - > playback_dma_data . addr = mem - > start + TEGRA20_SPDIF_DATA_OUT ;
2013-07-21 10:34:09 +08:00
spdif - > playback_dma_data . addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES ;
spdif - > playback_dma_data . maxburst = 4 ;
2013-04-03 11:06:03 +02:00
spdif - > playback_dma_data . slave_id = dmareq - > start ;
2011-07-05 10:55:27 -06:00
2012-04-09 09:52:22 -06:00
pm_runtime_enable ( & pdev - > dev ) ;
if ( ! pm_runtime_enabled ( & pdev - > dev ) ) {
ret = tegra20_spdif_runtime_resume ( & pdev - > dev ) ;
if ( ret )
goto err_pm_disable ;
}
2015-08-18 17:26:45 +05:30
ret = snd_soc_register_component ( & pdev - > dev , & tegra20_spdif_component ,
& tegra20_spdif_dai , 1 ) ;
2011-07-05 10:55:27 -06:00
if ( ret ) {
dev_err ( & pdev - > dev , " Could not register DAI: %d \n " , ret ) ;
ret = - ENOMEM ;
2012-04-09 09:52:22 -06:00
goto err_suspend ;
2011-07-05 10:55:27 -06: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 ) ;
2015-08-18 17:26:45 +05:30
goto err_unregister_component ;
2012-03-20 14:55:49 -06:00
}
2011-07-05 10:55:27 -06:00
return 0 ;
2015-08-18 17:26:45 +05:30
err_unregister_component :
snd_soc_unregister_component ( & pdev - > dev ) ;
2012-04-09 09:52:22 -06:00
err_suspend :
if ( ! pm_runtime_status_suspended ( & pdev - > dev ) )
tegra20_spdif_runtime_suspend ( & pdev - > dev ) ;
err_pm_disable :
pm_runtime_disable ( & pdev - > dev ) ;
2015-08-13 12:59:21 +05:30
2011-07-05 10:55:27 -06:00
return ret ;
}
2012-12-07 09:26:33 -05:00
static int tegra20_spdif_platform_remove ( struct platform_device * pdev )
2011-07-05 10:55:27 -06:00
{
2012-04-09 09:52:22 -06:00
pm_runtime_disable ( & pdev - > dev ) ;
if ( ! pm_runtime_status_suspended ( & pdev - > dev ) )
tegra20_spdif_runtime_suspend ( & pdev - > dev ) ;
2012-03-20 14:55:49 -06:00
tegra_pcm_platform_unregister ( & pdev - > dev ) ;
2015-08-18 17:26:45 +05:30
snd_soc_unregister_component ( & pdev - > dev ) ;
2011-07-05 10:55:27 -06:00
return 0 ;
}
2012-11-19 13:25:33 -05:00
static const struct dev_pm_ops tegra20_spdif_pm_ops = {
2012-04-09 09:52:22 -06:00
SET_RUNTIME_PM_OPS ( tegra20_spdif_runtime_suspend ,
tegra20_spdif_runtime_resume , NULL )
} ;
2012-04-06 10:30:52 -06:00
static struct platform_driver tegra20_spdif_driver = {
2011-07-05 10:55:27 -06:00
. driver = {
. name = DRV_NAME ,
2012-04-09 09:52:22 -06:00
. pm = & tegra20_spdif_pm_ops ,
2011-07-05 10:55:27 -06:00
} ,
2012-04-06 10:30:52 -06:00
. probe = tegra20_spdif_platform_probe ,
2012-12-07 09:26:33 -05:00
. remove = tegra20_spdif_platform_remove ,
2011-07-05 10:55:27 -06:00
} ;
2012-04-06 10:30:52 -06:00
module_platform_driver ( tegra20_spdif_driver ) ;
2011-07-05 10:55:27 -06:00
MODULE_AUTHOR ( " Stephen Warren <swarren@nvidia.com> " ) ;
2012-04-06 10:30:52 -06:00
MODULE_DESCRIPTION ( " Tegra20 SPDIF ASoC driver " ) ;
2011-07-05 10:55:27 -06:00
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS ( " platform: " DRV_NAME ) ;