2019-04-19 12:21:45 +02:00
// SPDX-License-Identifier: GPL-2.0+
//
// neo1973_wm8753.c - SoC audio for Openmoko Neo1973 and Freerunner devices
//
// Copyright 2007 Openmoko Inc
// Author: Graeme Gregory <graeme@openmoko.org>
// Copyright 2007 Wolfson Microelectronics PLC.
// Author: Graeme Gregory
// graeme.gregory@wolfsonmicro.com or linux@wolfsonmicro.com
// Copyright 2009 Wolfson Microelectronics
2007-05-14 11:03:52 +02:00
# include <linux/module.h>
# include <linux/platform_device.h>
2020-08-06 20:20:42 +02:00
# include <linux/gpio/consumer.h>
2011-03-07 08:04:58 +01:00
2007-05-14 11:03:52 +02:00
# include <sound/soc.h>
2013-04-11 19:08:42 +02:00
# include "regs-iis.h"
2007-05-14 11:03:52 +02:00
# include "../codecs/wm8753.h"
# include "s3c24xx-i2s.h"
static int neo1973_hifi_hw_params ( struct snd_pcm_substream * substream ,
struct snd_pcm_hw_params * params )
{
2020-07-20 10:18:14 +09:00
struct snd_soc_pcm_runtime * rtd = asoc_substream_to_rtd ( substream ) ;
2020-03-23 14:20:20 +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 ) ;
2007-05-14 11:03:52 +02:00
unsigned int pll_out = 0 , bclk = 0 ;
int ret = 0 ;
unsigned long iis_clkrate ;
iis_clkrate = s3c24xx_i2s_get_clockrate ( ) ;
switch ( params_rate ( params ) ) {
case 8000 :
case 16000 :
pll_out = 12288000 ;
break ;
case 48000 :
bclk = WM8753_BCLK_DIV_4 ;
pll_out = 12288000 ;
break ;
case 96000 :
bclk = WM8753_BCLK_DIV_2 ;
pll_out = 12288000 ;
break ;
case 11025 :
bclk = WM8753_BCLK_DIV_16 ;
pll_out = 11289600 ;
break ;
case 22050 :
bclk = WM8753_BCLK_DIV_8 ;
pll_out = 11289600 ;
break ;
case 44100 :
bclk = WM8753_BCLK_DIV_4 ;
pll_out = 11289600 ;
break ;
case 88200 :
bclk = WM8753_BCLK_DIV_2 ;
pll_out = 11289600 ;
break ;
}
/* set the codec system clock for DAC and ADC */
2008-07-08 13:19:18 +01:00
ret = snd_soc_dai_set_sysclk ( codec_dai , WM8753_MCLK , pll_out ,
2007-05-14 11:03:52 +02:00
SND_SOC_CLOCK_IN ) ;
if ( ret < 0 )
return ret ;
/* set MCLK division for sample rate */
2008-07-08 13:19:18 +01:00
ret = snd_soc_dai_set_clkdiv ( cpu_dai , S3C24XX_DIV_MCLK ,
2008-04-30 20:24:54 +02:00
S3C2410_IISMOD_32FS ) ;
2007-05-14 11:03:52 +02:00
if ( ret < 0 )
return ret ;
/* set codec BCLK division for sample rate */
2008-07-08 13:19:18 +01:00
ret = snd_soc_dai_set_clkdiv ( codec_dai , WM8753_BCLKDIV , bclk ) ;
2007-05-14 11:03:52 +02:00
if ( ret < 0 )
return ret ;
/* set prescaler division for sample rate */
2008-07-08 13:19:18 +01:00
ret = snd_soc_dai_set_clkdiv ( cpu_dai , S3C24XX_DIV_PRESCALER ,
2008-04-30 20:24:54 +02:00
S3C24XX_PRESCALE ( 4 , 4 ) ) ;
2007-05-14 11:03:52 +02:00
if ( ret < 0 )
return ret ;
/* codec PLL input is PCLK/4 */
2009-09-05 18:52:16 +01:00
ret = snd_soc_dai_set_pll ( codec_dai , WM8753_PLL1 , 0 ,
2007-05-14 11:03:52 +02:00
iis_clkrate / 4 , pll_out ) ;
if ( ret < 0 )
return ret ;
return 0 ;
}
static int neo1973_hifi_hw_free ( struct snd_pcm_substream * substream )
{
2020-07-20 10:18:14 +09:00
struct snd_soc_pcm_runtime * rtd = asoc_substream_to_rtd ( substream ) ;
2020-03-23 14:20:20 +09:00
struct snd_soc_dai * codec_dai = asoc_rtd_to_codec ( rtd , 0 ) ;
2007-05-14 11:03:52 +02:00
/* disable the PLL */
2009-10-01 08:40:32 +02:00
return snd_soc_dai_set_pll ( codec_dai , WM8753_PLL1 , 0 , 0 , 0 ) ;
2007-05-14 11:03:52 +02:00
}
/*
* Neo1973 WM8753 HiFi DAI opserations .
*/
2021-07-28 19:25:48 +02:00
static const struct snd_soc_ops neo1973_hifi_ops = {
2007-05-14 11:03:52 +02:00
. hw_params = neo1973_hifi_hw_params ,
. hw_free = neo1973_hifi_hw_free ,
} ;
static int neo1973_voice_hw_params ( struct snd_pcm_substream * substream ,
struct snd_pcm_hw_params * params )
{
2020-07-20 10:18:14 +09:00
struct snd_soc_pcm_runtime * rtd = asoc_substream_to_rtd ( substream ) ;
2020-03-23 14:20:20 +09:00
struct snd_soc_dai * codec_dai = asoc_rtd_to_codec ( rtd , 0 ) ;
2007-05-14 11:03:52 +02:00
unsigned int pcmdiv = 0 ;
int ret = 0 ;
unsigned long iis_clkrate ;
iis_clkrate = s3c24xx_i2s_get_clockrate ( ) ;
if ( params_rate ( params ) ! = 8000 )
return - EINVAL ;
if ( params_channels ( params ) ! = 1 )
return - EINVAL ;
pcmdiv = WM8753_PCM_DIV_6 ; /* 2.048 MHz */
/* set the codec system clock for DAC and ADC */
2008-07-08 13:19:18 +01:00
ret = snd_soc_dai_set_sysclk ( codec_dai , WM8753_PCMCLK , 12288000 ,
2007-05-14 11:03:52 +02:00
SND_SOC_CLOCK_IN ) ;
if ( ret < 0 )
return ret ;
/* set codec PCM division for sample rate */
2008-07-08 13:19:18 +01:00
ret = snd_soc_dai_set_clkdiv ( codec_dai , WM8753_PCMDIV , pcmdiv ) ;
2007-05-14 11:03:52 +02:00
if ( ret < 0 )
return ret ;
2010-10-16 15:19:20 +02:00
/* configure and enable PLL for 12.288MHz output */
2009-10-01 08:40:32 +02:00
ret = snd_soc_dai_set_pll ( codec_dai , WM8753_PLL2 , 0 ,
2007-05-14 11:03:52 +02:00
iis_clkrate / 4 , 12288000 ) ;
if ( ret < 0 )
return ret ;
return 0 ;
}
static int neo1973_voice_hw_free ( struct snd_pcm_substream * substream )
{
2020-07-20 10:18:14 +09:00
struct snd_soc_pcm_runtime * rtd = asoc_substream_to_rtd ( substream ) ;
2020-03-23 14:20:20 +09:00
struct snd_soc_dai * codec_dai = asoc_rtd_to_codec ( rtd , 0 ) ;
2007-05-14 11:03:52 +02:00
/* disable the PLL */
2009-10-01 08:40:32 +02:00
return snd_soc_dai_set_pll ( codec_dai , WM8753_PLL2 , 0 , 0 , 0 ) ;
2007-05-14 11:03:52 +02:00
}
2021-07-28 19:25:48 +02:00
static const struct snd_soc_ops neo1973_voice_ops = {
2007-05-14 11:03:52 +02:00
. hw_params = neo1973_voice_hw_params ,
. hw_free = neo1973_voice_hw_free ,
} ;
2020-08-06 20:20:42 +02:00
static struct gpio_desc * gpiod_hp_in , * gpiod_amp_shut ;
2011-03-07 08:04:58 +01:00
static int gta02_speaker_enabled ;
static int lm4853_set_spk ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
gta02_speaker_enabled = ucontrol - > value . integer . value [ 0 ] ;
2020-08-06 20:20:42 +02:00
gpiod_set_value ( gpiod_hp_in , ! gta02_speaker_enabled ) ;
2011-03-07 08:04:58 +01:00
return 0 ;
}
static int lm4853_get_spk ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
ucontrol - > value . integer . value [ 0 ] = gta02_speaker_enabled ;
return 0 ;
}
static int lm4853_event ( struct snd_soc_dapm_widget * w ,
struct snd_kcontrol * k , int event )
{
2020-08-06 20:20:42 +02:00
gpiod_set_value ( gpiod_amp_shut , SND_SOC_DAPM_EVENT_OFF ( event ) ) ;
2011-03-07 08:04:58 +01:00
return 0 ;
}
2014-03-01 16:17:02 +01:00
static const struct snd_soc_dapm_widget neo1973_wm8753_dapm_widgets [ ] = {
SND_SOC_DAPM_LINE ( " GSM Line Out " , NULL ) ,
SND_SOC_DAPM_LINE ( " GSM Line In " , NULL ) ,
SND_SOC_DAPM_MIC ( " Headset Mic " , NULL ) ,
SND_SOC_DAPM_MIC ( " Handset Mic " , NULL ) ,
SND_SOC_DAPM_SPK ( " Handset Spk " , NULL ) ,
SND_SOC_DAPM_SPK ( " Stereo Out " , lm4853_event ) ,
} ;
static const struct snd_soc_dapm_route neo1973_wm8753_routes [ ] = {
/* Connections to the GSM Module */
{ " GSM Line Out " , NULL , " MONO1 " } ,
{ " GSM Line Out " , NULL , " MONO2 " } ,
{ " RXP " , NULL , " GSM Line In " } ,
{ " RXN " , NULL , " GSM Line In " } ,
/* Connections to Headset */
{ " MIC1 " , NULL , " Mic Bias " } ,
{ " Mic Bias " , NULL , " Headset Mic " } ,
/* Call Mic */
{ " MIC2 " , NULL , " Mic Bias " } ,
{ " MIC2N " , NULL , " Mic Bias " } ,
{ " Mic Bias " , NULL , " Handset Mic " } ,
/* Connect the ALC pins */
{ " ACIN " , NULL , " ACOP " } ,
2011-03-07 08:04:58 +01:00
/* Connections to the amp */
{ " Stereo Out " , NULL , " LOUT1 " } ,
{ " Stereo Out " , NULL , " ROUT1 " } ,
/* Call Speaker */
{ " Handset Spk " , NULL , " LOUT2 " } ,
{ " Handset Spk " , NULL , " ROUT2 " } ,
} ;
2014-03-01 16:17:02 +01:00
static const struct snd_kcontrol_new neo1973_wm8753_controls [ ] = {
SOC_DAPM_PIN_SWITCH ( " GSM Line Out " ) ,
SOC_DAPM_PIN_SWITCH ( " GSM Line In " ) ,
SOC_DAPM_PIN_SWITCH ( " Headset Mic " ) ,
SOC_DAPM_PIN_SWITCH ( " Handset Mic " ) ,
2011-03-07 08:04:58 +01:00
SOC_DAPM_PIN_SWITCH ( " Handset Spk " ) ,
SOC_DAPM_PIN_SWITCH ( " Stereo Out " ) ,
SOC_SINGLE_BOOL_EXT ( " Amp Spk Switch " , 0 ,
lm4853_get_spk ,
lm4853_set_spk ) ,
} ;
2010-03-17 20:15:21 +00:00
static int neo1973_wm8753_init ( struct snd_soc_pcm_runtime * rtd )
2007-05-14 11:03:52 +02:00
{
2014-03-01 16:17:03 +01:00
struct snd_soc_card * card = rtd - > card ;
2008-05-05 14:16:12 +02:00
2011-03-07 08:04:58 +01:00
/* set endpoints to default off mode */
2014-03-01 16:17:03 +01:00
snd_soc_dapm_disable_pin ( & card - > dapm , " GSM Line Out " ) ;
snd_soc_dapm_disable_pin ( & card - > dapm , " GSM Line In " ) ;
snd_soc_dapm_disable_pin ( & card - > dapm , " Headset Mic " ) ;
snd_soc_dapm_disable_pin ( & card - > dapm , " Handset Mic " ) ;
snd_soc_dapm_disable_pin ( & card - > dapm , " Stereo Out " ) ;
snd_soc_dapm_disable_pin ( & card - > dapm , " Handset Spk " ) ;
2011-03-07 08:04:58 +01:00
/* allow audio paths from the GSM modem to run during suspend */
2014-03-01 16:17:03 +01:00
snd_soc_dapm_ignore_suspend ( & card - > dapm , " GSM Line Out " ) ;
snd_soc_dapm_ignore_suspend ( & card - > dapm , " GSM Line In " ) ;
snd_soc_dapm_ignore_suspend ( & card - > dapm , " Headset Mic " ) ;
snd_soc_dapm_ignore_suspend ( & card - > dapm , " Handset Mic " ) ;
snd_soc_dapm_ignore_suspend ( & card - > dapm , " Stereo Out " ) ;
snd_soc_dapm_ignore_suspend ( & card - > dapm , " Handset Spk " ) ;
2007-05-14 11:03:52 +02:00
return 0 ;
}
2019-06-06 13:10:32 +09:00
SND_SOC_DAILINK_DEFS ( wm8753 ,
DAILINK_COMP_ARRAY ( COMP_CPU ( " s3c24xx-iis " ) ) ,
DAILINK_COMP_ARRAY ( COMP_CODEC ( " wm8753.0-001a " , " wm8753-hifi " ) ) ,
DAILINK_COMP_ARRAY ( COMP_PLATFORM ( " s3c24xx-iis " ) ) ) ;
SND_SOC_DAILINK_DEFS ( bluetooth ,
DAILINK_COMP_ARRAY ( COMP_CPU ( " bt-sco-pcm " ) ) ,
DAILINK_COMP_ARRAY ( COMP_CODEC ( " wm8753.0-001a " , " wm8753-voice " ) ) ) ;
2007-05-14 11:03:52 +02:00
static struct snd_soc_dai_link neo1973_dai [ ] = {
{ /* Hifi Playback - for similatious use with voice below */
. name = " WM8753 " ,
. stream_name = " WM8753 HiFi " ,
2015-01-01 17:16:22 +01:00
. dai_fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF |
SND_SOC_DAIFMT_CBM_CFM ,
2007-05-14 11:03:52 +02:00
. init = neo1973_wm8753_init ,
. ops = & neo1973_hifi_ops ,
2019-06-06 13:10:32 +09:00
SND_SOC_DAILINK_REG ( wm8753 ) ,
2007-05-14 11:03:52 +02:00
} ,
{ /* Voice via BT */
. name = " Bluetooth " ,
. stream_name = " Voice " ,
2015-01-01 17:16:22 +01:00
. dai_fmt = SND_SOC_DAIFMT_DSP_B | SND_SOC_DAIFMT_NB_NF |
SND_SOC_DAIFMT_CBS_CFS ,
2007-05-14 11:03:52 +02:00
. ops = & neo1973_voice_ops ,
2019-06-06 13:10:32 +09:00
SND_SOC_DAILINK_REG ( bluetooth ) ,
2007-05-14 11:03:52 +02:00
} ,
} ;
2011-03-07 08:04:55 +01:00
static struct snd_soc_aux_dev neo1973_aux_devs [ ] = {
2011-03-07 08:04:59 +01:00
{
2019-08-08 14:54:10 +09:00
. dlc = COMP_AUX ( " dfbmcs320.0 " ) ,
2011-03-07 08:04:59 +01:00
} ,
2007-05-14 11:03:52 +02:00
} ;
2011-03-07 08:04:55 +01:00
static struct snd_soc_codec_conf neo1973_codec_conf [ ] = {
{
2019-12-13 09:55:39 +09:00
. dlc = COMP_CODEC_CONF ( " lm4857.0-007c " ) ,
2011-03-07 08:04:55 +01:00
. name_prefix = " Amp " ,
} ,
2008-09-01 17:46:57 +02:00
} ;
2011-03-07 08:04:55 +01:00
static struct snd_soc_card neo1973 = {
2020-08-06 20:20:42 +02:00
. name = " neo1973gta02 " ,
2011-12-22 10:53:15 +08:00
. owner = THIS_MODULE ,
2011-03-07 08:04:55 +01:00
. dai_link = neo1973_dai ,
. num_links = ARRAY_SIZE ( neo1973_dai ) ,
. aux_dev = neo1973_aux_devs ,
. num_aux_devs = ARRAY_SIZE ( neo1973_aux_devs ) ,
. codec_conf = neo1973_codec_conf ,
. num_configs = ARRAY_SIZE ( neo1973_codec_conf ) ,
2014-03-01 16:17:03 +01:00
. controls = neo1973_wm8753_controls ,
. num_controls = ARRAY_SIZE ( neo1973_wm8753_controls ) ,
. dapm_widgets = neo1973_wm8753_dapm_widgets ,
. num_dapm_widgets = ARRAY_SIZE ( neo1973_wm8753_dapm_widgets ) ,
. dapm_routes = neo1973_wm8753_routes ,
. num_dapm_routes = ARRAY_SIZE ( neo1973_wm8753_routes ) ,
2014-05-20 11:13:28 +02:00
. fully_routed = true ,
2007-05-14 11:03:52 +02:00
} ;
2020-08-06 20:20:42 +02:00
static int neo1973_probe ( struct platform_device * pdev )
2007-05-14 11:03:52 +02:00
{
2020-08-06 20:20:42 +02:00
struct device * dev = & pdev - > dev ;
2011-03-07 08:04:58 +01:00
2020-08-06 20:20:42 +02:00
gpiod_hp_in = devm_gpiod_get ( dev , " hp " , GPIOD_OUT_HIGH ) ;
if ( IS_ERR ( gpiod_hp_in ) ) {
dev_err ( dev , " missing gpio %s \n " , " hp " ) ;
return PTR_ERR ( gpiod_hp_in ) ;
2008-10-08 13:02:20 +01:00
}
2020-08-06 20:20:42 +02:00
gpiod_amp_shut = devm_gpiod_get ( dev , " amp-shut " , GPIOD_OUT_HIGH ) ;
if ( IS_ERR ( gpiod_amp_shut ) ) {
dev_err ( dev , " missing gpio %s \n " , " amp-shut " ) ;
return PTR_ERR ( gpiod_amp_shut ) ;
2011-03-07 08:04:58 +01:00
}
2020-08-06 20:20:42 +02:00
neo1973 . dev = dev ;
return devm_snd_soc_register_card ( dev , & neo1973 ) ;
2007-05-14 11:03:52 +02:00
}
2020-08-06 20:20:42 +02:00
struct platform_driver neo1973_audio = {
. driver = {
. name = " neo1973-audio " ,
. pm = & snd_soc_pm_ops ,
} ,
. probe = neo1973_probe ,
} ;
module_platform_driver ( neo1973_audio ) ;
2007-05-14 11:03:52 +02:00
/* Module information */
2008-04-30 20:25:23 +02:00
MODULE_AUTHOR ( " Graeme Gregory, graeme@openmoko.org, www.openmoko.org " ) ;
2011-03-07 08:04:58 +01:00
MODULE_DESCRIPTION ( " ALSA SoC WM8753 Neo1973 and Frerunner " ) ;
2007-05-14 11:03:52 +02:00
MODULE_LICENSE ( " GPL " ) ;
2020-08-06 20:20:42 +02:00
MODULE_ALIAS ( " platform:neo1973-audio " ) ;