2019-05-27 08:55:01 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2006-10-12 14:31:16 +02:00
/*
* corgi . c - - SoC audio for Corgi
*
* Copyright 2005 Wolfson Microelectronics PLC .
* Copyright 2005 Openedhand Ltd .
*
2008-10-12 13:17:36 +01:00
* Authors : Liam Girdwood < lrg @ slimlogic . co . uk >
2006-10-12 14:31:16 +02:00
* Richard Purdie < richard @ openedhand . com >
*/
# include <linux/module.h>
# include <linux/moduleparam.h>
# include <linux/timer.h>
2009-02-18 12:34:53 +00:00
# include <linux/i2c.h>
2006-10-12 14:31:16 +02:00
# include <linux/interrupt.h>
# include <linux/platform_device.h>
2008-09-05 18:15:22 +08:00
# include <linux/gpio.h>
2006-10-12 14:31:16 +02:00
# include <sound/core.h>
# include <sound/pcm.h>
# include <sound/soc.h>
# include <asm/mach-types.h>
2019-09-02 00:02:08 +02:00
# include <linux/platform_data/asoc-pxa.h>
2006-10-12 14:31:16 +02:00
# include "../codecs/wm8731.h"
2007-02-02 17:22:20 +01:00
# include "pxa2xx-i2s.h"
2006-10-12 14:31:16 +02:00
# define CORGI_HP 0
# define CORGI_MIC 1
# define CORGI_LINE 2
# define CORGI_HEADSET 3
# define CORGI_HP_OFF 4
# define CORGI_SPK_ON 0
# define CORGI_SPK_OFF 1
/* audio clock in Hz - rounded from 12.235MHz */
# define CORGI_AUDIO_CLOCK 12288000
static int corgi_jack_func ;
static int corgi_spk_func ;
2019-09-11 13:57:45 +02:00
static struct gpio_desc * gpiod_mute_l , * gpiod_mute_r ,
* gpiod_apm_on , * gpiod_mic_bias ;
2012-02-03 17:43:09 +00:00
static void corgi_ext_control ( struct snd_soc_dapm_context * dapm )
2006-10-12 14:31:16 +02:00
{
2014-02-18 15:22:25 +00:00
snd_soc_dapm_mutex_lock ( dapm ) ;
2006-10-12 14:31:16 +02:00
/* set up jack connection */
switch ( corgi_jack_func ) {
case CORGI_HP :
/* set = unmute headphone */
2019-09-11 13:57:45 +02:00
gpiod_set_value ( gpiod_mute_l , 1 ) ;
gpiod_set_value ( gpiod_mute_r , 1 ) ;
2014-02-18 15:22:25 +00:00
snd_soc_dapm_disable_pin_unlocked ( dapm , " Mic Jack " ) ;
snd_soc_dapm_disable_pin_unlocked ( dapm , " Line Jack " ) ;
snd_soc_dapm_enable_pin_unlocked ( dapm , " Headphone Jack " ) ;
snd_soc_dapm_disable_pin_unlocked ( dapm , " Headset Jack " ) ;
2006-10-12 14:31:16 +02:00
break ;
case CORGI_MIC :
/* reset = mute headphone */
2019-09-11 13:57:45 +02:00
gpiod_set_value ( gpiod_mute_l , 0 ) ;
gpiod_set_value ( gpiod_mute_r , 0 ) ;
2014-02-18 15:22:25 +00:00
snd_soc_dapm_enable_pin_unlocked ( dapm , " Mic Jack " ) ;
snd_soc_dapm_disable_pin_unlocked ( dapm , " Line Jack " ) ;
snd_soc_dapm_disable_pin_unlocked ( dapm , " Headphone Jack " ) ;
snd_soc_dapm_disable_pin_unlocked ( dapm , " Headset Jack " ) ;
2006-10-12 14:31:16 +02:00
break ;
case CORGI_LINE :
2019-09-11 13:57:45 +02:00
gpiod_set_value ( gpiod_mute_l , 0 ) ;
gpiod_set_value ( gpiod_mute_r , 0 ) ;
2014-02-18 15:22:25 +00:00
snd_soc_dapm_disable_pin_unlocked ( dapm , " Mic Jack " ) ;
snd_soc_dapm_enable_pin_unlocked ( dapm , " Line Jack " ) ;
snd_soc_dapm_disable_pin_unlocked ( dapm , " Headphone Jack " ) ;
snd_soc_dapm_disable_pin_unlocked ( dapm , " Headset Jack " ) ;
2006-10-12 14:31:16 +02:00
break ;
case CORGI_HEADSET :
2019-09-11 13:57:45 +02:00
gpiod_set_value ( gpiod_mute_l , 0 ) ;
gpiod_set_value ( gpiod_mute_r , 1 ) ;
2014-02-18 15:22:25 +00:00
snd_soc_dapm_enable_pin_unlocked ( dapm , " Mic Jack " ) ;
snd_soc_dapm_disable_pin_unlocked ( dapm , " Line Jack " ) ;
snd_soc_dapm_disable_pin_unlocked ( dapm , " Headphone Jack " ) ;
snd_soc_dapm_enable_pin_unlocked ( dapm , " Headset Jack " ) ;
2006-10-12 14:31:16 +02:00
break ;
}
if ( corgi_spk_func = = CORGI_SPK_ON )
2014-02-18 15:22:25 +00:00
snd_soc_dapm_enable_pin_unlocked ( dapm , " Ext Spk " ) ;
2008-07-07 13:35:17 +01:00
else
2014-02-18 15:22:25 +00:00
snd_soc_dapm_disable_pin_unlocked ( dapm , " Ext Spk " ) ;
2006-10-12 14:31:16 +02:00
/* signal a DAPM event */
2014-02-18 15:22:25 +00:00
snd_soc_dapm_sync_unlocked ( dapm ) ;
snd_soc_dapm_mutex_unlock ( dapm ) ;
2006-10-12 14:31:16 +02:00
}
static int corgi_startup ( struct snd_pcm_substream * substream )
{
2020-07-20 10:18:20 +09:00
struct snd_soc_pcm_runtime * rtd = asoc_substream_to_rtd ( substream ) ;
2010-11-05 13:50:48 -04:00
2006-10-12 14:31:16 +02:00
/* check the jack status at stream startup */
2014-03-01 15:48:13 +01:00
corgi_ext_control ( & rtd - > card - > dapm ) ;
2010-11-05 13:50:48 -04:00
2006-10-12 14:31:16 +02:00
return 0 ;
}
/* we need to unmute the HP at shutdown as the mute burns power on corgi */
2008-12-01 20:00:47 +01:00
static void corgi_shutdown ( struct snd_pcm_substream * substream )
2006-10-12 14:31:16 +02:00
{
/* set = unmute headphone */
2019-09-11 13:57:45 +02:00
gpiod_set_value ( gpiod_mute_l , 1 ) ;
gpiod_set_value ( gpiod_mute_r , 1 ) ;
2006-10-12 14:31:16 +02:00
}
2007-02-02 17:22:20 +01:00
static int corgi_hw_params ( struct snd_pcm_substream * substream ,
struct snd_pcm_hw_params * params )
{
2020-07-20 10:18:20 +09:00
struct snd_soc_pcm_runtime * rtd = asoc_substream_to_rtd ( substream ) ;
2020-03-23 14:19:49 +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-02-02 17:22:20 +01:00
unsigned int clk = 0 ;
int ret = 0 ;
switch ( params_rate ( params ) ) {
case 8000 :
case 16000 :
case 48000 :
case 96000 :
clk = 12288000 ;
break ;
case 11025 :
case 22050 :
case 44100 :
clk = 11289600 ;
break ;
}
/* set the codec system clock for DAC and ADC */
2010-08-17 23:40:24 +01:00
ret = snd_soc_dai_set_sysclk ( codec_dai , WM8731_SYSCLK_XTAL , clk ,
2007-02-02 17:22:20 +01:00
SND_SOC_CLOCK_IN ) ;
if ( ret < 0 )
return ret ;
/* set the I2S system clock as input (unused) */
2008-07-08 13:19:18 +01:00
ret = snd_soc_dai_set_sysclk ( cpu_dai , PXA2XX_I2S_SYSCLK , 0 ,
2007-02-02 17:22:20 +01:00
SND_SOC_CLOCK_IN ) ;
if ( ret < 0 )
return ret ;
return 0 ;
}
2017-03-14 01:16:40 +05:30
static const struct snd_soc_ops corgi_ops = {
2006-10-12 14:31:16 +02:00
. startup = corgi_startup ,
2007-02-02 17:22:20 +01:00
. hw_params = corgi_hw_params ,
2006-10-12 14:31:16 +02:00
. shutdown = corgi_shutdown ,
} ;
static int corgi_get_jack ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
2016-02-29 17:23:49 +01:00
ucontrol - > value . enumerated . item [ 0 ] = corgi_jack_func ;
2006-10-12 14:31:16 +02:00
return 0 ;
}
static int corgi_set_jack ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
2012-02-03 17:43:09 +00:00
struct snd_soc_card * card = snd_kcontrol_chip ( kcontrol ) ;
2006-10-12 14:31:16 +02:00
2016-02-29 17:23:49 +01:00
if ( corgi_jack_func = = ucontrol - > value . enumerated . item [ 0 ] )
2006-10-12 14:31:16 +02:00
return 0 ;
2016-02-29 17:23:49 +01:00
corgi_jack_func = ucontrol - > value . enumerated . item [ 0 ] ;
2012-02-03 17:43:09 +00:00
corgi_ext_control ( & card - > dapm ) ;
2006-10-12 14:31:16 +02:00
return 1 ;
}
static int corgi_get_spk ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
2016-02-29 17:23:49 +01:00
ucontrol - > value . enumerated . item [ 0 ] = corgi_spk_func ;
2006-10-12 14:31:16 +02:00
return 0 ;
}
static int corgi_set_spk ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
2012-02-03 17:43:09 +00:00
struct snd_soc_card * card = snd_kcontrol_chip ( kcontrol ) ;
2006-10-12 14:31:16 +02:00
2016-02-29 17:23:49 +01:00
if ( corgi_spk_func = = ucontrol - > value . enumerated . item [ 0 ] )
2006-10-12 14:31:16 +02:00
return 0 ;
2016-02-29 17:23:49 +01:00
corgi_spk_func = ucontrol - > value . enumerated . item [ 0 ] ;
2012-02-03 17:43:09 +00:00
corgi_ext_control ( & card - > dapm ) ;
2006-10-12 14:31:16 +02:00
return 1 ;
}
2008-02-28 12:34:48 +01:00
static int corgi_amp_event ( struct snd_soc_dapm_widget * w ,
struct snd_kcontrol * k , int event )
2006-10-12 14:31:16 +02:00
{
2019-09-11 13:57:45 +02:00
gpiod_set_value ( gpiod_apm_on , SND_SOC_DAPM_EVENT_ON ( event ) ) ;
2006-10-12 14:31:16 +02:00
return 0 ;
}
2008-02-28 12:34:48 +01:00
static int corgi_mic_event ( struct snd_soc_dapm_widget * w ,
struct snd_kcontrol * k , int event )
2006-10-12 14:31:16 +02:00
{
2019-09-11 13:57:45 +02:00
gpiod_set_value ( gpiod_mic_bias , SND_SOC_DAPM_EVENT_ON ( event ) ) ;
2006-10-12 14:31:16 +02:00
return 0 ;
}
/* corgi machine dapm widgets */
static const struct snd_soc_dapm_widget wm8731_dapm_widgets [ ] = {
SND_SOC_DAPM_HP ( " Headphone Jack " , NULL ) ,
SND_SOC_DAPM_MIC ( " Mic Jack " , corgi_mic_event ) ,
SND_SOC_DAPM_SPK ( " Ext Spk " , corgi_amp_event ) ,
SND_SOC_DAPM_LINE ( " Line Jack " , NULL ) ,
SND_SOC_DAPM_HP ( " Headset Jack " , NULL ) ,
} ;
/* Corgi machine audio map (connections to the codec pins) */
2011-12-30 11:16:32 +08:00
static const struct snd_soc_dapm_route corgi_audio_map [ ] = {
2006-10-12 14:31:16 +02:00
/* headset Jack - in = micin, out = LHPOUT*/
{ " Headset Jack " , NULL , " LHPOUT " } ,
/* headphone connected to LHPOUT1, RHPOUT1 */
{ " Headphone Jack " , NULL , " LHPOUT " } ,
{ " Headphone Jack " , NULL , " RHPOUT " } ,
/* speaker connected to LOUT, ROUT */
{ " Ext Spk " , NULL , " ROUT " } ,
{ " Ext Spk " , NULL , " LOUT " } ,
/* mic is connected to MICIN (via right channel of headphone jack) */
{ " MICIN " , NULL , " Mic Jack " } ,
/* Same as the above but no mic bias for line signals */
{ " MICIN " , NULL , " Line Jack " } ,
} ;
2016-11-12 14:23:13 +01:00
static const char * const jack_function [ ] = { " Headphone " , " Mic " , " Line " ,
" Headset " , " Off " } ;
static const char * const spk_function [ ] = { " On " , " Off " } ;
2006-10-12 14:31:16 +02:00
static const struct soc_enum corgi_enum [ ] = {
SOC_ENUM_SINGLE_EXT ( 5 , jack_function ) ,
SOC_ENUM_SINGLE_EXT ( 2 , spk_function ) ,
} ;
static const struct snd_kcontrol_new wm8731_corgi_controls [ ] = {
SOC_ENUM_EXT ( " Jack Function " , corgi_enum [ 0 ] , corgi_get_jack ,
corgi_set_jack ) ,
SOC_ENUM_EXT ( " Speaker Function " , corgi_enum [ 1 ] , corgi_get_spk ,
corgi_set_spk ) ,
} ;
/* corgi digital audio interface glue - connects codec <--> CPU */
2019-06-06 13:10:56 +09:00
SND_SOC_DAILINK_DEFS ( wm8731 ,
DAILINK_COMP_ARRAY ( COMP_CPU ( " pxa2xx-i2s " ) ) ,
DAILINK_COMP_ARRAY ( COMP_CODEC ( " wm8731.0-001b " , " wm8731-hifi " ) ) ,
DAILINK_COMP_ARRAY ( COMP_PLATFORM ( " pxa-pcm-audio " ) ) ) ;
2006-10-12 14:31:16 +02:00
static struct snd_soc_dai_link corgi_dai = {
. name = " WM8731 " ,
. stream_name = " WM8731 " ,
2011-12-30 11:13:24 +08:00
. dai_fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF |
SND_SOC_DAIFMT_CBS_CFS ,
2007-02-02 17:22:20 +01:00
. ops = & corgi_ops ,
2019-06-06 13:10:56 +09:00
SND_SOC_DAILINK_REG ( wm8731 ) ,
2006-10-12 14:31:16 +02:00
} ;
/* corgi audio machine driver */
2011-12-30 11:18:13 +08:00
static struct snd_soc_card corgi = {
2006-10-12 14:31:16 +02:00
. name = " Corgi " ,
2011-12-22 09:44:43 +08:00
. owner = THIS_MODULE ,
2006-10-12 14:31:16 +02:00
. dai_link = & corgi_dai ,
. num_links = 1 ,
2011-12-30 11:16:32 +08:00
. controls = wm8731_corgi_controls ,
. num_controls = ARRAY_SIZE ( wm8731_corgi_controls ) ,
. dapm_widgets = wm8731_dapm_widgets ,
. num_dapm_widgets = ARRAY_SIZE ( wm8731_dapm_widgets ) ,
. dapm_routes = corgi_audio_map ,
. num_dapm_routes = ARRAY_SIZE ( corgi_audio_map ) ,
2015-01-09 22:03:26 +01:00
. fully_routed = true ,
2006-10-12 14:31:16 +02:00
} ;
2012-12-07 09:26:17 -05:00
static int corgi_probe ( struct platform_device * pdev )
2006-10-12 14:31:16 +02:00
{
2011-12-30 11:18:13 +08:00
struct snd_soc_card * card = & corgi ;
2006-10-12 14:31:16 +02:00
int ret ;
2011-12-30 11:18:13 +08:00
card - > dev = & pdev - > dev ;
2006-10-12 14:31:16 +02:00
2019-09-11 13:57:45 +02:00
gpiod_mute_l = devm_gpiod_get ( & pdev - > dev , " mute-l " , GPIOD_OUT_HIGH ) ;
if ( IS_ERR ( gpiod_mute_l ) )
return PTR_ERR ( gpiod_mute_l ) ;
gpiod_mute_r = devm_gpiod_get ( & pdev - > dev , " mute-r " , GPIOD_OUT_HIGH ) ;
if ( IS_ERR ( gpiod_mute_r ) )
return PTR_ERR ( gpiod_mute_r ) ;
gpiod_apm_on = devm_gpiod_get ( & pdev - > dev , " apm-on " , GPIOD_OUT_LOW ) ;
if ( IS_ERR ( gpiod_apm_on ) )
return PTR_ERR ( gpiod_apm_on ) ;
gpiod_mic_bias = devm_gpiod_get ( & pdev - > dev , " mic-bias " , GPIOD_OUT_LOW ) ;
if ( IS_ERR ( gpiod_mic_bias ) )
return PTR_ERR ( gpiod_mic_bias ) ;
2015-09-01 10:32:59 +08:00
ret = devm_snd_soc_register_card ( & pdev - > dev , card ) ;
2006-10-12 14:31:16 +02:00
if ( ret )
2011-12-30 11:18:13 +08:00
dev_err ( & pdev - > dev , " snd_soc_register_card() failed: %d \n " ,
ret ) ;
2006-10-12 14:31:16 +02:00
return ret ;
}
2011-12-30 11:18:13 +08:00
static struct platform_driver corgi_driver = {
. driver = {
. name = " corgi-audio " ,
2013-10-17 14:01:37 +04:00
. pm = & snd_soc_pm_ops ,
2011-12-30 11:18:13 +08:00
} ,
. probe = corgi_probe ,
} ;
module_platform_driver ( corgi_driver ) ;
2006-10-12 14:31:16 +02:00
/* Module information */
MODULE_AUTHOR ( " Richard Purdie " ) ;
MODULE_DESCRIPTION ( " ALSA SoC Corgi " ) ;
MODULE_LICENSE ( " GPL " ) ;
2011-12-30 11:18:13 +08:00
MODULE_ALIAS ( " platform:corgi-audio " ) ;