2006-10-12 16:33:45 +04:00
/*
* poodle . c - - SoC audio for Poodle
*
* Copyright 2005 Wolfson Microelectronics PLC .
* Copyright 2005 Openedhand Ltd .
*
2008-10-12 16:17:36 +04:00
* Authors : Liam Girdwood < lrg @ slimlogic . co . uk >
2006-10-12 16:33:45 +04:00
* Richard Purdie < richard @ openedhand . com >
*
* 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/moduleparam.h>
# include <linux/timer.h>
2009-02-18 15:34:53 +03:00
# include <linux/i2c.h>
2006-10-12 16:33:45 +04:00
# include <linux/interrupt.h>
# include <linux/platform_device.h>
# include <sound/core.h>
# include <sound/pcm.h>
# include <sound/soc.h>
# include <asm/mach-types.h>
# include <asm/hardware/locomo.h>
2008-08-05 19:14:15 +04:00
# include <mach/poodle.h>
# include <mach/audio.h>
2006-10-12 16:33:45 +04:00
# include "../codecs/wm8731.h"
2007-02-02 19:23:42 +03:00
# include "pxa2xx-i2s.h"
2006-10-12 16:33:45 +04:00
# define POODLE_HP 1
# define POODLE_HP_OFF 0
# define POODLE_SPK_ON 1
# define POODLE_SPK_OFF 0
/* audio clock in Hz - rounded from 12.235MHz */
# define POODLE_AUDIO_CLOCK 12288000
static int poodle_jack_func ;
static int poodle_spk_func ;
static void poodle_ext_control ( struct snd_soc_codec * codec )
{
2010-11-05 16:53:46 +03:00
struct snd_soc_dapm_context * dapm = & codec - > dapm ;
2006-10-12 16:33:45 +04:00
/* set up jack connection */
if ( poodle_jack_func = = POODLE_HP ) {
/* set = unmute headphone */
locomo_gpio_write ( & poodle_locomo_device . dev ,
POODLE_LOCOMO_GPIO_MUTE_L , 1 ) ;
locomo_gpio_write ( & poodle_locomo_device . dev ,
POODLE_LOCOMO_GPIO_MUTE_R , 1 ) ;
2010-11-05 16:53:46 +03:00
snd_soc_dapm_enable_pin ( dapm , " Headphone Jack " ) ;
2006-10-12 16:33:45 +04:00
} else {
locomo_gpio_write ( & poodle_locomo_device . dev ,
POODLE_LOCOMO_GPIO_MUTE_L , 0 ) ;
locomo_gpio_write ( & poodle_locomo_device . dev ,
POODLE_LOCOMO_GPIO_MUTE_R , 0 ) ;
2010-11-05 16:53:46 +03:00
snd_soc_dapm_disable_pin ( dapm , " Headphone Jack " ) ;
2006-10-12 16:33:45 +04:00
}
/* set the enpoints to their new connetion states */
2008-07-07 16:35:17 +04:00
if ( poodle_spk_func = = POODLE_SPK_ON )
2010-11-05 16:53:46 +03:00
snd_soc_dapm_enable_pin ( dapm , " Ext Spk " ) ;
2008-07-07 16:35:17 +04:00
else
2010-11-05 16:53:46 +03:00
snd_soc_dapm_disable_pin ( dapm , " Ext Spk " ) ;
2006-10-12 16:33:45 +04:00
/* signal a DAPM event */
2010-11-05 16:53:46 +03:00
snd_soc_dapm_sync ( dapm ) ;
2006-10-12 16:33:45 +04:00
}
static int poodle_startup ( struct snd_pcm_substream * substream )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
2010-03-17 23:15:21 +03:00
struct snd_soc_codec * codec = rtd - > codec ;
2006-10-12 16:33:45 +04:00
2010-11-05 20:50:48 +03:00
mutex_lock ( & codec - > mutex ) ;
2006-10-12 16:33:45 +04:00
/* check the jack status at stream startup */
poodle_ext_control ( codec ) ;
2010-11-05 20:50:48 +03:00
mutex_unlock ( & codec - > mutex ) ;
2006-10-12 16:33:45 +04:00
return 0 ;
}
/* we need to unmute the HP at shutdown as the mute burns power on poodle */
2008-07-29 14:42:22 +04:00
static void poodle_shutdown ( struct snd_pcm_substream * substream )
2006-10-12 16:33:45 +04:00
{
/* set = unmute headphone */
locomo_gpio_write ( & poodle_locomo_device . dev ,
POODLE_LOCOMO_GPIO_MUTE_L , 1 ) ;
locomo_gpio_write ( & poodle_locomo_device . dev ,
POODLE_LOCOMO_GPIO_MUTE_R , 1 ) ;
}
2007-02-02 19:23:42 +03:00
static int poodle_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 23:15:21 +03:00
struct snd_soc_dai * codec_dai = rtd - > codec_dai ;
struct snd_soc_dai * cpu_dai = rtd - > cpu_dai ;
2007-02-02 19:23:42 +03: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 codec DAI configuration */
2008-07-08 16:19:18 +04:00
ret = snd_soc_dai_set_fmt ( codec_dai , SND_SOC_DAIFMT_I2S |
2007-02-02 19:23:42 +03:00
SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_CBS_CFS ) ;
if ( ret < 0 )
return ret ;
/* set cpu DAI configuration */
2008-07-08 16:19:18 +04:00
ret = snd_soc_dai_set_fmt ( cpu_dai , SND_SOC_DAIFMT_I2S |
2007-02-02 19:23:42 +03: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 */
2010-08-18 02:40:24 +04:00
ret = snd_soc_dai_set_sysclk ( codec_dai , WM8731_SYSCLK_XTAL , clk ,
2007-02-02 19:23:42 +03:00
SND_SOC_CLOCK_IN ) ;
if ( ret < 0 )
return ret ;
/* set the I2S system clock as input (unused) */
2008-07-08 16:19:18 +04:00
ret = snd_soc_dai_set_sysclk ( cpu_dai , PXA2XX_I2S_SYSCLK , 0 ,
2007-02-02 19:23:42 +03:00
SND_SOC_CLOCK_IN ) ;
if ( ret < 0 )
return ret ;
return 0 ;
}
2006-10-12 16:33:45 +04:00
static struct snd_soc_ops poodle_ops = {
. startup = poodle_startup ,
2007-02-02 19:23:42 +03:00
. hw_params = poodle_hw_params ,
2006-10-12 16:33:45 +04:00
. shutdown = poodle_shutdown ,
} ;
static int poodle_get_jack ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
ucontrol - > value . integer . value [ 0 ] = poodle_jack_func ;
return 0 ;
}
static int poodle_set_jack ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
struct snd_soc_codec * codec = snd_kcontrol_chip ( kcontrol ) ;
if ( poodle_jack_func = = ucontrol - > value . integer . value [ 0 ] )
return 0 ;
poodle_jack_func = ucontrol - > value . integer . value [ 0 ] ;
poodle_ext_control ( codec ) ;
return 1 ;
}
static int poodle_get_spk ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
ucontrol - > value . integer . value [ 0 ] = poodle_spk_func ;
return 0 ;
}
static int poodle_set_spk ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
struct snd_soc_codec * codec = snd_kcontrol_chip ( kcontrol ) ;
if ( poodle_spk_func = = ucontrol - > value . integer . value [ 0 ] )
return 0 ;
poodle_spk_func = ucontrol - > value . integer . value [ 0 ] ;
poodle_ext_control ( codec ) ;
return 1 ;
}
2008-02-28 14:34:48 +03:00
static int poodle_amp_event ( struct snd_soc_dapm_widget * w ,
struct snd_kcontrol * k , int event )
2006-10-12 16:33:45 +04:00
{
if ( SND_SOC_DAPM_EVENT_ON ( event ) )
locomo_gpio_write ( & poodle_locomo_device . dev ,
POODLE_LOCOMO_GPIO_AMP_ON , 0 ) ;
else
locomo_gpio_write ( & poodle_locomo_device . dev ,
POODLE_LOCOMO_GPIO_AMP_ON , 1 ) ;
return 0 ;
}
/* poodle machine dapm widgets */
static const struct snd_soc_dapm_widget wm8731_dapm_widgets [ ] = {
SND_SOC_DAPM_HP ( " Headphone Jack " , NULL ) ,
SND_SOC_DAPM_SPK ( " Ext Spk " , poodle_amp_event ) ,
} ;
2008-05-13 16:55:48 +04:00
/* Corgi machine connections to the codec pins */
static const struct snd_soc_dapm_route audio_map [ ] = {
2006-10-12 16:33:45 +04:00
/* 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 " } ,
} ;
static const char * jack_function [ ] = { " Off " , " Headphone " } ;
static const char * spk_function [ ] = { " Off " , " On " } ;
static const struct soc_enum poodle_enum [ ] = {
SOC_ENUM_SINGLE_EXT ( 2 , jack_function ) ,
SOC_ENUM_SINGLE_EXT ( 2 , spk_function ) ,
} ;
2008-07-29 14:42:22 +04:00
static const struct snd_kcontrol_new wm8731_poodle_controls [ ] = {
2006-10-12 16:33:45 +04:00
SOC_ENUM_EXT ( " Jack Function " , poodle_enum [ 0 ] , poodle_get_jack ,
poodle_set_jack ) ,
SOC_ENUM_EXT ( " Speaker Function " , poodle_enum [ 1 ] , poodle_get_spk ,
poodle_set_spk ) ,
} ;
/*
* Logic for a wm8731 as connected on a Sharp SL - C7x0 Device
*/
2010-03-17 23:15:21 +03:00
static int poodle_wm8731_init ( struct snd_soc_pcm_runtime * rtd )
2006-10-12 16:33:45 +04:00
{
2010-03-17 23:15:21 +03:00
struct snd_soc_codec * codec = rtd - > codec ;
2010-11-05 16:53:46 +03:00
struct snd_soc_dapm_context * dapm = & codec - > dapm ;
2009-03-12 13:07:54 +03:00
int err ;
2006-10-12 16:33:45 +04:00
2010-11-05 16:53:46 +03:00
snd_soc_dapm_nc_pin ( dapm , " LLINEIN " ) ;
snd_soc_dapm_nc_pin ( dapm , " RLINEIN " ) ;
snd_soc_dapm_enable_pin ( dapm , " MICIN " ) ;
2006-10-12 16:33:45 +04:00
/* Add poodle specific controls */
2009-03-12 13:07:54 +03:00
err = snd_soc_add_controls ( codec , wm8731_poodle_controls ,
ARRAY_SIZE ( wm8731_poodle_controls ) ) ;
if ( err < 0 )
return err ;
2006-10-12 16:33:45 +04:00
/* Add poodle specific widgets */
2010-11-05 16:53:46 +03:00
snd_soc_dapm_new_controls ( dapm , wm8731_dapm_widgets ,
2008-05-13 16:55:48 +04:00
ARRAY_SIZE ( wm8731_dapm_widgets ) ) ;
2006-10-12 16:33:45 +04:00
/* Set up poodle specific audio path audio_map */
2010-11-05 16:53:46 +03:00
snd_soc_dapm_add_routes ( dapm , audio_map , ARRAY_SIZE ( audio_map ) ) ;
2006-10-12 16:33:45 +04:00
2010-11-05 16:53:46 +03:00
snd_soc_dapm_sync ( dapm ) ;
2006-10-12 16:33:45 +04:00
return 0 ;
}
/* poodle digital audio interface glue - connects codec <--> CPU */
static struct snd_soc_dai_link poodle_dai = {
. name = " WM8731 " ,
. stream_name = " WM8731 " ,
2010-08-20 20:18:43 +04:00
. cpu_dai_name = " pxa2xx-i2s " ,
2010-08-18 02:35:31 +04:00
. codec_dai_name = " wm8731-hifi " ,
2010-03-17 23:15:21 +03:00
. platform_name = " pxa-pcm-audio " ,
2011-03-27 17:35:15 +04:00
. codec_name = " wm8731.0-001b " ,
2006-10-12 16:33:45 +04:00
. init = poodle_wm8731_init ,
2007-02-02 19:23:42 +03:00
. ops = & poodle_ops ,
2006-10-12 16:33:45 +04:00
} ;
/* poodle audio machine driver */
2008-11-18 23:50:34 +03:00
static struct snd_soc_card snd_soc_poodle = {
2006-10-12 16:33:45 +04:00
. name = " Poodle " ,
. dai_link = & poodle_dai ,
. num_links = 1 ,
2010-03-17 23:15:21 +03:00
. owner = THIS_MODULE ,
2006-10-12 16:33:45 +04:00
} ;
static struct platform_device * poodle_snd_device ;
static int __init poodle_init ( void )
{
int ret ;
if ( ! machine_is_poodle ( ) )
return - ENODEV ;
locomo_gpio_set_dir ( & poodle_locomo_device . dev ,
POODLE_LOCOMO_GPIO_AMP_ON , 0 ) ;
/* should we mute HP at startup - burning power ?*/
locomo_gpio_set_dir ( & poodle_locomo_device . dev ,
POODLE_LOCOMO_GPIO_MUTE_L , 0 ) ;
locomo_gpio_set_dir ( & poodle_locomo_device . dev ,
POODLE_LOCOMO_GPIO_MUTE_R , 0 ) ;
poodle_snd_device = platform_device_alloc ( " soc-audio " , - 1 ) ;
if ( ! poodle_snd_device )
return - ENOMEM ;
2010-03-17 23:15:21 +03:00
platform_set_drvdata ( poodle_snd_device , & snd_soc_poodle ) ;
2006-10-12 16:33:45 +04:00
ret = platform_device_add ( poodle_snd_device ) ;
if ( ret )
platform_device_put ( poodle_snd_device ) ;
return ret ;
}
static void __exit poodle_exit ( void )
{
platform_device_unregister ( poodle_snd_device ) ;
}
module_init ( poodle_init ) ;
module_exit ( poodle_exit ) ;
/* Module information */
MODULE_AUTHOR ( " Richard Purdie " ) ;
MODULE_DESCRIPTION ( " ALSA SoC Poodle " ) ;
MODULE_LICENSE ( " GPL " ) ;