2007-05-14 11:03:52 +02:00
/*
2011-03-07 08:04:58 +01:00
* neo1973_wm8753 . c - - SoC audio for Openmoko Neo1973 and Freerunner devices
2007-05-14 11:03:52 +02:00
*
2011-03-07 08:04:58 +01:00
* Copyright 2007 Openmoko Inc
* Author : Graeme Gregory < graeme @ openmoko . org >
2007-05-14 11:03:52 +02:00
* Copyright 2007 Wolfson Microelectronics PLC .
* Author : Graeme Gregory
* graeme . gregory @ wolfsonmicro . com or linux @ wolfsonmicro . com
2011-03-07 08:04:58 +01:00
* Copyright 2009 Wolfson Microelectronics
2007-05-14 11:03:52 +02:00
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation ; either version 2 of the License , or ( at your
* option ) any later version .
*/
# include <linux/module.h>
# include <linux/platform_device.h>
2011-03-07 08:04:58 +01:00
# include <linux/gpio.h>
2007-05-14 11:03:52 +02:00
# include <sound/soc.h>
2008-10-08 13:02:20 +01:00
# include <asm/mach-types.h>
2013-04-11 19:08:42 +02:00
# include "regs-iis.h"
2007-12-19 15:37:49 +01:00
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 )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
2010-03-17 20:15:21 +00:00
struct snd_soc_dai * codec_dai = rtd - > codec_dai ;
struct snd_soc_dai * cpu_dai = rtd - > cpu_dai ;
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 codec DAI configuration */
2008-07-08 13:19:18 +01:00
ret = snd_soc_dai_set_fmt ( codec_dai ,
2007-05-14 11:03:52 +02:00
SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF |
SND_SOC_DAIFMT_CBM_CFM ) ;
if ( ret < 0 )
return ret ;
/* set cpu DAI configuration */
2008-07-08 13:19:18 +01:00
ret = snd_soc_dai_set_fmt ( cpu_dai ,
2007-05-14 11:03:52 +02:00
SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF |
SND_SOC_DAIFMT_CBM_CFM ) ;
if ( ret < 0 )
return ret ;
/* 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 )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
2010-03-17 20:15:21 +00:00
struct snd_soc_dai * codec_dai = rtd - > codec_dai ;
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 .
*/
static struct snd_soc_ops neo1973_hifi_ops = {
. 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 )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
2010-03-17 20:15:21 +00:00
struct snd_soc_dai * codec_dai = rtd - > codec_dai ;
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 */
/* todo: gg check mode (DSP_B) against CSR datasheet */
/* set codec DAI configuration */
2008-07-08 13:19:18 +01:00
ret = snd_soc_dai_set_fmt ( codec_dai , SND_SOC_DAIFMT_DSP_B |
2007-05-14 11:03:52 +02:00
SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_CBS_CFS ) ;
if ( ret < 0 )
return ret ;
/* 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 )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
2010-03-17 20:15:21 +00:00
struct snd_soc_dai * codec_dai = rtd - > codec_dai ;
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
}
static struct snd_soc_ops neo1973_voice_ops = {
. hw_params = neo1973_voice_hw_params ,
. hw_free = neo1973_voice_hw_free ,
} ;
2011-03-07 08:04:58 +01:00
/* Shared routes and controls */
static const struct snd_soc_dapm_widget neo1973_wm8753_dapm_widgets [ ] = {
2007-05-14 11:03:52 +02:00
SND_SOC_DAPM_LINE ( " GSM Line Out " , NULL ) ,
SND_SOC_DAPM_LINE ( " GSM Line In " , NULL ) ,
SND_SOC_DAPM_MIC ( " Headset Mic " , NULL ) ,
2011-03-07 08:04:58 +01:00
SND_SOC_DAPM_MIC ( " Handset Mic " , NULL ) ,
2007-05-14 11:03:52 +02:00
} ;
2011-03-07 08:04:55 +01:00
static const struct snd_soc_dapm_route neo1973_wm8753_routes [ ] = {
2007-05-14 11:03:52 +02:00
/* 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 " } ,
2011-03-07 08:04:58 +01:00
{ " Mic Bias " , NULL , " Handset Mic " } ,
2007-05-14 11:03:52 +02:00
/* Connect the ALC pins */
{ " ACIN " , NULL , " ACOP " } ,
} ;
2011-03-07 08:04:55 +01:00
static const struct snd_kcontrol_new neo1973_wm8753_controls [ ] = {
2011-03-07 08:04:54 +01:00
SOC_DAPM_PIN_SWITCH ( " GSM Line Out " ) ,
SOC_DAPM_PIN_SWITCH ( " GSM Line In " ) ,
SOC_DAPM_PIN_SWITCH ( " Headset Mic " ) ,
2011-03-07 08:04:58 +01:00
SOC_DAPM_PIN_SWITCH ( " Handset Mic " ) ,
2007-05-14 11:03:52 +02:00
} ;
2011-03-30 22:57:33 -03:00
/* GTA02 specific routes and controls */
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 ] ;
2013-01-02 09:57:59 -08:00
gpio_set_value ( S3C2410_GPJ ( 2 ) , ! 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 )
{
2013-01-02 09:57:59 -08:00
gpio_set_value ( S3C2410_GPJ ( 1 ) , SND_SOC_DAPM_EVENT_OFF ( event ) ) ;
2011-03-07 08:04:58 +01:00
return 0 ;
}
static const struct snd_soc_dapm_route neo1973_gta02_routes [ ] = {
/* Connections to the amp */
{ " Stereo Out " , NULL , " LOUT1 " } ,
{ " Stereo Out " , NULL , " ROUT1 " } ,
/* Call Speaker */
{ " Handset Spk " , NULL , " LOUT2 " } ,
{ " Handset Spk " , NULL , " ROUT2 " } ,
} ;
static const struct snd_kcontrol_new neo1973_gta02_wm8753_controls [ ] = {
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 ) ,
} ;
static const struct snd_soc_dapm_widget neo1973_gta02_wm8753_dapm_widgets [ ] = {
SND_SOC_DAPM_SPK ( " Handset Spk " , NULL ) ,
SND_SOC_DAPM_SPK ( " Stereo Out " , lm4853_event ) ,
} ;
static int neo1973_gta02_wm8753_init ( struct snd_soc_codec * codec )
{
struct snd_soc_dapm_context * dapm = & codec - > dapm ;
int ret ;
ret = snd_soc_dapm_new_controls ( dapm , neo1973_gta02_wm8753_dapm_widgets ,
ARRAY_SIZE ( neo1973_gta02_wm8753_dapm_widgets ) ) ;
if ( ret )
return ret ;
ret = snd_soc_dapm_add_routes ( dapm , neo1973_gta02_routes ,
ARRAY_SIZE ( neo1973_gta02_routes ) ) ;
if ( ret )
return ret ;
2012-02-03 17:43:09 +00:00
ret = snd_soc_add_card_controls ( codec - > card , neo1973_gta02_wm8753_controls ,
2011-03-07 08:04:58 +01:00
ARRAY_SIZE ( neo1973_gta02_wm8753_controls ) ) ;
if ( ret )
return ret ;
snd_soc_dapm_disable_pin ( dapm , " Stereo Out " ) ;
snd_soc_dapm_disable_pin ( dapm , " Handset Spk " ) ;
snd_soc_dapm_ignore_suspend ( dapm , " Stereo Out " ) ;
snd_soc_dapm_ignore_suspend ( dapm , " Handset Spk " ) ;
return 0 ;
}
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
{
2010-03-17 20:15:21 +00:00
struct snd_soc_codec * codec = rtd - > codec ;
2010-11-05 15:53:46 +02:00
struct snd_soc_dapm_context * dapm = & codec - > dapm ;
2011-03-07 08:04:58 +01:00
int ret ;
2008-05-05 14:16:12 +02:00
2007-05-14 11:03:52 +02:00
/* set up NC codec pins */
2010-11-05 15:53:46 +02:00
snd_soc_dapm_nc_pin ( dapm , " OUT3 " ) ;
snd_soc_dapm_nc_pin ( dapm , " OUT4 " ) ;
snd_soc_dapm_nc_pin ( dapm , " LINE1 " ) ;
snd_soc_dapm_nc_pin ( dapm , " LINE2 " ) ;
2007-05-14 11:03:52 +02:00
/* Add neo1973 specific widgets */
2011-03-07 08:04:58 +01:00
ret = snd_soc_dapm_new_controls ( dapm , neo1973_wm8753_dapm_widgets ,
ARRAY_SIZE ( neo1973_wm8753_dapm_widgets ) ) ;
if ( ret )
return ret ;
2008-10-01 18:17:12 +01:00
2007-05-14 11:03:52 +02:00
/* add neo1973 specific controls */
2012-02-03 17:43:09 +00:00
ret = snd_soc_add_card_controls ( rtd - > card , neo1973_wm8753_controls ,
2011-03-07 08:04:58 +01:00
ARRAY_SIZE ( neo1973_wm8753_controls ) ) ;
if ( ret )
return ret ;
2007-05-14 11:03:52 +02:00
2008-05-13 14:58:03 +02:00
/* set up neo1973 specific audio routes */
2011-03-07 08:04:58 +01:00
ret = snd_soc_dapm_add_routes ( dapm , neo1973_wm8753_routes ,
2011-03-07 08:04:55 +01:00
ARRAY_SIZE ( neo1973_wm8753_routes ) ) ;
2011-03-07 08:04:58 +01:00
if ( ret )
return ret ;
/* set endpoints to default off mode */
snd_soc_dapm_disable_pin ( dapm , " GSM Line Out " ) ;
snd_soc_dapm_disable_pin ( dapm , " GSM Line In " ) ;
snd_soc_dapm_disable_pin ( dapm , " Headset Mic " ) ;
snd_soc_dapm_disable_pin ( dapm , " Handset Mic " ) ;
/* allow audio paths from the GSM modem to run during suspend */
snd_soc_dapm_ignore_suspend ( dapm , " GSM Line Out " ) ;
snd_soc_dapm_ignore_suspend ( dapm , " GSM Line In " ) ;
snd_soc_dapm_ignore_suspend ( dapm , " Headset Mic " ) ;
snd_soc_dapm_ignore_suspend ( dapm , " Handset Mic " ) ;
if ( machine_is_neo1973_gta02 ( ) ) {
ret = neo1973_gta02_wm8753_init ( codec ) ;
if ( ret )
return ret ;
}
2007-05-14 11:03:52 +02:00
return 0 ;
}
static struct snd_soc_dai_link neo1973_dai [ ] = {
{ /* Hifi Playback - for similatious use with voice below */
. name = " WM8753 " ,
. stream_name = " WM8753 HiFi " ,
2012-12-07 13:59:21 +05:30
. platform_name = " s3c24xx-iis " ,
2011-01-24 22:12:42 +01:00
. cpu_dai_name = " s3c24xx-iis " ,
2010-03-17 20:15:21 +00:00
. codec_dai_name = " wm8753-hifi " ,
2012-02-26 19:21:54 +01:00
. codec_name = " wm8753.0-001a " ,
2007-05-14 11:03:52 +02:00
. init = neo1973_wm8753_init ,
. ops = & neo1973_hifi_ops ,
} ,
{ /* Voice via BT */
. name = " Bluetooth " ,
. stream_name = " Voice " ,
2011-03-07 08:04:59 +01:00
. cpu_dai_name = " dfbmcs320-pcm " ,
2010-03-17 20:15:21 +00:00
. codec_dai_name = " wm8753-voice " ,
2012-02-26 19:21:54 +01:00
. codec_name = " wm8753.0-001a " ,
2007-05-14 11:03:52 +02:00
. ops = & neo1973_voice_ops ,
} ,
} ;
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
{
. name = " dfbmcs320 " ,
. codec_name = " dfbmcs320.0 " ,
} ,
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 [ ] = {
{
. dev_name = " lm4857.0-007c " ,
. name_prefix = " Amp " ,
} ,
2008-09-01 17:46:57 +02:00
} ;
2011-03-07 08:04:58 +01:00
static const struct gpio neo1973_gta02_gpios [ ] = {
2013-01-02 09:57:59 -08:00
{ S3C2410_GPJ ( 2 ) , GPIOF_OUT_INIT_HIGH , " GTA02_HP_IN " } ,
{ S3C2410_GPJ ( 1 ) , GPIOF_OUT_INIT_HIGH , " GTA02_AMP_SHUT " } ,
2011-03-07 08:04:58 +01:00
} ;
2011-03-07 08:04:55 +01:00
static struct snd_soc_card neo1973 = {
. name = " neo1973 " ,
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 ) ,
2007-05-14 11:03:52 +02:00
} ;
static struct platform_device * neo1973_snd_device ;
static int __init neo1973_init ( void )
{
int ret ;
2012-01-30 00:31:47 +01:00
if ( ! machine_is_neo1973_gta02 ( ) )
2008-10-08 13:02:20 +01:00
return - ENODEV ;
2011-03-07 08:04:58 +01:00
if ( machine_is_neo1973_gta02 ( ) ) {
neo1973 . name = " neo1973gta02 " ;
2011-03-07 08:04:59 +01:00
neo1973 . num_aux_devs = 1 ;
2011-03-07 08:04:58 +01:00
ret = gpio_request_array ( neo1973_gta02_gpios ,
ARRAY_SIZE ( neo1973_gta02_gpios ) ) ;
if ( ret )
return ret ;
2008-10-08 13:02:20 +01:00
}
2007-05-14 11:03:52 +02:00
neo1973_snd_device = platform_device_alloc ( " soc-audio " , - 1 ) ;
2011-03-07 08:04:58 +01:00
if ( ! neo1973_snd_device ) {
ret = - ENOMEM ;
goto err_gpio_free ;
}
2010-03-17 20:15:21 +00:00
platform_set_drvdata ( neo1973_snd_device , & neo1973 ) ;
2007-05-14 11:03:52 +02:00
ret = platform_device_add ( neo1973_snd_device ) ;
2011-03-07 08:04:58 +01:00
if ( ret )
2011-03-07 08:04:59 +01:00
goto err_put_device ;
2007-05-14 11:03:52 +02:00
2011-03-07 08:04:58 +01:00
return 0 ;
2007-05-14 11:03:52 +02:00
2011-03-07 08:04:58 +01:00
err_put_device :
platform_device_put ( neo1973_snd_device ) ;
err_gpio_free :
if ( machine_is_neo1973_gta02 ( ) ) {
gpio_free_array ( neo1973_gta02_gpios ,
ARRAY_SIZE ( neo1973_gta02_gpios ) ) ;
}
2007-05-14 11:03:52 +02:00
return ret ;
}
2011-03-07 08:04:58 +01:00
module_init ( neo1973_init ) ;
2007-05-14 11:03:52 +02:00
static void __exit neo1973_exit ( void )
{
platform_device_unregister ( neo1973_snd_device ) ;
2011-03-07 08:04:58 +01:00
if ( machine_is_neo1973_gta02 ( ) ) {
gpio_free_array ( neo1973_gta02_gpios ,
ARRAY_SIZE ( neo1973_gta02_gpios ) ) ;
}
}
2007-05-14 11:03:52 +02:00
module_exit ( neo1973_exit ) ;
/* 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 " ) ;