2019-05-19 15:51:31 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2011-07-21 12:36:57 +08:00
/*
* Copyright 2011 Freescale Semiconductor , Inc .
*/
# include <linux/module.h>
# include <linux/device.h>
2012-05-11 22:24:18 +08:00
# include <linux/of.h>
# include <linux/of_device.h>
2011-07-21 12:36:57 +08:00
# include <sound/core.h>
# include <sound/pcm.h>
# include <sound/soc.h>
# include <sound/jack.h>
# include <sound/soc-dapm.h>
# include "../codecs/sgtl5000.h"
# include "mxs-saif.h"
static int mxs_sgtl5000_hw_params ( struct snd_pcm_substream * substream ,
struct snd_pcm_hw_params * params )
{
2020-07-20 10:19:51 +09:00
struct snd_soc_pcm_runtime * rtd = asoc_substream_to_rtd ( substream ) ;
2020-03-23 14:19:39 +09:00
struct snd_soc_dai * codec_dai = asoc_rtd_to_codec ( rtd , 0 ) ;
struct snd_soc_dai * cpu_dai = asoc_rtd_to_cpu ( rtd , 0 ) ;
2011-07-21 12:36:57 +08:00
unsigned int rate = params_rate ( params ) ;
2015-01-01 17:16:15 +01:00
u32 mclk ;
2011-07-21 12:36:57 +08:00
int ret ;
/* sgtl5000 does not support 512*rate when in 96000 fs */
switch ( rate ) {
case 96000 :
mclk = 256 * rate ;
break ;
default :
mclk = 512 * rate ;
break ;
}
/* Set SGTL5000's SYSCLK (provided by SAIF MCLK) */
ret = snd_soc_dai_set_sysclk ( codec_dai , SGTL5000_SYSCLK , mclk , 0 ) ;
2013-08-02 10:30:15 +02:00
if ( ret ) {
dev_err ( codec_dai - > dev , " Failed to set sysclk to %u.%03uMHz \n " ,
mclk / 1000000 , mclk / 1000 % 1000 ) ;
2011-07-21 12:36:57 +08:00
return ret ;
2013-08-02 10:30:15 +02:00
}
2011-07-21 12:36:57 +08:00
/* The SAIF MCLK should be the same as SGTL5000_SYSCLK */
ret = snd_soc_dai_set_sysclk ( cpu_dai , MXS_SAIF_MCLK , mclk , 0 ) ;
2013-08-02 10:30:15 +02:00
if ( ret ) {
dev_err ( cpu_dai - > dev , " Failed to set sysclk to %u.%03uMHz \n " ,
mclk / 1000000 , mclk / 1000 % 1000 ) ;
2011-07-21 12:36:57 +08:00
return ret ;
2013-08-02 10:30:15 +02:00
}
2011-07-21 12:36:57 +08:00
return 0 ;
}
2016-10-15 16:55:49 +02:00
static const struct snd_soc_ops mxs_sgtl5000_hifi_ops = {
2011-07-21 12:36:57 +08:00
. hw_params = mxs_sgtl5000_hw_params ,
} ;
2015-01-01 17:16:15 +01:00
# define MXS_SGTL5000_DAI_FMT (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF | \
SND_SOC_DAIFMT_CBS_CFS )
2019-06-06 13:16:37 +09:00
SND_SOC_DAILINK_DEFS ( hifi_tx ,
DAILINK_COMP_ARRAY ( COMP_EMPTY ( ) ) ,
2019-06-28 10:47:42 +09:00
DAILINK_COMP_ARRAY ( COMP_CODEC ( NULL , " sgtl5000 " ) ) ,
DAILINK_COMP_ARRAY ( COMP_EMPTY ( ) ) ) ;
2019-06-06 13:16:37 +09:00
SND_SOC_DAILINK_DEFS ( hifi_rx ,
DAILINK_COMP_ARRAY ( COMP_EMPTY ( ) ) ,
2019-06-28 10:47:42 +09:00
DAILINK_COMP_ARRAY ( COMP_CODEC ( NULL , " sgtl5000 " ) ) ,
DAILINK_COMP_ARRAY ( COMP_EMPTY ( ) ) ) ;
2019-06-06 13:16:37 +09:00
2011-07-21 12:36:57 +08:00
static struct snd_soc_dai_link mxs_sgtl5000_dai [ ] = {
{
2011-08-22 00:02:46 +08:00
. name = " HiFi Tx " ,
2011-07-21 12:36:57 +08:00
. stream_name = " HiFi Playback " ,
2015-01-01 17:16:15 +01:00
. dai_fmt = MXS_SGTL5000_DAI_FMT ,
2011-07-21 12:36:57 +08:00
. ops = & mxs_sgtl5000_hifi_ops ,
2013-08-29 10:32:14 -03:00
. playback_only = true ,
2019-06-06 13:16:37 +09:00
SND_SOC_DAILINK_REG ( hifi_tx ) ,
2011-08-22 00:02:46 +08:00
} , {
. name = " HiFi Rx " ,
. stream_name = " HiFi Capture " ,
2015-01-01 17:16:15 +01:00
. dai_fmt = MXS_SGTL5000_DAI_FMT ,
2011-08-22 00:02:46 +08:00
. ops = & mxs_sgtl5000_hifi_ops ,
2013-08-29 10:32:14 -03:00
. capture_only = true ,
2019-06-06 13:16:37 +09:00
SND_SOC_DAILINK_REG ( hifi_rx ) ,
2011-07-21 12:36:57 +08:00
} ,
} ;
2018-01-12 14:22:59 +01:00
static const struct snd_soc_dapm_widget mxs_sgtl5000_dapm_widgets [ ] = {
SND_SOC_DAPM_MIC ( " Mic Jack " , NULL ) ,
SND_SOC_DAPM_LINE ( " Line In Jack " , NULL ) ,
SND_SOC_DAPM_HP ( " Headphone Jack " , NULL ) ,
SND_SOC_DAPM_SPK ( " Line Out Jack " , NULL ) ,
SND_SOC_DAPM_SPK ( " Ext Spk " , NULL ) ,
} ;
2011-07-21 12:36:57 +08:00
static struct snd_soc_card mxs_sgtl5000 = {
. name = " mxs_sgtl5000 " ,
2011-12-28 20:06:21 +08:00
. owner = THIS_MODULE ,
2011-07-21 12:36:57 +08:00
. dai_link = mxs_sgtl5000_dai ,
. num_links = ARRAY_SIZE ( mxs_sgtl5000_dai ) ,
} ;
2013-09-02 17:56:18 -03:00
static int mxs_sgtl5000_probe ( struct platform_device * pdev )
2012-05-11 22:24:18 +08:00
{
2013-09-02 17:56:18 -03:00
struct snd_soc_card * card = & mxs_sgtl5000 ;
int ret , i ;
2012-05-11 22:24:18 +08:00
struct device_node * np = pdev - > dev . of_node ;
struct device_node * saif_np [ 2 ] , * codec_np ;
saif_np [ 0 ] = of_parse_phandle ( np , " saif-controllers " , 0 ) ;
saif_np [ 1 ] = of_parse_phandle ( np , " saif-controllers " , 1 ) ;
codec_np = of_parse_phandle ( np , " audio-codec " , 0 ) ;
if ( ! saif_np [ 0 ] | | ! saif_np [ 1 ] | | ! codec_np ) {
dev_err ( & pdev - > dev , " phandle missing or invalid \n " ) ;
return - EINVAL ;
}
for ( i = 0 ; i < 2 ; i + + ) {
2019-06-06 13:16:37 +09:00
mxs_sgtl5000_dai [ i ] . codecs - > name = NULL ;
mxs_sgtl5000_dai [ i ] . codecs - > of_node = codec_np ;
mxs_sgtl5000_dai [ i ] . cpus - > dai_name = NULL ;
mxs_sgtl5000_dai [ i ] . cpus - > of_node = saif_np [ i ] ;
2019-06-28 10:47:42 +09:00
mxs_sgtl5000_dai [ i ] . platforms - > name = NULL ;
mxs_sgtl5000_dai [ i ] . platforms - > of_node = saif_np [ i ] ;
2012-05-11 22:24:18 +08:00
}
of_node_put ( codec_np ) ;
of_node_put ( saif_np [ 0 ] ) ;
of_node_put ( saif_np [ 1 ] ) ;
2011-07-21 12:36:57 +08:00
/*
* Set an init clock ( 11.28 Mhz ) for sgtl5000 initialization ( i2c r / w ) .
* The Sgtl5000 sysclk is derived from saif0 mclk and it ' s range
* should be > = 8 MHz and < = 27 M .
*/
ret = mxs_saif_get_mclk ( 0 , 44100 * 256 , 44100 ) ;
2013-08-02 10:30:15 +02:00
if ( ret ) {
dev_err ( & pdev - > dev , " failed to get mclk \n " ) ;
2011-07-21 12:36:57 +08:00
return ret ;
2013-08-02 10:30:15 +02:00
}
2011-07-21 12:36:57 +08:00
card - > dev = & pdev - > dev ;
2018-01-12 14:22:59 +01:00
if ( of_find_property ( np , " audio-routing " , NULL ) ) {
card - > dapm_widgets = mxs_sgtl5000_dapm_widgets ;
card - > num_dapm_widgets = ARRAY_SIZE ( mxs_sgtl5000_dapm_widgets ) ;
ret = snd_soc_of_parse_audio_routing ( card , " audio-routing " ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " failed to parse audio-routing (%d) \n " ,
ret ) ;
return ret ;
}
}
2015-08-30 17:20:01 +08:00
ret = devm_snd_soc_register_card ( & pdev - > dev , card ) ;
2021-12-14 11:08:38 +09:00
if ( ret )
return dev_err_probe ( & pdev - > dev , ret , " snd_soc_register_card failed \n " ) ;
2011-07-21 12:36:57 +08:00
return 0 ;
}
2012-12-07 09:26:27 -05:00
static int mxs_sgtl5000_remove ( struct platform_device * pdev )
2011-07-21 12:36:57 +08:00
{
mxs_saif_put_mclk ( 0 ) ;
return 0 ;
}
2012-05-11 22:24:18 +08:00
static const struct of_device_id mxs_sgtl5000_dt_ids [ ] = {
{ . compatible = " fsl,mxs-audio-sgtl5000 " , } ,
{ /* sentinel */ }
} ;
MODULE_DEVICE_TABLE ( of , mxs_sgtl5000_dt_ids ) ;
2011-07-21 12:36:57 +08:00
static struct platform_driver mxs_sgtl5000_audio_driver = {
. driver = {
. name = " mxs-sgtl5000 " ,
2012-05-11 22:24:18 +08:00
. of_match_table = mxs_sgtl5000_dt_ids ,
2011-07-21 12:36:57 +08:00
} ,
. probe = mxs_sgtl5000_probe ,
2012-12-07 09:26:27 -05:00
. remove = mxs_sgtl5000_remove ,
2011-07-21 12:36:57 +08:00
} ;
2011-11-24 14:21:29 +08:00
module_platform_driver ( mxs_sgtl5000_audio_driver ) ;
2011-07-21 12:36:57 +08:00
MODULE_AUTHOR ( " Freescale Semiconductor, Inc. " ) ;
MODULE_DESCRIPTION ( " MXS ALSA SoC Machine driver " ) ;
MODULE_LICENSE ( " GPL " ) ;
2011-12-09 14:38:12 +01:00
MODULE_ALIAS ( " platform:mxs-sgtl5000 " ) ;