2018-07-02 06:31:16 +00:00
// SPDX-License-Identifier: GPL-2.0
//
// ASoC audio graph sound card support
//
// Copyright (C) 2016 Renesas Solutions Corp.
// Kuninori Morimoto <kuninori.morimoto.gx@renesas.com>
//
// based on ${LINUX}/sound/soc/generic/simple-card.c
2017-04-20 01:36:08 +00:00
# include <linux/clk.h>
# include <linux/device.h>
# include <linux/gpio.h>
2017-06-29 21:26:38 +08:00
# include <linux/gpio/consumer.h>
2017-04-20 01:36:08 +00:00
# include <linux/module.h>
# include <linux/of.h>
# include <linux/of_device.h>
# include <linux/of_gpio.h>
# include <linux/of_graph.h>
# include <linux/platform_device.h>
# include <linux/string.h>
# include <sound/simple_card_utils.h>
struct graph_card_data {
struct snd_soc_card snd_card ;
struct graph_dai_props {
2018-11-30 02:07:48 +00:00
struct asoc_simple_dai * cpu_dai ;
struct asoc_simple_dai * codec_dai ;
2018-08-31 03:09:05 +00:00
struct snd_soc_dai_link_component codecs ; /* single codec */
2018-08-31 03:10:58 +00:00
struct snd_soc_dai_link_component platform ;
2018-12-14 11:32:25 +09:00
struct asoc_simple_card_data adata ;
struct snd_soc_codec_conf * codec_conf ;
2017-11-09 15:07:58 +01:00
unsigned int mclk_fs ;
2017-04-20 01:36:08 +00:00
} * dai_props ;
2018-06-11 17:32:14 +09:00
struct asoc_simple_jack hp_jack ;
struct asoc_simple_jack mic_jack ;
2017-04-20 01:36:08 +00:00
struct snd_soc_dai_link * dai_link ;
2018-11-30 02:07:48 +00:00
struct asoc_simple_dai * dais ;
2018-12-14 11:32:25 +09:00
struct snd_soc_codec_conf * codec_conf ;
2017-06-29 21:26:38 +08:00
struct gpio_desc * pa_gpio ;
} ;
2018-12-20 10:45:59 +09:00
struct link_info {
int dais ; /* number of dai */
int link ; /* number of link */
int conf ; /* number of codec_conf */
int cpu ; /* turn for CPU / Codec */
} ;
2018-11-22 00:57:40 +00:00
# define graph_priv_to_card(priv) (&(priv)->snd_card)
# define graph_priv_to_props(priv, i) ((priv)->dai_props + (i))
# define graph_priv_to_dev(priv) (graph_priv_to_card(priv)->dev)
# define graph_priv_to_link(priv, i) (graph_priv_to_card(priv)->dai_link + (i))
2018-12-14 11:32:25 +09:00
# define PREFIX "audio-graph-card,"
2017-06-29 21:26:38 +08:00
static int asoc_graph_card_outdrv_event ( struct snd_soc_dapm_widget * w ,
struct snd_kcontrol * kcontrol ,
int event )
{
struct snd_soc_dapm_context * dapm = w - > dapm ;
struct graph_card_data * priv = snd_soc_card_get_drvdata ( dapm - > card ) ;
switch ( event ) {
case SND_SOC_DAPM_POST_PMU :
gpiod_set_value_cansleep ( priv - > pa_gpio , 1 ) ;
break ;
case SND_SOC_DAPM_PRE_PMD :
gpiod_set_value_cansleep ( priv - > pa_gpio , 0 ) ;
break ;
default :
return - EINVAL ;
}
return 0 ;
}
static const struct snd_soc_dapm_widget asoc_graph_card_dapm_widgets [ ] = {
SND_SOC_DAPM_OUT_DRV_E ( " Amplifier " , SND_SOC_NOPM ,
0 , 0 , NULL , 0 , asoc_graph_card_outdrv_event ,
SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD ) ,
2017-04-20 01:36:08 +00:00
} ;
static int asoc_graph_card_startup ( struct snd_pcm_substream * substream )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
struct graph_card_data * priv = snd_soc_card_get_drvdata ( rtd - > card ) ;
struct graph_dai_props * dai_props = graph_priv_to_props ( priv , rtd - > num ) ;
int ret ;
2018-11-30 02:07:48 +00:00
ret = asoc_simple_card_clk_enable ( dai_props - > cpu_dai ) ;
2017-04-20 01:36:08 +00:00
if ( ret )
return ret ;
2018-11-30 02:07:48 +00:00
ret = asoc_simple_card_clk_enable ( dai_props - > codec_dai ) ;
2017-04-20 01:36:08 +00:00
if ( ret )
2018-11-30 02:07:48 +00:00
asoc_simple_card_clk_disable ( dai_props - > cpu_dai ) ;
2017-04-20 01:36:08 +00:00
return ret ;
}
static void asoc_graph_card_shutdown ( struct snd_pcm_substream * substream )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
struct graph_card_data * priv = snd_soc_card_get_drvdata ( rtd - > card ) ;
struct graph_dai_props * dai_props = graph_priv_to_props ( priv , rtd - > num ) ;
2018-11-30 02:07:48 +00:00
asoc_simple_card_clk_disable ( dai_props - > cpu_dai ) ;
2017-04-20 01:36:08 +00:00
2018-11-30 02:07:48 +00:00
asoc_simple_card_clk_disable ( dai_props - > codec_dai ) ;
2017-04-20 01:36:08 +00:00
}
2017-11-09 15:07:58 +01:00
static int asoc_graph_card_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 ;
struct snd_soc_dai * cpu_dai = rtd - > cpu_dai ;
struct graph_card_data * priv = snd_soc_card_get_drvdata ( rtd - > card ) ;
struct graph_dai_props * dai_props = graph_priv_to_props ( priv , rtd - > num ) ;
unsigned int mclk , mclk_fs = 0 ;
int ret = 0 ;
2018-12-14 11:32:30 +09:00
if ( dai_props - > mclk_fs )
2017-11-09 15:07:58 +01:00
mclk_fs = dai_props - > mclk_fs ;
if ( mclk_fs ) {
mclk = params_rate ( params ) * mclk_fs ;
ret = snd_soc_dai_set_sysclk ( codec_dai , 0 , mclk ,
SND_SOC_CLOCK_IN ) ;
if ( ret & & ret ! = - ENOTSUPP )
goto err ;
ret = snd_soc_dai_set_sysclk ( cpu_dai , 0 , mclk ,
SND_SOC_CLOCK_OUT ) ;
if ( ret & & ret ! = - ENOTSUPP )
goto err ;
}
return 0 ;
err :
return ret ;
}
2017-08-16 22:29:26 +05:30
static const struct snd_soc_ops asoc_graph_card_ops = {
2017-04-20 01:36:08 +00:00
. startup = asoc_graph_card_startup ,
. shutdown = asoc_graph_card_shutdown ,
2017-11-09 15:07:58 +01:00
. hw_params = asoc_graph_card_hw_params ,
2017-04-20 01:36:08 +00:00
} ;
static int asoc_graph_card_dai_init ( struct snd_soc_pcm_runtime * rtd )
{
struct graph_card_data * priv = snd_soc_card_get_drvdata ( rtd - > card ) ;
2018-11-30 02:07:48 +00:00
struct graph_dai_props * dai_props = graph_priv_to_props ( priv , rtd - > num ) ;
int ret = 0 ;
2017-04-20 01:36:08 +00:00
2018-11-30 02:07:48 +00:00
ret = asoc_simple_card_init_dai ( rtd - > codec_dai ,
dai_props - > codec_dai ) ;
2017-04-20 01:36:08 +00:00
if ( ret < 0 )
return ret ;
2018-11-30 02:07:48 +00:00
ret = asoc_simple_card_init_dai ( rtd - > cpu_dai ,
dai_props - > cpu_dai ) ;
2017-04-20 01:36:08 +00:00
if ( ret < 0 )
return ret ;
return 0 ;
}
2018-12-14 11:32:25 +09:00
static int asoc_graph_card_be_hw_params_fixup ( struct snd_soc_pcm_runtime * rtd ,
struct snd_pcm_hw_params * params )
{
struct graph_card_data * priv = snd_soc_card_get_drvdata ( rtd - > card ) ;
struct graph_dai_props * dai_props = graph_priv_to_props ( priv , rtd - > num ) ;
asoc_simple_card_convert_fixup ( & dai_props - > adata , params ) ;
return 0 ;
}
2018-12-20 10:45:48 +09:00
static void asoc_graph_card_get_conversion ( struct device * dev ,
struct device_node * ep ,
struct asoc_simple_card_data * adata )
{
struct device_node * top = dev - > of_node ;
struct device_node * port = of_get_parent ( ep ) ;
struct device_node * ports = of_get_parent ( port ) ;
struct device_node * node = of_graph_get_port_parent ( ep ) ;
asoc_simple_card_parse_convert ( dev , top , NULL , adata ) ;
asoc_simple_card_parse_convert ( dev , node , PREFIX , adata ) ;
asoc_simple_card_parse_convert ( dev , ports , NULL , adata ) ;
asoc_simple_card_parse_convert ( dev , port , NULL , adata ) ;
asoc_simple_card_parse_convert ( dev , ep , NULL , adata ) ;
}
2018-12-20 10:46:05 +09:00
static int asoc_graph_card_dai_link_of_dpcm ( struct graph_card_data * priv ,
2018-12-14 11:32:30 +09:00
struct device_node * cpu_ep ,
2018-12-14 11:32:25 +09:00
struct device_node * codec_ep ,
2018-12-20 10:46:05 +09:00
struct link_info * li ,
int dup_codec )
2018-12-14 11:32:25 +09:00
{
struct device * dev = graph_priv_to_dev ( priv ) ;
2018-12-20 10:45:59 +09:00
struct snd_soc_dai_link * dai_link = graph_priv_to_link ( priv , li - > link ) ;
struct graph_dai_props * dai_props = graph_priv_to_props ( priv , li - > link ) ;
2018-12-20 10:46:05 +09:00
struct device_node * top = dev - > of_node ;
2018-12-20 10:45:59 +09:00
struct device_node * ep = li - > cpu ? cpu_ep : codec_ep ;
2018-12-20 10:46:05 +09:00
struct device_node * port ;
struct device_node * ports ;
struct device_node * node ;
2018-12-14 11:32:25 +09:00
struct asoc_simple_dai * dai ;
2018-12-14 11:32:43 +09:00
struct snd_soc_dai_link_component * codecs = dai_link - > codecs ;
2018-12-14 11:32:25 +09:00
int ret ;
2018-12-20 10:46:05 +09:00
/* Do it all CPU endpoint, and 1st Codec endpoint */
if ( ! li - > cpu & & dup_codec )
return 0 ;
port = of_get_parent ( ep ) ;
ports = of_get_parent ( port ) ;
node = of_graph_get_port_parent ( ep ) ;
2018-12-20 10:45:59 +09:00
li - > link + + ;
dev_dbg ( dev , " link_of DPCM (%pOF) \n " , ep ) ;
2018-12-14 11:32:25 +09:00
2018-12-14 11:32:30 +09:00
of_property_read_u32 ( top , " mclk-fs " , & dai_props - > mclk_fs ) ;
of_property_read_u32 ( ports , " mclk-fs " , & dai_props - > mclk_fs ) ;
of_property_read_u32 ( port , " mclk-fs " , & dai_props - > mclk_fs ) ;
of_property_read_u32 ( ep , " mclk-fs " , & dai_props - > mclk_fs ) ;
2018-12-14 11:32:36 +09:00
2018-12-20 10:45:48 +09:00
asoc_graph_card_get_conversion ( dev , ep , & dai_props - > adata ) ;
2018-12-14 11:32:36 +09:00
2018-12-14 11:32:30 +09:00
of_node_put ( ports ) ;
of_node_put ( port ) ;
2018-12-20 10:46:05 +09:00
of_node_put ( node ) ;
2018-12-14 11:32:30 +09:00
2018-12-20 10:45:59 +09:00
if ( li - > cpu ) {
2018-12-14 11:32:25 +09:00
/* BE is dummy */
codecs - > of_node = NULL ;
codecs - > dai_name = " snd-soc-dummy-dai " ;
codecs - > name = " snd-soc-dummy " ;
/* FE settings */
dai_link - > dynamic = 1 ;
dai_link - > dpcm_merged_format = 1 ;
dai =
2018-12-20 10:45:59 +09:00
dai_props - > cpu_dai = & priv - > dais [ li - > dais + + ] ;
2018-12-14 11:32:25 +09:00
ret = asoc_simple_card_parse_graph_cpu ( ep , dai_link ) ;
if ( ret )
return ret ;
ret = asoc_simple_card_parse_clk_cpu ( dev , ep , dai_link , dai ) ;
if ( ret < 0 )
return ret ;
ret = asoc_simple_card_set_dailink_name ( dev , dai_link ,
" fe.%s " ,
dai_link - > cpu_dai_name ) ;
if ( ret < 0 )
return ret ;
/* card->num_links includes Codec */
asoc_simple_card_canonicalize_cpu ( dai_link ,
of_graph_get_endpoint_count ( dai_link - > cpu_of_node ) = = 1 ) ;
} else {
struct snd_soc_codec_conf * cconf ;
/* FE is dummy */
dai_link - > cpu_of_node = NULL ;
dai_link - > cpu_dai_name = " snd-soc-dummy-dai " ;
dai_link - > cpu_name = " snd-soc-dummy " ;
/* BE settings */
dai_link - > no_pcm = 1 ;
dai_link - > be_hw_params_fixup = asoc_graph_card_be_hw_params_fixup ;
dai =
2018-12-20 10:45:59 +09:00
dai_props - > codec_dai = & priv - > dais [ li - > dais + + ] ;
2018-12-14 11:32:25 +09:00
cconf =
2018-12-20 10:45:59 +09:00
dai_props - > codec_conf = & priv - > codec_conf [ li - > conf + + ] ;
2018-12-14 11:32:25 +09:00
ret = asoc_simple_card_parse_graph_codec ( ep , dai_link ) ;
if ( ret < 0 )
return ret ;
ret = asoc_simple_card_parse_clk_codec ( dev , ep , dai_link , dai ) ;
if ( ret < 0 )
return ret ;
ret = asoc_simple_card_set_dailink_name ( dev , dai_link ,
" be.%s " ,
2018-12-14 11:32:43 +09:00
codecs - > dai_name ) ;
2018-12-14 11:32:25 +09:00
if ( ret < 0 )
return ret ;
/* check "prefix" from top node */
2018-12-14 11:32:43 +09:00
snd_soc_of_parse_node_prefix ( top , cconf , codecs - > of_node ,
2018-12-14 11:32:25 +09:00
" prefix " ) ;
2018-12-14 11:32:43 +09:00
snd_soc_of_parse_node_prefix ( node , cconf , codecs - > of_node ,
PREFIX " prefix " ) ;
snd_soc_of_parse_node_prefix ( ports , cconf , codecs - > of_node ,
" prefix " ) ;
snd_soc_of_parse_node_prefix ( port , cconf , codecs - > of_node ,
" prefix " ) ;
2018-12-14 11:32:25 +09:00
}
ret = asoc_simple_card_of_parse_tdm ( ep , dai ) ;
if ( ret )
return ret ;
ret = asoc_simple_card_canonicalize_dailink ( dai_link ) ;
if ( ret < 0 )
return ret ;
ret = asoc_simple_card_parse_daifmt ( dev , cpu_ep , codec_ep ,
NULL , & dai_link - > dai_fmt ) ;
if ( ret < 0 )
return ret ;
dai_link - > dpcm_playback = 1 ;
dai_link - > dpcm_capture = 1 ;
dai_link - > ops = & asoc_graph_card_ops ;
dai_link - > init = asoc_graph_card_dai_init ;
return 0 ;
}
2018-12-20 10:46:05 +09:00
static int asoc_graph_card_dai_link_of ( struct graph_card_data * priv ,
2018-12-14 11:32:30 +09:00
struct device_node * cpu_ep ,
2018-12-14 11:32:25 +09:00
struct device_node * codec_ep ,
2018-12-20 10:45:59 +09:00
struct link_info * li )
2017-04-20 01:36:08 +00:00
{
struct device * dev = graph_priv_to_dev ( priv ) ;
2018-12-20 10:45:59 +09:00
struct snd_soc_dai_link * dai_link = graph_priv_to_link ( priv , li - > link ) ;
struct graph_dai_props * dai_props = graph_priv_to_props ( priv , li - > link ) ;
2018-12-20 10:46:05 +09:00
struct device_node * top = dev - > of_node ;
struct device_node * cpu_port ;
struct device_node * codec_port ;
struct device_node * cpu_ports ;
struct device_node * codec_ports ;
2018-11-30 02:07:48 +00:00
struct asoc_simple_dai * cpu_dai ;
struct asoc_simple_dai * codec_dai ;
2017-04-20 01:36:08 +00:00
int ret ;
2018-12-20 10:46:05 +09:00
/* Do it only CPU turn */
if ( ! li - > cpu )
return 0 ;
cpu_port = of_get_parent ( cpu_ep ) ;
cpu_ports = of_get_parent ( cpu_port ) ;
codec_port = of_get_parent ( codec_ep ) ;
codec_ports = of_get_parent ( codec_port ) ;
2018-12-20 10:45:59 +09:00
dev_dbg ( dev , " link_of (%pOF) \n " , cpu_ep ) ;
li - > link + + ;
2018-12-14 11:32:25 +09:00
2018-11-30 02:07:48 +00:00
cpu_dai =
2018-12-20 10:45:59 +09:00
dai_props - > cpu_dai = & priv - > dais [ li - > dais + + ] ;
2018-11-30 02:07:48 +00:00
codec_dai =
2018-12-20 10:45:59 +09:00
dai_props - > codec_dai = & priv - > dais [ li - > dais + + ] ;
2018-11-30 02:07:48 +00:00
2018-12-14 11:32:30 +09:00
/* Factor to mclk, used in hw_params() */
of_property_read_u32 ( top , " mclk-fs " , & dai_props - > mclk_fs ) ;
of_property_read_u32 ( cpu_ports , " mclk-fs " , & dai_props - > mclk_fs ) ;
of_property_read_u32 ( codec_ports , " mclk-fs " , & dai_props - > mclk_fs ) ;
of_property_read_u32 ( cpu_port , " mclk-fs " , & dai_props - > mclk_fs ) ;
of_property_read_u32 ( codec_port , " mclk-fs " , & dai_props - > mclk_fs ) ;
of_property_read_u32 ( cpu_ep , " mclk-fs " , & dai_props - > mclk_fs ) ;
of_property_read_u32 ( codec_ep , " mclk-fs " , & dai_props - > mclk_fs ) ;
of_node_put ( cpu_port ) ;
of_node_put ( cpu_ports ) ;
of_node_put ( codec_port ) ;
of_node_put ( codec_ports ) ;
2017-04-20 01:36:08 +00:00
ret = asoc_simple_card_parse_daifmt ( dev , cpu_ep , codec_ep ,
NULL , & dai_link - > dai_fmt ) ;
if ( ret < 0 )
2018-12-14 11:32:25 +09:00
return ret ;
2017-04-20 01:36:08 +00:00
ret = asoc_simple_card_parse_graph_cpu ( cpu_ep , dai_link ) ;
if ( ret < 0 )
2018-12-14 11:32:25 +09:00
return ret ;
2017-04-20 01:36:08 +00:00
ret = asoc_simple_card_parse_graph_codec ( codec_ep , dai_link ) ;
if ( ret < 0 )
2018-12-14 11:32:25 +09:00
return ret ;
2017-04-20 01:36:08 +00:00
2017-06-14 00:35:30 +00:00
ret = asoc_simple_card_of_parse_tdm ( cpu_ep , cpu_dai ) ;
2017-04-20 01:36:08 +00:00
if ( ret < 0 )
2018-12-14 11:32:25 +09:00
return ret ;
2017-04-20 01:36:08 +00:00
2017-06-14 00:35:30 +00:00
ret = asoc_simple_card_of_parse_tdm ( codec_ep , codec_dai ) ;
2017-04-20 01:36:08 +00:00
if ( ret < 0 )
2018-12-14 11:32:25 +09:00
return ret ;
2017-04-20 01:36:08 +00:00
ret = asoc_simple_card_parse_clk_cpu ( dev , cpu_ep , dai_link , cpu_dai ) ;
if ( ret < 0 )
2018-12-14 11:32:25 +09:00
return ret ;
2017-04-20 01:36:08 +00:00
ret = asoc_simple_card_parse_clk_codec ( dev , codec_ep , dai_link , codec_dai ) ;
if ( ret < 0 )
2018-12-14 11:32:25 +09:00
return ret ;
2017-04-20 01:36:08 +00:00
ret = asoc_simple_card_canonicalize_dailink ( dai_link ) ;
if ( ret < 0 )
2018-12-14 11:32:25 +09:00
return ret ;
2017-04-20 01:36:08 +00:00
ret = asoc_simple_card_set_dailink_name ( dev , dai_link ,
" %s-%s " ,
dai_link - > cpu_dai_name ,
2018-08-31 03:09:05 +00:00
dai_link - > codecs - > dai_name ) ;
2017-04-20 01:36:08 +00:00
if ( ret < 0 )
2018-12-14 11:32:25 +09:00
return ret ;
2017-04-20 01:36:08 +00:00
dai_link - > ops = & asoc_graph_card_ops ;
dai_link - > init = asoc_graph_card_dai_init ;
asoc_simple_card_canonicalize_cpu ( dai_link ,
2017-06-22 06:21:49 +00:00
of_graph_get_endpoint_count ( dai_link - > cpu_of_node ) = = 1 ) ;
2017-04-20 01:36:08 +00:00
2018-12-14 11:32:25 +09:00
return 0 ;
2017-04-20 01:36:08 +00:00
}
2018-12-20 10:46:20 +09:00
static int asoc_graph_card_for_each_link ( struct graph_card_data * priv ,
struct link_info * li ,
int ( * func_noml ) ( struct graph_card_data * priv ,
struct device_node * cpu_ep ,
struct device_node * codec_ep ,
struct link_info * li ) ,
int ( * func_dpcm ) ( struct graph_card_data * priv ,
struct device_node * cpu_ep ,
struct device_node * codec_ep ,
struct link_info * li , int dup_codec ) )
2017-04-20 01:36:08 +00:00
{
struct of_phandle_iterator it ;
struct device * dev = graph_priv_to_dev ( priv ) ;
2018-12-20 10:46:20 +09:00
struct device_node * node = dev - > of_node ;
2018-12-14 11:32:25 +09:00
struct device_node * cpu_port ;
2018-12-20 10:46:20 +09:00
struct device_node * cpu_ep ;
struct device_node * codec_ep ;
struct device_node * codec_port ;
struct device_node * codec_port_old = NULL ;
2018-12-20 10:45:54 +09:00
struct asoc_simple_card_data adata ;
2018-11-30 02:07:48 +00:00
int rc , ret ;
2017-04-20 01:36:08 +00:00
2018-12-20 10:46:20 +09:00
/* loop for all listed CPU port */
of_for_each_phandle ( & it , rc , node , " dais " , NULL , 0 ) {
cpu_port = it . node ;
cpu_ep = NULL ;
/* loop for all CPU endpoint */
while ( 1 ) {
cpu_ep = of_get_next_child ( cpu_port , cpu_ep ) ;
if ( ! cpu_ep )
break ;
/* get codec */
codec_ep = of_graph_get_remote_endpoint ( cpu_ep ) ;
codec_port = of_get_parent ( codec_ep ) ;
of_node_put ( codec_ep ) ;
of_node_put ( codec_port ) ;
/* get convert-xxx property */
memset ( & adata , 0 , sizeof ( adata ) ) ;
asoc_graph_card_get_conversion ( dev , codec_ep , & adata ) ;
asoc_graph_card_get_conversion ( dev , cpu_ep , & adata ) ;
/*
* It is DPCM
* if Codec port has many endpoints ,
* or has convert - xxx property
*/
if ( ( of_get_child_count ( codec_port ) > 1 ) | |
adata . convert_rate | | adata . convert_channels )
ret = func_dpcm ( priv , cpu_ep , codec_ep , li ,
( codec_port_old = = codec_port ) ) ;
/* else normal sound */
else
ret = func_noml ( priv , cpu_ep , codec_ep , li ) ;
if ( ret < 0 )
return ret ;
codec_port_old = codec_port ;
}
}
return 0 ;
}
static int asoc_graph_card_parse_of ( struct graph_card_data * priv )
{
struct snd_soc_card * card = graph_priv_to_card ( priv ) ;
struct link_info li ;
int ret ;
2017-06-29 21:26:38 +08:00
ret = asoc_simple_card_of_parse_widgets ( card , NULL ) ;
if ( ret < 0 )
return ret ;
2018-11-21 02:11:13 +00:00
ret = asoc_simple_card_of_parse_routing ( card , NULL ) ;
2017-06-29 21:26:38 +08:00
if ( ret < 0 )
return ret ;
2018-12-20 10:45:59 +09:00
memset ( & li , 0 , sizeof ( li ) ) ;
for ( li . cpu = 1 ; li . cpu > = 0 ; li . cpu - - ) {
2018-12-14 11:32:25 +09:00
/*
* Detect all CPU first , and Detect all Codec 2 nd .
*
* In Normal sound case , all DAIs are detected
* as " CPU-Codec " .
*
* In DPCM sound case ,
* all CPUs are detected as " CPU-dummy " , and
* all Codecs are detected as " dummy-Codec " .
* To avoid random sub - device numbering ,
* detect " dummy-Codec " in last ;
*/
2018-12-20 10:46:20 +09:00
ret = asoc_graph_card_for_each_link ( priv , & li ,
asoc_graph_card_dai_link_of ,
asoc_graph_card_dai_link_of_dpcm ) ;
if ( ret < 0 )
return ret ;
2017-04-20 01:36:08 +00:00
}
return asoc_simple_card_parse_card_name ( card , NULL ) ;
}
2018-12-20 10:46:05 +09:00
static int asoc_graph_card_count_noml ( struct graph_card_data * priv ,
struct device_node * cpu_ep ,
struct device_node * codec_ep ,
2018-12-20 10:45:59 +09:00
struct link_info * li )
2017-04-20 01:36:08 +00:00
{
2018-12-20 10:46:05 +09:00
struct device * dev = graph_priv_to_dev ( priv ) ;
li - > link + = 1 ; /* 1xCPU-Codec */
li - > dais + = 2 ; /* 1xCPU + 1xCodec */
dev_dbg ( dev , " Count As Normal \n " ) ;
return 0 ;
}
static int asoc_graph_card_count_dpcm ( struct graph_card_data * priv ,
struct device_node * cpu_ep ,
struct device_node * codec_ep ,
struct link_info * li ,
int dup_codec )
{
struct device * dev = graph_priv_to_dev ( priv ) ;
li - > link + + ; /* 1xCPU-dummy */
li - > dais + + ; /* 1xCPU */
if ( ! dup_codec ) {
li - > link + + ; /* 1xdummy-Codec */
li - > conf + + ; /* 1xdummy-Codec */
li - > dais + + ; /* 1xCodec */
}
dev_dbg ( dev , " Count As DPCM \n " ) ;
return 0 ;
}
static void asoc_graph_get_dais_count ( struct graph_card_data * priv ,
struct link_info * li )
{
struct device * dev = graph_priv_to_dev ( priv ) ;
2017-04-20 01:36:08 +00:00
2018-12-14 11:32:25 +09:00
/*
* link_num : number of links .
* CPU - Codec / CPU - dummy / dummy - Codec
* dais_num : number of DAIs
* ccnf_num : number of codec_conf
* same number for " dummy-Codec "
*
* ex1 )
* CPU0 - - - Codec0 link : 5
* CPU1 - - - Codec1 dais : 7
* CPU2 - / ccnf : 1
* CPU3 - - - Codec2
*
* = > 5 links = 2 xCPU - Codec + 2 xCPU - dummy + 1 xdummy - Codec
* = > 7 DAIs = 4 xCPU + 3 xCodec
* = > 1 ccnf = 1 xdummy - Codec
*
* ex2 )
* CPU0 - - - Codec0 link : 5
* CPU1 - - - Codec1 dais : 6
* CPU2 - / ccnf : 1
* CPU3 - /
*
* = > 5 links = 1 xCPU - Codec + 3 xCPU - dummy + 1 xdummy - Codec
* = > 6 DAIs = 4 xCPU + 2 xCodec
* = > 1 ccnf = 1 xdummy - Codec
*
* ex3 )
* CPU0 - - - Codec0 link : 6
* CPU1 - / dais : 6
* CPU2 - - - Codec1 ccnf : 2
* CPU3 - /
*
* = > 6 links = 0xC PU - Codec + 4 xCPU - dummy + 2 xdummy - Codec
* = > 6 DAIs = 4 xCPU + 2 xCodec
* = > 2 ccnf = 2 xdummy - Codec
2018-12-20 10:45:54 +09:00
*
* ex4 )
* CPU0 - - - Codec0 ( convert - rate ) link : 3
* CPU1 - - - Codec1 dais : 4
* ccnf : 1
*
* = > 3 links = 1 xCPU - Codec + 1 xCPU - dummy + 1 xdummy - Codec
* = > 4 DAIs = 2 xCPU + 2 xCodec
* = > 1 ccnf = 1 xdummy - Codec
2018-12-14 11:32:25 +09:00
*/
2018-12-20 10:46:20 +09:00
asoc_graph_card_for_each_link ( priv , li ,
asoc_graph_card_count_noml ,
asoc_graph_card_count_dpcm ) ;
dev_dbg ( dev , " link %d, dais %d, ccnf %d \n " ,
li - > link , li - > dais , li - > conf ) ;
2017-04-20 01:36:08 +00:00
}
2018-06-11 17:32:14 +09:00
static int asoc_graph_soc_card_probe ( struct snd_soc_card * card )
{
struct graph_card_data * priv = snd_soc_card_get_drvdata ( card ) ;
int ret ;
ret = asoc_simple_card_init_hp ( card , & priv - > hp_jack , NULL ) ;
if ( ret < 0 )
return ret ;
ret = asoc_simple_card_init_mic ( card , & priv - > mic_jack , NULL ) ;
if ( ret < 0 )
return ret ;
return 0 ;
}
2017-04-20 01:36:08 +00:00
static int asoc_graph_card_probe ( struct platform_device * pdev )
{
struct graph_card_data * priv ;
struct snd_soc_dai_link * dai_link ;
struct graph_dai_props * dai_props ;
2018-11-30 02:07:48 +00:00
struct asoc_simple_dai * dais ;
2017-04-20 01:36:08 +00:00
struct device * dev = & pdev - > dev ;
struct snd_soc_card * card ;
2018-12-14 11:32:25 +09:00
struct snd_soc_codec_conf * cconf ;
2018-12-20 10:45:59 +09:00
struct link_info li ;
2018-12-14 11:32:25 +09:00
int ret , i ;
2017-04-20 01:36:08 +00:00
/* Allocate the private data and the DAI link array */
priv = devm_kzalloc ( dev , sizeof ( * priv ) , GFP_KERNEL ) ;
if ( ! priv )
return - ENOMEM ;
2018-12-20 10:46:05 +09:00
card = graph_priv_to_card ( priv ) ;
card - > owner = THIS_MODULE ;
card - > dev = dev ;
card - > dapm_widgets = asoc_graph_card_dapm_widgets ;
card - > num_dapm_widgets = ARRAY_SIZE ( asoc_graph_card_dapm_widgets ) ;
card - > probe = asoc_graph_soc_card_probe ;
2018-12-20 10:45:59 +09:00
memset ( & li , 0 , sizeof ( li ) ) ;
2018-12-20 10:46:05 +09:00
asoc_graph_get_dais_count ( priv , & li ) ;
2018-12-20 10:45:59 +09:00
if ( ! li . link | | ! li . dais )
2017-04-20 01:36:08 +00:00
return - EINVAL ;
2018-12-20 10:45:59 +09:00
dai_props = devm_kcalloc ( dev , li . link , sizeof ( * dai_props ) , GFP_KERNEL ) ;
dai_link = devm_kcalloc ( dev , li . link , sizeof ( * dai_link ) , GFP_KERNEL ) ;
dais = devm_kcalloc ( dev , li . dais , sizeof ( * dais ) , GFP_KERNEL ) ;
cconf = devm_kcalloc ( dev , li . conf , sizeof ( * cconf ) , GFP_KERNEL ) ;
2018-11-30 02:07:48 +00:00
if ( ! dai_props | | ! dai_link | | ! dais )
2017-04-20 01:36:08 +00:00
return - ENOMEM ;
2018-08-31 03:09:05 +00:00
/*
* Use snd_soc_dai_link_component instead of legacy style
* It is codec only . but cpu / platform will be supported in the future .
* see
* soc - core . c : : snd_soc_init_multicodec ( )
*/
2018-12-20 10:45:59 +09:00
for ( i = 0 ; i < li . link ; i + + ) {
2018-08-31 03:09:05 +00:00
dai_link [ i ] . codecs = & dai_props [ i ] . codecs ;
dai_link [ i ] . num_codecs = 1 ;
2018-08-31 03:10:58 +00:00
dai_link [ i ] . platform = & dai_props [ i ] . platform ;
2018-08-31 03:09:05 +00:00
}
2017-06-29 21:26:38 +08:00
priv - > pa_gpio = devm_gpiod_get_optional ( dev , " pa " , GPIOD_OUT_LOW ) ;
if ( IS_ERR ( priv - > pa_gpio ) ) {
ret = PTR_ERR ( priv - > pa_gpio ) ;
dev_err ( dev , " failed to get amplifier gpio: %d \n " , ret ) ;
return ret ;
}
2018-12-20 10:46:05 +09:00
priv - > dai_props = dai_props ;
priv - > dai_link = dai_link ;
priv - > dais = dais ;
priv - > codec_conf = cconf ;
2017-04-20 01:36:08 +00:00
2018-12-14 11:32:25 +09:00
card - > dai_link = dai_link ;
2018-12-20 10:45:59 +09:00
card - > num_links = li . link ;
2018-12-14 11:32:25 +09:00
card - > codec_conf = cconf ;
2018-12-20 10:45:59 +09:00
card - > num_configs = li . conf ;
2017-04-20 01:36:08 +00:00
ret = asoc_graph_card_parse_of ( priv ) ;
if ( ret < 0 ) {
if ( ret ! = - EPROBE_DEFER )
dev_err ( dev , " parse error %d \n " , ret ) ;
goto err ;
}
snd_soc_card_set_drvdata ( card , priv ) ;
ret = devm_snd_soc_register_card ( dev , card ) ;
2017-05-19 00:58:00 +00:00
if ( ret < 0 )
goto err ;
return 0 ;
2017-04-20 01:36:08 +00:00
err :
asoc_simple_card_clean_reference ( card ) ;
return ret ;
}
static int asoc_graph_card_remove ( struct platform_device * pdev )
{
struct snd_soc_card * card = platform_get_drvdata ( pdev ) ;
return asoc_simple_card_clean_reference ( card ) ;
}
static const struct of_device_id asoc_graph_of_match [ ] = {
{ . compatible = " audio-graph-card " , } ,
2018-12-14 11:32:25 +09:00
{ . compatible = " audio-graph-scu-card " , } ,
2017-04-20 01:36:08 +00:00
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , asoc_graph_of_match ) ;
static struct platform_driver asoc_graph_card = {
. driver = {
. name = " asoc-audio-graph-card " ,
2017-08-22 04:56:44 +00:00
. pm = & snd_soc_pm_ops ,
2017-04-20 01:36:08 +00:00
. of_match_table = asoc_graph_of_match ,
} ,
. probe = asoc_graph_card_probe ,
. remove = asoc_graph_card_remove ,
} ;
module_platform_driver ( asoc_graph_card ) ;
MODULE_ALIAS ( " platform:asoc-audio-graph-card " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
MODULE_DESCRIPTION ( " ASoC Audio Graph Sound Card " ) ;
MODULE_AUTHOR ( " Kuninori Morimoto <kuninori.morimoto.gx@renesas.com> " ) ;