2019-04-19 12:21:58 +02:00
// SPDX-License-Identifier: GPL-2.0
//
// ASoC machine driver for Snow boards
2014-04-28 10:14:39 +05:30
2018-03-12 19:49:39 +01:00
# include <linux/clk.h>
2014-04-28 10:14:39 +05:30
# include <linux/module.h>
# include <linux/platform_device.h>
# include <linux/of.h>
# include <linux/of_device.h>
2018-03-12 19:49:39 +01:00
# include <sound/pcm_params.h>
2014-04-28 10:14:39 +05:30
# include <sound/soc.h>
# include "i2s.h"
# define FIN_PLL_RATE 24000000
2019-06-28 10:48:23 +09:00
SND_SOC_DAILINK_DEFS ( links ,
DAILINK_COMP_ARRAY ( COMP_EMPTY ( ) ) ,
DAILINK_COMP_ARRAY ( COMP_EMPTY ( ) ) ,
2019-06-06 13:09:56 +09:00
DAILINK_COMP_ARRAY ( COMP_EMPTY ( ) ) ) ;
2018-03-12 19:49:39 +01:00
struct snow_priv {
struct snd_soc_dai_link dai_link ;
struct clk * clk_i2s_bus ;
} ;
static int snow_card_hw_params ( struct snd_pcm_substream * substream ,
struct snd_pcm_hw_params * params )
{
static const unsigned int pll_rate [ ] = {
73728000U , 67737602U , 49152000U , 45158401U , 32768001U
} ;
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
struct snow_priv * priv = snd_soc_card_get_drvdata ( rtd - > card ) ;
int bfs , psr , rfs , bitwidth ;
unsigned long int rclk ;
long int freq = - EINVAL ;
int ret , i ;
bitwidth = snd_pcm_format_width ( params_format ( params ) ) ;
if ( bitwidth < 0 ) {
dev_err ( rtd - > card - > dev , " Invalid bit-width: %d \n " , bitwidth ) ;
return bitwidth ;
}
if ( bitwidth ! = 16 & & bitwidth ! = 24 ) {
dev_err ( rtd - > card - > dev , " Unsupported bit-width: %d \n " , bitwidth ) ;
return - EINVAL ;
}
bfs = 2 * bitwidth ;
switch ( params_rate ( params ) ) {
case 16000 :
case 22050 :
case 24000 :
case 32000 :
case 44100 :
case 48000 :
case 88200 :
case 96000 :
rfs = 8 * bfs ;
break ;
case 64000 :
rfs = 384 ;
break ;
case 8000 :
case 11025 :
case 12000 :
rfs = 16 * bfs ;
break ;
default :
return - EINVAL ;
}
rclk = params_rate ( params ) * rfs ;
for ( psr = 8 ; psr > 0 ; psr / = 2 ) {
for ( i = 0 ; i < ARRAY_SIZE ( pll_rate ) ; i + + ) {
if ( ( pll_rate [ i ] - rclk * psr ) < = 2 ) {
freq = pll_rate [ i ] ;
break ;
}
}
}
if ( freq < 0 ) {
dev_err ( rtd - > card - > dev , " Unsupported RCLK rate: %lu \n " , rclk ) ;
return - EINVAL ;
}
ret = clk_set_rate ( priv - > clk_i2s_bus , freq ) ;
if ( ret < 0 ) {
dev_err ( rtd - > card - > dev , " I2S bus clock rate set failed \n " ) ;
return ret ;
}
return 0 ;
}
static const struct snd_soc_ops snow_card_ops = {
. hw_params = snow_card_hw_params ,
2014-04-28 10:14:39 +05:30
} ;
static int snow_late_probe ( struct snd_soc_card * card )
{
2015-11-18 02:34:01 -05:00
struct snd_soc_pcm_runtime * rtd ;
struct snd_soc_dai * codec_dai ;
2014-04-28 10:14:39 +05:30
2019-12-10 09:34:08 +09:00
rtd = snd_soc_get_pcm_runtime ( card , & card - > dai_link [ 0 ] ) ;
2014-04-28 10:14:39 +05:30
2018-03-12 19:49:39 +01:00
/* In the multi-codec case codec_dais 0 is MAX98095 and 1 is HDMI. */
if ( rtd - > num_codecs > 1 )
codec_dai = rtd - > codec_dais [ 0 ] ;
else
codec_dai = rtd - > codec_dai ;
2014-04-28 10:14:39 +05:30
2018-03-12 19:49:39 +01:00
/* Set the MCLK rate for the codec */
return snd_soc_dai_set_sysclk ( codec_dai , 0 ,
FIN_PLL_RATE , SND_SOC_CLOCK_IN ) ;
2014-04-28 10:14:39 +05:30
}
static struct snd_soc_card snow_snd = {
. name = " Snow-I2S " ,
2015-08-21 20:59:21 +08:00
. owner = THIS_MODULE ,
2014-04-28 10:14:39 +05:30
. late_probe = snow_late_probe ,
} ;
static int snow_probe ( struct platform_device * pdev )
{
2018-03-12 19:49:39 +01:00
struct device * dev = & pdev - > dev ;
2014-04-28 10:14:39 +05:30
struct snd_soc_card * card = & snow_snd ;
2018-03-12 19:49:39 +01:00
struct device_node * cpu , * codec ;
struct snd_soc_dai_link * link ;
struct snow_priv * priv ;
int ret ;
2014-04-28 10:14:39 +05:30
2018-03-12 19:49:39 +01:00
priv = devm_kzalloc ( dev , sizeof ( * priv ) , GFP_KERNEL ) ;
if ( ! priv )
return - ENOMEM ;
link = & priv - > dai_link ;
link - > dai_fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF |
SND_SOC_DAIFMT_CBS_CFS ;
link - > name = " Primary " ;
link - > stream_name = link - > name ;
2019-06-06 13:09:56 +09:00
link - > cpus = links_cpus ;
link - > num_cpus = ARRAY_SIZE ( links_cpus ) ;
link - > codecs = links_codecs ;
link - > num_codecs = ARRAY_SIZE ( links_codecs ) ;
2019-06-28 10:48:23 +09:00
link - > platforms = links_platforms ;
link - > num_platforms = ARRAY_SIZE ( links_platforms ) ;
2019-06-06 13:09:56 +09:00
2018-03-12 19:49:39 +01:00
card - > dai_link = link ;
card - > num_links = 1 ;
card - > dev = dev ;
/* Try new DT bindings with HDMI support first. */
cpu = of_get_child_by_name ( dev - > of_node , " cpu " ) ;
if ( cpu ) {
link - > ops = & snow_card_ops ;
2014-04-28 10:14:39 +05:30
2019-06-06 13:09:56 +09:00
link - > cpus - > of_node = of_parse_phandle ( cpu , " sound-dai " , 0 ) ;
2018-03-12 19:49:39 +01:00
of_node_put ( cpu ) ;
2019-06-06 13:09:56 +09:00
if ( ! link - > cpus - > of_node ) {
2018-03-12 19:49:39 +01:00
dev_err ( dev , " Failed parsing cpu/sound-dai property \n " ) ;
return - EINVAL ;
}
codec = of_get_child_by_name ( dev - > of_node , " codec " ) ;
ret = snd_soc_of_get_dai_link_codecs ( dev , codec , link ) ;
of_node_put ( codec ) ;
if ( ret < 0 ) {
2019-06-06 13:09:56 +09:00
of_node_put ( link - > cpus - > of_node ) ;
2018-03-12 19:49:39 +01:00
dev_err ( dev , " Failed parsing codec node \n " ) ;
return ret ;
}
2019-06-06 13:09:56 +09:00
priv - > clk_i2s_bus = of_clk_get_by_name ( link - > cpus - > of_node ,
2018-03-12 19:49:39 +01:00
" i2s_opclk0 " ) ;
if ( IS_ERR ( priv - > clk_i2s_bus ) ) {
snd_soc_of_put_dai_link_codecs ( link ) ;
2019-06-06 13:09:56 +09:00
of_node_put ( link - > cpus - > of_node ) ;
2018-03-12 19:49:39 +01:00
return PTR_ERR ( priv - > clk_i2s_bus ) ;
}
} else {
2019-06-06 13:09:56 +09:00
link - > codecs - > dai_name = " HiFi " ,
2018-03-12 19:49:39 +01:00
2019-06-06 13:09:56 +09:00
link - > cpus - > of_node = of_parse_phandle ( dev - > of_node ,
2018-03-12 19:49:39 +01:00
" samsung,i2s-controller " , 0 ) ;
2019-06-06 13:09:56 +09:00
if ( ! link - > cpus - > of_node ) {
2018-03-12 19:49:39 +01:00
dev_err ( dev , " i2s-controller property parse error \n " ) ;
return - EINVAL ;
}
2019-06-06 13:09:56 +09:00
link - > codecs - > of_node = of_parse_phandle ( dev - > of_node ,
2018-03-12 19:49:39 +01:00
" samsung,audio-codec " , 0 ) ;
2019-06-06 13:09:56 +09:00
if ( ! link - > codecs - > of_node ) {
of_node_put ( link - > cpus - > of_node ) ;
2018-03-12 19:49:39 +01:00
dev_err ( dev , " audio-codec property parse error \n " ) ;
return - EINVAL ;
}
2014-04-28 10:14:39 +05:30
}
2019-06-28 10:48:23 +09:00
link - > platforms - > of_node = link - > cpus - > of_node ;
2014-07-04 14:22:59 +05:30
/* Update card-name if provided through DT, else use default name */
snd_soc_of_parse_card_name ( card , " samsung,model " ) ;
2018-03-12 19:49:39 +01:00
snd_soc_card_set_drvdata ( card , priv ) ;
ret = devm_snd_soc_register_card ( dev , card ) ;
2014-04-28 10:14:39 +05:30
if ( ret ) {
dev_err ( & pdev - > dev , " snd_soc_register_card failed (%d) \n " , ret ) ;
return ret ;
}
return ret ;
}
2018-03-12 19:49:39 +01:00
static int snow_remove ( struct platform_device * pdev )
{
struct snow_priv * priv = platform_get_drvdata ( pdev ) ;
struct snd_soc_dai_link * link = & priv - > dai_link ;
2019-06-06 13:09:56 +09:00
of_node_put ( link - > cpus - > of_node ) ;
of_node_put ( link - > codecs - > of_node ) ;
2018-03-12 19:49:39 +01:00
snd_soc_of_put_dai_link_codecs ( link ) ;
clk_put ( priv - > clk_i2s_bus ) ;
return 0 ;
}
2014-04-28 10:14:39 +05:30
static const struct of_device_id snow_of_match [ ] = {
{ . compatible = " google,snow-audio-max98090 " , } ,
2014-06-20 13:33:16 +05:30
{ . compatible = " google,snow-audio-max98091 " , } ,
2014-04-28 10:14:39 +05:30
{ . compatible = " google,snow-audio-max98095 " , } ,
{ } ,
} ;
2014-11-05 17:44:52 +01:00
MODULE_DEVICE_TABLE ( of , snow_of_match ) ;
2014-04-28 10:14:39 +05:30
static struct platform_driver snow_driver = {
. driver = {
. name = " snow-audio " ,
2014-05-14 08:49:06 +05:30
. pm = & snd_soc_pm_ops ,
2014-04-28 10:14:39 +05:30
. of_match_table = snow_of_match ,
} ,
. probe = snow_probe ,
2018-03-12 19:49:39 +01:00
. remove = snow_remove ,
2014-04-28 10:14:39 +05:30
} ;
module_platform_driver ( snow_driver ) ;
MODULE_DESCRIPTION ( " ALSA SoC Audio machine driver for Snow " ) ;
MODULE_LICENSE ( " GPL " ) ;