2006-10-06 18:31:09 +02:00
/*
* soc - core . c - - ALSA SoC Audio Layer
*
* Copyright 2005 Wolfson Microelectronics PLC .
2006-12-18 14:39:02 +01:00
* Copyright 2005 Openedhand Ltd .
*
2008-10-12 13:17:36 +01:00
* Author : Liam Girdwood < lrg @ slimlogic . co . uk >
2006-12-18 14:39:02 +01:00
* with code , comments and ideas from : -
* Richard Purdie < richard @ openedhand . com >
2006-10-06 18:31:09 +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 .
*
* TODO :
* o Add hw rules to enforce rates , etc .
* o More testing with other codecs / machines .
* o Add more codecs and platforms to ensure good API coverage .
* o Support TDM on PCM and I2S
*/
# include <linux/module.h>
# include <linux/moduleparam.h>
# include <linux/init.h>
# include <linux/delay.h>
# include <linux/pm.h>
# include <linux/bitops.h>
2008-10-13 17:42:14 -07:00
# include <linux/debugfs.h>
2006-10-06 18:31:09 +02:00
# include <linux/platform_device.h>
# include <sound/core.h>
# include <sound/pcm.h>
# include <sound/pcm_params.h>
# include <sound/soc.h>
# include <sound/soc-dapm.h>
# include <sound/initval.h>
static DEFINE_MUTEX ( pcm_mutex ) ;
static DEFINE_MUTEX ( io_mutex ) ;
static DECLARE_WAIT_QUEUE_HEAD ( soc_pm_waitq ) ;
/*
* This is a timeout to do a DAPM powerdown after a stream is closed ( ) .
* It can be used to eliminate pops between different playback streams , e . g .
* between two audio tracks .
*/
static int pmdown_time = 5000 ;
module_param ( pmdown_time , int , 0 ) ;
MODULE_PARM_DESC ( pmdown_time , " DAPM stream powerdown time (msecs) " ) ;
2007-01-31 14:14:57 +01:00
/*
* This function forces any delayed work to be queued and run .
*/
static int run_delayed_work ( struct delayed_work * dwork )
{
int ret ;
/* cancel any work waiting to be queued. */
ret = cancel_delayed_work ( dwork ) ;
/* if there was any work waiting then we run it now and
* wait for it ' s completion */
if ( ret ) {
schedule_delayed_work ( dwork , 0 ) ;
flush_scheduled_work ( ) ;
}
return ret ;
}
2006-10-06 18:31:09 +02:00
# ifdef CONFIG_SND_SOC_AC97_BUS
/* unregister ac97 codec */
static int soc_ac97_dev_unregister ( struct snd_soc_codec * codec )
{
if ( codec - > ac97 - > dev . bus )
device_unregister ( & codec - > ac97 - > dev ) ;
return 0 ;
}
/* stop no dev release warning */
static void soc_ac97_device_release ( struct device * dev ) { }
/* register ac97 codec to bus */
static int soc_ac97_dev_register ( struct snd_soc_codec * codec )
{
int err ;
codec - > ac97 - > dev . bus = & ac97_bus_type ;
codec - > ac97 - > dev . parent = NULL ;
codec - > ac97 - > dev . release = soc_ac97_device_release ;
2008-11-02 03:50:35 +01:00
dev_set_name ( & codec - > ac97 - > dev , " %d-%d:%s " ,
codec - > card - > number , 0 , codec - > name ) ;
2006-10-06 18:31:09 +02:00
err = device_register ( & codec - > ac97 - > dev ) ;
if ( err < 0 ) {
snd_printk ( KERN_ERR " Can't register ac97 bus \n " ) ;
codec - > ac97 - > dev . bus = NULL ;
return err ;
}
return 0 ;
}
# endif
/*
* Called by ALSA when a PCM substream is opened , the runtime - > hw record is
* then initialized and any private data can be allocated . This also calls
* startup for the cpu DAI , platform , machine and codec DAI .
*/
static int soc_pcm_open ( struct snd_pcm_substream * substream )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
struct snd_soc_device * socdev = rtd - > socdev ;
2008-12-02 16:01:14 +00:00
struct snd_soc_card * card = socdev - > card ;
2006-10-06 18:31:09 +02:00
struct snd_pcm_runtime * runtime = substream - > runtime ;
2007-02-02 17:13:49 +01:00
struct snd_soc_dai_link * machine = rtd - > dai ;
2008-12-02 16:01:14 +00:00
struct snd_soc_platform * platform = card - > platform ;
2008-07-07 16:07:17 +01:00
struct snd_soc_dai * cpu_dai = machine - > cpu_dai ;
struct snd_soc_dai * codec_dai = machine - > codec_dai ;
2006-10-06 18:31:09 +02:00
int ret = 0 ;
mutex_lock ( & pcm_mutex ) ;
/* startup the audio subsystem */
2007-02-02 17:13:49 +01:00
if ( cpu_dai - > ops . startup ) {
2008-11-18 22:11:38 +00:00
ret = cpu_dai - > ops . startup ( substream , cpu_dai ) ;
2006-10-06 18:31:09 +02:00
if ( ret < 0 ) {
printk ( KERN_ERR " asoc: can't open interface %s \n " ,
2007-02-02 17:13:49 +01:00
cpu_dai - > name ) ;
2006-10-06 18:31:09 +02:00
goto out ;
}
}
if ( platform - > pcm_ops - > open ) {
ret = platform - > pcm_ops - > open ( substream ) ;
if ( ret < 0 ) {
printk ( KERN_ERR " asoc: can't open platform %s \n " , platform - > name ) ;
goto platform_err ;
}
}
2007-02-02 17:13:49 +01:00
if ( codec_dai - > ops . startup ) {
2008-11-18 22:11:38 +00:00
ret = codec_dai - > ops . startup ( substream , codec_dai ) ;
2006-10-06 18:31:09 +02:00
if ( ret < 0 ) {
2007-02-02 17:13:49 +01:00
printk ( KERN_ERR " asoc: can't open codec %s \n " ,
codec_dai - > name ) ;
goto codec_dai_err ;
2006-10-06 18:31:09 +02:00
}
}
2007-02-02 17:13:49 +01:00
if ( machine - > ops & & machine - > ops - > startup ) {
ret = machine - > ops - > startup ( substream ) ;
2006-10-06 18:31:09 +02:00
if ( ret < 0 ) {
2007-02-02 17:13:49 +01:00
printk ( KERN_ERR " asoc: %s startup failed \n " , machine - > name ) ;
goto machine_err ;
2006-10-06 18:31:09 +02:00
}
}
/* Check that the codec and cpu DAI's are compatible */
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK ) {
runtime - > hw . rate_min =
2008-05-19 12:32:25 +02:00
max ( codec_dai - > playback . rate_min ,
cpu_dai - > playback . rate_min ) ;
2006-10-06 18:31:09 +02:00
runtime - > hw . rate_max =
2008-05-19 12:32:25 +02:00
min ( codec_dai - > playback . rate_max ,
cpu_dai - > playback . rate_max ) ;
2006-10-06 18:31:09 +02:00
runtime - > hw . channels_min =
2007-02-02 17:13:49 +01:00
max ( codec_dai - > playback . channels_min ,
cpu_dai - > playback . channels_min ) ;
2006-10-06 18:31:09 +02:00
runtime - > hw . channels_max =
2007-02-02 17:13:49 +01:00
min ( codec_dai - > playback . channels_max ,
cpu_dai - > playback . channels_max ) ;
runtime - > hw . formats =
codec_dai - > playback . formats & cpu_dai - > playback . formats ;
runtime - > hw . rates =
codec_dai - > playback . rates & cpu_dai - > playback . rates ;
2006-10-06 18:31:09 +02:00
} else {
runtime - > hw . rate_min =
2008-05-19 12:32:25 +02:00
max ( codec_dai - > capture . rate_min ,
cpu_dai - > capture . rate_min ) ;
2006-10-06 18:31:09 +02:00
runtime - > hw . rate_max =
2008-05-19 12:32:25 +02:00
min ( codec_dai - > capture . rate_max ,
cpu_dai - > capture . rate_max ) ;
2006-10-06 18:31:09 +02:00
runtime - > hw . channels_min =
2007-02-02 17:13:49 +01:00
max ( codec_dai - > capture . channels_min ,
cpu_dai - > capture . channels_min ) ;
2006-10-06 18:31:09 +02:00
runtime - > hw . channels_max =
2007-02-02 17:13:49 +01:00
min ( codec_dai - > capture . channels_max ,
cpu_dai - > capture . channels_max ) ;
runtime - > hw . formats =
codec_dai - > capture . formats & cpu_dai - > capture . formats ;
runtime - > hw . rates =
codec_dai - > capture . rates & cpu_dai - > capture . rates ;
2006-10-06 18:31:09 +02:00
}
snd_pcm_limit_hw_rates ( runtime ) ;
if ( ! runtime - > hw . rates ) {
printk ( KERN_ERR " asoc: %s <-> %s No matching rates \n " ,
2007-02-02 17:13:49 +01:00
codec_dai - > name , cpu_dai - > name ) ;
goto machine_err ;
2006-10-06 18:31:09 +02:00
}
if ( ! runtime - > hw . formats ) {
printk ( KERN_ERR " asoc: %s <-> %s No matching formats \n " ,
2007-02-02 17:13:49 +01:00
codec_dai - > name , cpu_dai - > name ) ;
goto machine_err ;
2006-10-06 18:31:09 +02:00
}
if ( ! runtime - > hw . channels_min | | ! runtime - > hw . channels_max ) {
printk ( KERN_ERR " asoc: %s <-> %s No matching channels \n " ,
2007-02-02 17:13:49 +01:00
codec_dai - > name , cpu_dai - > name ) ;
goto machine_err ;
2006-10-06 18:31:09 +02:00
}
2008-10-21 21:45:08 +01:00
pr_debug ( " asoc: %s <-> %s info: \n " , codec_dai - > name , cpu_dai - > name ) ;
pr_debug ( " asoc: rate mask 0x%x \n " , runtime - > hw . rates ) ;
pr_debug ( " asoc: min ch %d max ch %d \n " , runtime - > hw . channels_min ,
runtime - > hw . channels_max ) ;
pr_debug ( " asoc: min rate %d max rate %d \n " , runtime - > hw . rate_min ,
runtime - > hw . rate_max ) ;
2006-10-06 18:31:09 +02:00
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK )
2007-02-02 17:13:49 +01:00
cpu_dai - > playback . active = codec_dai - > playback . active = 1 ;
2006-10-06 18:31:09 +02:00
else
2007-02-02 17:13:49 +01:00
cpu_dai - > capture . active = codec_dai - > capture . active = 1 ;
cpu_dai - > active = codec_dai - > active = 1 ;
cpu_dai - > runtime = runtime ;
2006-10-06 18:31:09 +02:00
socdev - > codec - > active + + ;
mutex_unlock ( & pcm_mutex ) ;
return 0 ;
2007-02-02 17:13:49 +01:00
machine_err :
2006-10-06 18:31:09 +02:00
if ( machine - > ops & & machine - > ops - > shutdown )
machine - > ops - > shutdown ( substream ) ;
2007-02-02 17:13:49 +01:00
codec_dai_err :
2006-10-06 18:31:09 +02:00
if ( platform - > pcm_ops - > close )
platform - > pcm_ops - > close ( substream ) ;
platform_err :
2007-02-02 17:13:49 +01:00
if ( cpu_dai - > ops . shutdown )
2008-11-18 22:11:38 +00:00
cpu_dai - > ops . shutdown ( substream , cpu_dai ) ;
2006-10-06 18:31:09 +02:00
out :
mutex_unlock ( & pcm_mutex ) ;
return ret ;
}
/*
2007-10-19 23:10:43 +02:00
* Power down the audio subsystem pmdown_time msecs after close is called .
2006-10-06 18:31:09 +02:00
* This is to ensure there are no pops or clicks in between any music tracks
* due to DAPM power cycling .
*/
2006-12-15 09:30:07 +01:00
static void close_delayed_work ( struct work_struct * work )
2006-10-06 18:31:09 +02:00
{
2008-12-02 15:08:03 +00:00
struct snd_soc_card * card = container_of ( work , struct snd_soc_card ,
delayed_work . work ) ;
struct snd_soc_device * socdev = card - > socdev ;
2006-10-06 18:31:09 +02:00
struct snd_soc_codec * codec = socdev - > codec ;
2008-07-07 16:07:17 +01:00
struct snd_soc_dai * codec_dai ;
2006-10-06 18:31:09 +02:00
int i ;
mutex_lock ( & pcm_mutex ) ;
2008-05-19 12:32:25 +02:00
for ( i = 0 ; i < codec - > num_dai ; i + + ) {
2006-10-06 18:31:09 +02:00
codec_dai = & codec - > dai [ i ] ;
2008-10-21 21:45:08 +01:00
pr_debug ( " pop wq checking: %s status: %s waiting: %s \n " ,
codec_dai - > playback . stream_name ,
codec_dai - > playback . active ? " active " : " inactive " ,
codec_dai - > pop_wait ? " yes " : " no " ) ;
2006-10-06 18:31:09 +02:00
/* are we waiting on this codec DAI stream */
if ( codec_dai - > pop_wait = = 1 ) {
2008-05-19 12:31:28 +02:00
/* Reduce power if no longer active */
2008-01-10 14:38:24 +01:00
if ( codec - > active = = 0 ) {
2008-10-21 21:45:08 +01:00
pr_debug ( " pop wq D1 %s %s \n " , codec - > name ,
codec_dai - > playback . stream_name ) ;
2008-05-19 12:31:28 +02:00
snd_soc_dapm_set_bias_level ( socdev ,
SND_SOC_BIAS_PREPARE ) ;
2008-01-10 14:38:24 +01:00
}
2006-10-06 18:31:09 +02:00
codec_dai - > pop_wait = 0 ;
2008-01-10 14:36:20 +01:00
snd_soc_dapm_stream_event ( codec ,
codec_dai - > playback . stream_name ,
2006-10-06 18:31:09 +02:00
SND_SOC_DAPM_STREAM_STOP ) ;
2008-05-19 12:31:28 +02:00
/* Fall into standby if no longer active */
2006-10-06 18:31:09 +02:00
if ( codec - > active = = 0 ) {
2008-10-21 21:45:08 +01:00
pr_debug ( " pop wq D3 %s %s \n " , codec - > name ,
codec_dai - > playback . stream_name ) ;
2008-05-19 12:31:28 +02:00
snd_soc_dapm_set_bias_level ( socdev ,
SND_SOC_BIAS_STANDBY ) ;
2006-10-06 18:31:09 +02:00
}
}
}
mutex_unlock ( & pcm_mutex ) ;
}
/*
* Called by ALSA when a PCM substream is closed . Private data can be
* freed here . The cpu DAI , codec DAI , machine and platform are also
* shutdown .
*/
static int soc_codec_close ( struct snd_pcm_substream * substream )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
struct snd_soc_device * socdev = rtd - > socdev ;
2008-12-02 15:08:03 +00:00
struct snd_soc_card * card = socdev - > card ;
2007-02-02 17:13:49 +01:00
struct snd_soc_dai_link * machine = rtd - > dai ;
2008-12-02 16:01:14 +00:00
struct snd_soc_platform * platform = card - > platform ;
2008-07-07 16:07:17 +01:00
struct snd_soc_dai * cpu_dai = machine - > cpu_dai ;
struct snd_soc_dai * codec_dai = machine - > codec_dai ;
2006-10-06 18:31:09 +02:00
struct snd_soc_codec * codec = socdev - > codec ;
mutex_lock ( & pcm_mutex ) ;
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK )
2007-02-02 17:13:49 +01:00
cpu_dai - > playback . active = codec_dai - > playback . active = 0 ;
2006-10-06 18:31:09 +02:00
else
2007-02-02 17:13:49 +01:00
cpu_dai - > capture . active = codec_dai - > capture . active = 0 ;
2006-10-06 18:31:09 +02:00
2007-02-02 17:13:49 +01:00
if ( codec_dai - > playback . active = = 0 & &
codec_dai - > capture . active = = 0 ) {
cpu_dai - > active = codec_dai - > active = 0 ;
2006-10-06 18:31:09 +02:00
}
codec - > active - - ;
2008-09-06 18:33:24 +01:00
/* Muting the DAC suppresses artifacts caused during digital
* shutdown , for example from stopping clocks .
*/
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK )
snd_soc_dai_digital_mute ( codec_dai , 1 ) ;
2007-02-02 17:13:49 +01:00
if ( cpu_dai - > ops . shutdown )
2008-11-18 22:11:38 +00:00
cpu_dai - > ops . shutdown ( substream , cpu_dai ) ;
2006-10-06 18:31:09 +02:00
2007-02-02 17:13:49 +01:00
if ( codec_dai - > ops . shutdown )
2008-11-18 22:11:38 +00:00
codec_dai - > ops . shutdown ( substream , codec_dai ) ;
2006-10-06 18:31:09 +02:00
if ( machine - > ops & & machine - > ops - > shutdown )
machine - > ops - > shutdown ( substream ) ;
if ( platform - > pcm_ops - > close )
platform - > pcm_ops - > close ( substream ) ;
2007-02-02 17:13:49 +01:00
cpu_dai - > runtime = NULL ;
2006-10-06 18:31:09 +02:00
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK ) {
/* start delayed pop wq here for playback streams */
2007-02-02 17:13:49 +01:00
codec_dai - > pop_wait = 1 ;
2008-12-02 15:08:03 +00:00
schedule_delayed_work ( & card - > delayed_work ,
2006-10-06 18:31:09 +02:00
msecs_to_jiffies ( pmdown_time ) ) ;
} else {
/* capture streams can be powered down now */
2007-02-02 17:13:49 +01:00
snd_soc_dapm_stream_event ( codec ,
2008-01-10 14:36:20 +01:00
codec_dai - > capture . stream_name ,
SND_SOC_DAPM_STREAM_STOP ) ;
2006-10-06 18:31:09 +02:00
2008-01-10 14:36:20 +01:00
if ( codec - > active = = 0 & & codec_dai - > pop_wait = = 0 )
2008-05-19 12:31:28 +02:00
snd_soc_dapm_set_bias_level ( socdev ,
SND_SOC_BIAS_STANDBY ) ;
2006-10-06 18:31:09 +02:00
}
mutex_unlock ( & pcm_mutex ) ;
return 0 ;
}
/*
* Called by ALSA when the PCM substream is prepared , can set format , sample
* rate , etc . This function is non atomic and can be called multiple times ,
* it can refer to the runtime info .
*/
static int soc_pcm_prepare ( struct snd_pcm_substream * substream )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
struct snd_soc_device * socdev = rtd - > socdev ;
2008-12-02 15:08:03 +00:00
struct snd_soc_card * card = socdev - > card ;
2007-02-02 17:13:49 +01:00
struct snd_soc_dai_link * machine = rtd - > dai ;
2008-12-02 16:01:14 +00:00
struct snd_soc_platform * platform = card - > platform ;
2008-07-07 16:07:17 +01:00
struct snd_soc_dai * cpu_dai = machine - > cpu_dai ;
struct snd_soc_dai * codec_dai = machine - > codec_dai ;
2006-10-06 18:31:09 +02:00
struct snd_soc_codec * codec = socdev - > codec ;
int ret = 0 ;
mutex_lock ( & pcm_mutex ) ;
2007-02-02 17:13:49 +01:00
if ( machine - > ops & & machine - > ops - > prepare ) {
ret = machine - > ops - > prepare ( substream ) ;
if ( ret < 0 ) {
printk ( KERN_ERR " asoc: machine prepare error \n " ) ;
goto out ;
}
}
2006-10-06 18:31:09 +02:00
if ( platform - > pcm_ops - > prepare ) {
ret = platform - > pcm_ops - > prepare ( substream ) ;
2006-10-19 20:35:56 +02:00
if ( ret < 0 ) {
printk ( KERN_ERR " asoc: platform prepare error \n " ) ;
2006-10-06 18:31:09 +02:00
goto out ;
2006-10-19 20:35:56 +02:00
}
2006-10-06 18:31:09 +02:00
}
2007-02-02 17:13:49 +01:00
if ( codec_dai - > ops . prepare ) {
2008-11-18 22:11:38 +00:00
ret = codec_dai - > ops . prepare ( substream , codec_dai ) ;
2006-10-19 20:35:56 +02:00
if ( ret < 0 ) {
printk ( KERN_ERR " asoc: codec DAI prepare error \n " ) ;
2006-10-06 18:31:09 +02:00
goto out ;
2006-10-19 20:35:56 +02:00
}
2006-10-06 18:31:09 +02:00
}
2007-02-02 17:13:49 +01:00
if ( cpu_dai - > ops . prepare ) {
2008-11-18 22:11:38 +00:00
ret = cpu_dai - > ops . prepare ( substream , cpu_dai ) ;
2007-02-02 17:13:49 +01:00
if ( ret < 0 ) {
printk ( KERN_ERR " asoc: cpu DAI prepare error \n " ) ;
goto out ;
}
}
2006-10-06 18:31:09 +02:00
2008-10-14 13:58:36 +01:00
/* cancel any delayed stream shutdown that is pending */
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK & &
codec_dai - > pop_wait ) {
codec_dai - > pop_wait = 0 ;
2008-12-02 15:08:03 +00:00
cancel_delayed_work ( & card - > delayed_work ) ;
2008-10-14 13:58:36 +01:00
}
2006-10-06 18:31:09 +02:00
2008-10-14 13:58:36 +01:00
/* do we need to power up codec */
if ( codec - > bias_level ! = SND_SOC_BIAS_ON ) {
snd_soc_dapm_set_bias_level ( socdev ,
SND_SOC_BIAS_PREPARE ) ;
2006-10-06 18:31:09 +02:00
2008-10-14 13:58:36 +01:00
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK )
snd_soc_dapm_stream_event ( codec ,
2007-02-02 17:13:49 +01:00
codec_dai - > playback . stream_name ,
2006-10-06 18:31:09 +02:00
SND_SOC_DAPM_STREAM_START ) ;
2008-10-14 13:58:36 +01:00
else
snd_soc_dapm_stream_event ( codec ,
2007-02-02 17:13:49 +01:00
codec_dai - > capture . stream_name ,
2006-10-06 18:31:09 +02:00
SND_SOC_DAPM_STREAM_START ) ;
2008-10-14 13:58:36 +01:00
snd_soc_dapm_set_bias_level ( socdev , SND_SOC_BIAS_ON ) ;
snd_soc_dai_digital_mute ( codec_dai , 0 ) ;
2006-10-06 18:31:09 +02:00
2008-10-14 13:58:36 +01:00
} else {
/* codec already powered - power on widgets */
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK )
snd_soc_dapm_stream_event ( codec ,
2007-02-02 17:13:49 +01:00
codec_dai - > playback . stream_name ,
2006-10-06 18:31:09 +02:00
SND_SOC_DAPM_STREAM_START ) ;
2008-10-14 13:58:36 +01:00
else
snd_soc_dapm_stream_event ( codec ,
2007-02-02 17:13:49 +01:00
codec_dai - > capture . stream_name ,
2006-10-06 18:31:09 +02:00
SND_SOC_DAPM_STREAM_START ) ;
2008-07-08 13:19:13 +01:00
2008-10-14 13:58:36 +01:00
snd_soc_dai_digital_mute ( codec_dai , 0 ) ;
2006-10-06 18:31:09 +02:00
}
out :
mutex_unlock ( & pcm_mutex ) ;
return ret ;
}
/*
* Called by ALSA when the hardware params are set by application . This
* function can also be called multiple times and can allocate buffers
* ( using snd_pcm_lib_ * ) . It ' s non - atomic .
*/
static int soc_pcm_hw_params ( struct snd_pcm_substream * substream ,
struct snd_pcm_hw_params * params )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
struct snd_soc_device * socdev = rtd - > socdev ;
2007-02-02 17:13:49 +01:00
struct snd_soc_dai_link * machine = rtd - > dai ;
2008-12-02 16:01:14 +00:00
struct snd_soc_card * card = socdev - > card ;
struct snd_soc_platform * platform = card - > platform ;
2008-07-07 16:07:17 +01:00
struct snd_soc_dai * cpu_dai = machine - > cpu_dai ;
struct snd_soc_dai * codec_dai = machine - > codec_dai ;
2006-10-06 18:31:09 +02:00
int ret = 0 ;
mutex_lock ( & pcm_mutex ) ;
2007-02-02 17:13:49 +01:00
if ( machine - > ops & & machine - > ops - > hw_params ) {
ret = machine - > ops - > hw_params ( substream , params ) ;
if ( ret < 0 ) {
printk ( KERN_ERR " asoc: machine hw_params failed \n " ) ;
2006-10-06 18:31:09 +02:00
goto out ;
2007-02-02 17:13:49 +01:00
}
2006-10-06 18:31:09 +02:00
}
2007-02-02 17:13:49 +01:00
if ( codec_dai - > ops . hw_params ) {
2008-11-18 22:11:38 +00:00
ret = codec_dai - > ops . hw_params ( substream , params , codec_dai ) ;
2006-10-06 18:31:09 +02:00
if ( ret < 0 ) {
printk ( KERN_ERR " asoc: can't set codec %s hw params \n " ,
2007-02-02 17:13:49 +01:00
codec_dai - > name ) ;
goto codec_err ;
2006-10-06 18:31:09 +02:00
}
}
2007-02-02 17:13:49 +01:00
if ( cpu_dai - > ops . hw_params ) {
2008-11-18 22:11:38 +00:00
ret = cpu_dai - > ops . hw_params ( substream , params , cpu_dai ) ;
2006-10-06 18:31:09 +02:00
if ( ret < 0 ) {
2008-05-19 12:32:25 +02:00
printk ( KERN_ERR " asoc: interface %s hw params failed \n " ,
2007-02-02 17:13:49 +01:00
cpu_dai - > name ) ;
2006-10-06 18:31:09 +02:00
goto interface_err ;
}
}
if ( platform - > pcm_ops - > hw_params ) {
ret = platform - > pcm_ops - > hw_params ( substream , params ) ;
if ( ret < 0 ) {
2008-05-19 12:32:25 +02:00
printk ( KERN_ERR " asoc: platform %s hw params failed \n " ,
2006-10-06 18:31:09 +02:00
platform - > name ) ;
goto platform_err ;
}
}
out :
mutex_unlock ( & pcm_mutex ) ;
return ret ;
platform_err :
2007-02-02 17:13:49 +01:00
if ( cpu_dai - > ops . hw_free )
2008-11-18 22:11:38 +00:00
cpu_dai - > ops . hw_free ( substream , cpu_dai ) ;
2006-10-06 18:31:09 +02:00
interface_err :
2007-02-02 17:13:49 +01:00
if ( codec_dai - > ops . hw_free )
2008-11-18 22:11:38 +00:00
codec_dai - > ops . hw_free ( substream , codec_dai ) ;
2007-02-02 17:13:49 +01:00
codec_err :
2008-05-19 12:32:25 +02:00
if ( machine - > ops & & machine - > ops - > hw_free )
2007-02-02 17:13:49 +01:00
machine - > ops - > hw_free ( substream ) ;
2006-10-06 18:31:09 +02:00
mutex_unlock ( & pcm_mutex ) ;
return ret ;
}
/*
* Free ' s resources allocated by hw_params , can be called multiple times
*/
static int soc_pcm_hw_free ( struct snd_pcm_substream * substream )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
struct snd_soc_device * socdev = rtd - > socdev ;
2007-02-02 17:13:49 +01:00
struct snd_soc_dai_link * machine = rtd - > dai ;
2008-12-02 16:01:14 +00:00
struct snd_soc_card * card = socdev - > card ;
struct snd_soc_platform * platform = card - > platform ;
2008-07-07 16:07:17 +01:00
struct snd_soc_dai * cpu_dai = machine - > cpu_dai ;
struct snd_soc_dai * codec_dai = machine - > codec_dai ;
2006-10-06 18:31:09 +02:00
struct snd_soc_codec * codec = socdev - > codec ;
mutex_lock ( & pcm_mutex ) ;
/* apply codec digital mute */
2008-07-08 13:19:13 +01:00
if ( ! codec - > active )
snd_soc_dai_digital_mute ( codec_dai , 1 ) ;
2006-10-06 18:31:09 +02:00
/* free any machine hw params */
if ( machine - > ops & & machine - > ops - > hw_free )
machine - > ops - > hw_free ( substream ) ;
/* free any DMA resources */
if ( platform - > pcm_ops - > hw_free )
platform - > pcm_ops - > hw_free ( substream ) ;
/* now free hw params for the DAI's */
2007-02-02 17:13:49 +01:00
if ( codec_dai - > ops . hw_free )
2008-11-18 22:11:38 +00:00
codec_dai - > ops . hw_free ( substream , codec_dai ) ;
2006-10-06 18:31:09 +02:00
2007-02-02 17:13:49 +01:00
if ( cpu_dai - > ops . hw_free )
2008-11-18 22:11:38 +00:00
cpu_dai - > ops . hw_free ( substream , cpu_dai ) ;
2006-10-06 18:31:09 +02:00
mutex_unlock ( & pcm_mutex ) ;
return 0 ;
}
static int soc_pcm_trigger ( struct snd_pcm_substream * substream , int cmd )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
struct snd_soc_device * socdev = rtd - > socdev ;
2008-12-02 16:01:14 +00:00
struct snd_soc_card * card = socdev - > card ;
2007-02-02 17:13:49 +01:00
struct snd_soc_dai_link * machine = rtd - > dai ;
2008-12-02 16:01:14 +00:00
struct snd_soc_platform * platform = card - > platform ;
2008-07-07 16:07:17 +01:00
struct snd_soc_dai * cpu_dai = machine - > cpu_dai ;
struct snd_soc_dai * codec_dai = machine - > codec_dai ;
2006-10-06 18:31:09 +02:00
int ret ;
2007-02-02 17:13:49 +01:00
if ( codec_dai - > ops . trigger ) {
2008-11-18 22:11:38 +00:00
ret = codec_dai - > ops . trigger ( substream , cmd , codec_dai ) ;
2006-10-06 18:31:09 +02:00
if ( ret < 0 )
return ret ;
}
if ( platform - > pcm_ops - > trigger ) {
ret = platform - > pcm_ops - > trigger ( substream , cmd ) ;
if ( ret < 0 )
return ret ;
}
2007-02-02 17:13:49 +01:00
if ( cpu_dai - > ops . trigger ) {
2008-11-18 22:11:38 +00:00
ret = cpu_dai - > ops . trigger ( substream , cmd , cpu_dai ) ;
2006-10-06 18:31:09 +02:00
if ( ret < 0 )
return ret ;
}
return 0 ;
}
/* ASoC PCM operations */
static struct snd_pcm_ops soc_pcm_ops = {
. open = soc_pcm_open ,
. close = soc_codec_close ,
. hw_params = soc_pcm_hw_params ,
. hw_free = soc_pcm_hw_free ,
. prepare = soc_pcm_prepare ,
. trigger = soc_pcm_trigger ,
} ;
# ifdef CONFIG_PM
/* powers down audio subsystem for suspend */
static int soc_suspend ( struct platform_device * pdev , pm_message_t state )
{
2008-05-19 12:32:25 +02:00
struct snd_soc_device * socdev = platform_get_drvdata ( pdev ) ;
2008-11-18 20:50:34 +00:00
struct snd_soc_card * card = socdev - > card ;
2008-12-02 16:01:14 +00:00
struct snd_soc_platform * platform = card - > platform ;
2008-05-19 12:32:25 +02:00
struct snd_soc_codec_device * codec_dev = socdev - > codec_dev ;
2006-10-06 18:31:09 +02:00
struct snd_soc_codec * codec = socdev - > codec ;
int i ;
2008-06-13 16:24:05 +01:00
/* Due to the resume being scheduled into a workqueue we could
* suspend before that ' s finished - wait for it to complete .
*/
snd_power_lock ( codec - > card ) ;
snd_power_wait ( codec - > card , SNDRV_CTL_POWER_D0 ) ;
snd_power_unlock ( codec - > card ) ;
/* we're going to block userspace touching us until resume completes */
snd_power_change_state ( codec - > card , SNDRV_CTL_POWER_D3hot ) ;
2006-10-06 18:31:09 +02:00
/* mute any active DAC's */
2008-11-18 22:11:38 +00:00
for ( i = 0 ; i < card - > num_links ; i + + ) {
struct snd_soc_dai * dai = card - > dai_link [ i ] . codec_dai ;
if ( dai - > ops . digital_mute & & dai - > playback . active )
dai - > ops . digital_mute ( dai , 1 ) ;
2006-10-06 18:31:09 +02:00
}
2008-01-10 14:39:01 +01:00
/* suspend all pcms */
2008-11-18 20:50:34 +00:00
for ( i = 0 ; i < card - > num_links ; i + + )
snd_pcm_suspend_all ( card - > dai_link [ i ] . pcm ) ;
2008-01-10 14:39:01 +01:00
2008-11-18 20:50:34 +00:00
if ( card - > suspend_pre )
card - > suspend_pre ( pdev , state ) ;
2006-10-06 18:31:09 +02:00
2008-11-18 20:50:34 +00:00
for ( i = 0 ; i < card - > num_links ; i + + ) {
struct snd_soc_dai * cpu_dai = card - > dai_link [ i ] . cpu_dai ;
2008-11-24 18:01:05 +00:00
if ( cpu_dai - > suspend & & ! cpu_dai - > ac97_control )
2006-10-06 18:31:09 +02:00
cpu_dai - > suspend ( pdev , cpu_dai ) ;
if ( platform - > suspend )
platform - > suspend ( pdev , cpu_dai ) ;
}
/* close any waiting streams and save state */
2008-12-02 15:08:03 +00:00
run_delayed_work ( & card - > delayed_work ) ;
2008-05-19 12:31:28 +02:00
codec - > suspend_bias_level = codec - > bias_level ;
2006-10-06 18:31:09 +02:00
2008-05-19 12:32:25 +02:00
for ( i = 0 ; i < codec - > num_dai ; i + + ) {
2006-10-06 18:31:09 +02:00
char * stream = codec - > dai [ i ] . playback . stream_name ;
if ( stream ! = NULL )
snd_soc_dapm_stream_event ( codec , stream ,
SND_SOC_DAPM_STREAM_SUSPEND ) ;
stream = codec - > dai [ i ] . capture . stream_name ;
if ( stream ! = NULL )
snd_soc_dapm_stream_event ( codec , stream ,
SND_SOC_DAPM_STREAM_SUSPEND ) ;
}
if ( codec_dev - > suspend )
codec_dev - > suspend ( pdev , state ) ;
2008-11-18 20:50:34 +00:00
for ( i = 0 ; i < card - > num_links ; i + + ) {
struct snd_soc_dai * cpu_dai = card - > dai_link [ i ] . cpu_dai ;
2008-11-24 18:01:05 +00:00
if ( cpu_dai - > suspend & & cpu_dai - > ac97_control )
2006-10-06 18:31:09 +02:00
cpu_dai - > suspend ( pdev , cpu_dai ) ;
}
2008-11-18 20:50:34 +00:00
if ( card - > suspend_post )
card - > suspend_post ( pdev , state ) ;
2006-10-06 18:31:09 +02:00
return 0 ;
}
2008-06-13 16:24:05 +01:00
/* deferred resume work, so resume can complete before we finished
* setting our codec back up , which can be very slow on I2C
*/
static void soc_resume_deferred ( struct work_struct * work )
2006-10-06 18:31:09 +02:00
{
2008-12-02 15:08:03 +00:00
struct snd_soc_card * card = container_of ( work ,
struct snd_soc_card ,
deferred_resume_work ) ;
struct snd_soc_device * socdev = card - > socdev ;
2008-12-02 16:01:14 +00:00
struct snd_soc_platform * platform = card - > platform ;
2008-05-19 12:32:25 +02:00
struct snd_soc_codec_device * codec_dev = socdev - > codec_dev ;
2006-10-06 18:31:09 +02:00
struct snd_soc_codec * codec = socdev - > codec ;
2008-06-13 16:24:05 +01:00
struct platform_device * pdev = to_platform_device ( socdev - > dev ) ;
2006-10-06 18:31:09 +02:00
int i ;
2008-06-13 16:24:05 +01:00
/* our power state is still SNDRV_CTL_POWER_D3hot from suspend time,
* so userspace apps are blocked from touching us
*/
2008-11-24 18:08:18 +00:00
dev_dbg ( socdev - > dev , " starting resume work \n " ) ;
2008-06-13 16:24:05 +01:00
2008-11-18 20:50:34 +00:00
if ( card - > resume_pre )
card - > resume_pre ( pdev ) ;
2006-10-06 18:31:09 +02:00
2008-11-18 20:50:34 +00:00
for ( i = 0 ; i < card - > num_links ; i + + ) {
struct snd_soc_dai * cpu_dai = card - > dai_link [ i ] . cpu_dai ;
2008-11-24 18:01:05 +00:00
if ( cpu_dai - > resume & & cpu_dai - > ac97_control )
2006-10-06 18:31:09 +02:00
cpu_dai - > resume ( pdev , cpu_dai ) ;
}
if ( codec_dev - > resume )
codec_dev - > resume ( pdev ) ;
2008-05-19 12:32:25 +02:00
for ( i = 0 ; i < codec - > num_dai ; i + + ) {
char * stream = codec - > dai [ i ] . playback . stream_name ;
2006-10-06 18:31:09 +02:00
if ( stream ! = NULL )
snd_soc_dapm_stream_event ( codec , stream ,
SND_SOC_DAPM_STREAM_RESUME ) ;
stream = codec - > dai [ i ] . capture . stream_name ;
if ( stream ! = NULL )
snd_soc_dapm_stream_event ( codec , stream ,
SND_SOC_DAPM_STREAM_RESUME ) ;
}
2008-05-19 12:32:25 +02:00
/* unmute any active DACs */
2008-11-18 22:11:38 +00:00
for ( i = 0 ; i < card - > num_links ; i + + ) {
struct snd_soc_dai * dai = card - > dai_link [ i ] . codec_dai ;
if ( dai - > ops . digital_mute & & dai - > playback . active )
dai - > ops . digital_mute ( dai , 0 ) ;
2006-10-06 18:31:09 +02:00
}
2008-11-18 20:50:34 +00:00
for ( i = 0 ; i < card - > num_links ; i + + ) {
struct snd_soc_dai * cpu_dai = card - > dai_link [ i ] . cpu_dai ;
2008-11-24 18:01:05 +00:00
if ( cpu_dai - > resume & & ! cpu_dai - > ac97_control )
2006-10-06 18:31:09 +02:00
cpu_dai - > resume ( pdev , cpu_dai ) ;
if ( platform - > resume )
platform - > resume ( pdev , cpu_dai ) ;
}
2008-11-18 20:50:34 +00:00
if ( card - > resume_post )
card - > resume_post ( pdev ) ;
2006-10-06 18:31:09 +02:00
2008-11-24 18:08:18 +00:00
dev_dbg ( socdev - > dev , " resume work completed \n " ) ;
2008-06-13 16:24:05 +01:00
/* userspace can access us now we are back as we were before */
snd_power_change_state ( codec - > card , SNDRV_CTL_POWER_D0 ) ;
}
/* powers up audio subsystem after a suspend */
static int soc_resume ( struct platform_device * pdev )
{
struct snd_soc_device * socdev = platform_get_drvdata ( pdev ) ;
2008-12-02 15:08:03 +00:00
struct snd_soc_card * card = socdev - > card ;
2008-06-13 16:24:05 +01:00
2008-11-24 18:08:18 +00:00
dev_dbg ( socdev - > dev , " scheduling resume work \n " ) ;
2008-06-13 16:24:05 +01:00
2008-12-02 15:08:03 +00:00
if ( ! schedule_work ( & card - > deferred_resume_work ) )
2008-11-24 18:08:18 +00:00
dev_err ( socdev - > dev , " resume work item may be lost \n " ) ;
2008-06-13 16:24:05 +01:00
2006-10-06 18:31:09 +02:00
return 0 ;
}
# else
# define soc_suspend NULL
# define soc_resume NULL
# endif
/* probes a new socdev */
static int soc_probe ( struct platform_device * pdev )
{
int ret = 0 , i ;
struct snd_soc_device * socdev = platform_get_drvdata ( pdev ) ;
2008-11-18 20:50:34 +00:00
struct snd_soc_card * card = socdev - > card ;
2008-12-02 16:01:14 +00:00
struct snd_soc_platform * platform = card - > platform ;
2006-10-06 18:31:09 +02:00
struct snd_soc_codec_device * codec_dev = socdev - > codec_dev ;
2008-12-02 15:08:03 +00:00
/* Bodge while we push things out of socdev */
card - > socdev = socdev ;
2008-11-18 20:50:34 +00:00
if ( card - > probe ) {
ret = card - > probe ( pdev ) ;
2008-05-19 12:32:25 +02:00
if ( ret < 0 )
2006-10-06 18:31:09 +02:00
return ret ;
}
2008-11-18 20:50:34 +00:00
for ( i = 0 ; i < card - > num_links ; i + + ) {
struct snd_soc_dai * cpu_dai = card - > dai_link [ i ] . cpu_dai ;
2006-10-06 18:31:09 +02:00
if ( cpu_dai - > probe ) {
2008-06-11 13:47:10 +01:00
ret = cpu_dai - > probe ( pdev , cpu_dai ) ;
2008-05-19 12:32:25 +02:00
if ( ret < 0 )
2006-10-06 18:31:09 +02:00
goto cpu_dai_err ;
}
}
if ( codec_dev - > probe ) {
ret = codec_dev - > probe ( pdev ) ;
2008-05-19 12:32:25 +02:00
if ( ret < 0 )
2006-10-06 18:31:09 +02:00
goto cpu_dai_err ;
}
if ( platform - > probe ) {
ret = platform - > probe ( pdev ) ;
2008-05-19 12:32:25 +02:00
if ( ret < 0 )
2006-10-06 18:31:09 +02:00
goto platform_err ;
}
/* DAPM stream work */
2008-12-02 15:08:03 +00:00
INIT_DELAYED_WORK ( & card - > delayed_work , close_delayed_work ) ;
2008-06-17 19:19:34 +01:00
# ifdef CONFIG_PM
2008-06-13 16:24:05 +01:00
/* deferred resume work */
2008-12-02 15:08:03 +00:00
INIT_WORK ( & card - > deferred_resume_work , soc_resume_deferred ) ;
2008-06-17 19:19:34 +01:00
# endif
2008-06-13 16:24:05 +01:00
2006-10-06 18:31:09 +02:00
return 0 ;
platform_err :
if ( codec_dev - > remove )
codec_dev - > remove ( pdev ) ;
cpu_dai_err :
2007-01-30 17:18:45 +01:00
for ( i - - ; i > = 0 ; i - - ) {
2008-11-18 20:50:34 +00:00
struct snd_soc_dai * cpu_dai = card - > dai_link [ i ] . cpu_dai ;
2006-10-06 18:31:09 +02:00
if ( cpu_dai - > remove )
2008-06-11 13:47:10 +01:00
cpu_dai - > remove ( pdev , cpu_dai ) ;
2006-10-06 18:31:09 +02:00
}
2008-11-18 20:50:34 +00:00
if ( card - > remove )
card - > remove ( pdev ) ;
2006-10-06 18:31:09 +02:00
return ret ;
}
/* removes a socdev */
static int soc_remove ( struct platform_device * pdev )
{
int i ;
struct snd_soc_device * socdev = platform_get_drvdata ( pdev ) ;
2008-11-18 20:50:34 +00:00
struct snd_soc_card * card = socdev - > card ;
2008-12-02 16:01:14 +00:00
struct snd_soc_platform * platform = card - > platform ;
2006-10-06 18:31:09 +02:00
struct snd_soc_codec_device * codec_dev = socdev - > codec_dev ;
2008-12-02 15:08:03 +00:00
run_delayed_work ( & card - > delayed_work ) ;
2007-01-31 14:14:57 +01:00
2006-10-06 18:31:09 +02:00
if ( platform - > remove )
platform - > remove ( pdev ) ;
if ( codec_dev - > remove )
codec_dev - > remove ( pdev ) ;
2008-11-18 20:50:34 +00:00
for ( i = 0 ; i < card - > num_links ; i + + ) {
struct snd_soc_dai * cpu_dai = card - > dai_link [ i ] . cpu_dai ;
2006-10-06 18:31:09 +02:00
if ( cpu_dai - > remove )
2008-06-11 13:47:10 +01:00
cpu_dai - > remove ( pdev , cpu_dai ) ;
2006-10-06 18:31:09 +02:00
}
2008-11-18 20:50:34 +00:00
if ( card - > remove )
card - > remove ( pdev ) ;
2006-10-06 18:31:09 +02:00
return 0 ;
}
/* ASoC platform driver */
static struct platform_driver soc_driver = {
. driver = {
. name = " soc-audio " ,
2008-04-14 13:33:36 +02:00
. owner = THIS_MODULE ,
2006-10-06 18:31:09 +02:00
} ,
. probe = soc_probe ,
. remove = soc_remove ,
. suspend = soc_suspend ,
. resume = soc_resume ,
} ;
/* create a new pcm */
static int soc_new_pcm ( struct snd_soc_device * socdev ,
struct snd_soc_dai_link * dai_link , int num )
{
struct snd_soc_codec * codec = socdev - > codec ;
2008-12-02 16:01:14 +00:00
struct snd_soc_card * card = socdev - > card ;
struct snd_soc_platform * platform = card - > platform ;
2008-07-07 16:07:17 +01:00
struct snd_soc_dai * codec_dai = dai_link - > codec_dai ;
struct snd_soc_dai * cpu_dai = dai_link - > cpu_dai ;
2006-10-06 18:31:09 +02:00
struct snd_soc_pcm_runtime * rtd ;
struct snd_pcm * pcm ;
char new_name [ 64 ] ;
int ret = 0 , playback = 0 , capture = 0 ;
rtd = kzalloc ( sizeof ( struct snd_soc_pcm_runtime ) , GFP_KERNEL ) ;
if ( rtd = = NULL )
return - ENOMEM ;
2007-02-02 17:13:49 +01:00
rtd - > dai = dai_link ;
2006-10-06 18:31:09 +02:00
rtd - > socdev = socdev ;
2007-02-02 17:13:49 +01:00
codec_dai - > codec = socdev - > codec ;
2006-10-06 18:31:09 +02:00
/* check client and interface hw capabilities */
2008-11-24 18:01:05 +00:00
sprintf ( new_name , " %s %s-%d " , dai_link - > stream_name , codec_dai - > name ,
num ) ;
2006-10-06 18:31:09 +02:00
if ( codec_dai - > playback . channels_min )
playback = 1 ;
if ( codec_dai - > capture . channels_min )
capture = 1 ;
ret = snd_pcm_new ( codec - > card , new_name , codec - > pcm_devs + + , playback ,
capture , & pcm ) ;
if ( ret < 0 ) {
2008-05-19 12:32:25 +02:00
printk ( KERN_ERR " asoc: can't create pcm for codec %s \n " ,
codec - > name ) ;
2006-10-06 18:31:09 +02:00
kfree ( rtd ) ;
return ret ;
}
2008-01-10 14:39:01 +01:00
dai_link - > pcm = pcm ;
2006-10-06 18:31:09 +02:00
pcm - > private_data = rtd ;
2008-12-02 16:01:14 +00:00
soc_pcm_ops . mmap = platform - > pcm_ops - > mmap ;
soc_pcm_ops . pointer = platform - > pcm_ops - > pointer ;
soc_pcm_ops . ioctl = platform - > pcm_ops - > ioctl ;
soc_pcm_ops . copy = platform - > pcm_ops - > copy ;
soc_pcm_ops . silence = platform - > pcm_ops - > silence ;
soc_pcm_ops . ack = platform - > pcm_ops - > ack ;
soc_pcm_ops . page = platform - > pcm_ops - > page ;
2006-10-06 18:31:09 +02:00
if ( playback )
snd_pcm_set_ops ( pcm , SNDRV_PCM_STREAM_PLAYBACK , & soc_pcm_ops ) ;
if ( capture )
snd_pcm_set_ops ( pcm , SNDRV_PCM_STREAM_CAPTURE , & soc_pcm_ops ) ;
2008-12-02 16:01:14 +00:00
ret = platform - > pcm_new ( codec - > card , codec_dai , pcm ) ;
2006-10-06 18:31:09 +02:00
if ( ret < 0 ) {
printk ( KERN_ERR " asoc: platform pcm constructor failed \n " ) ;
kfree ( rtd ) ;
return ret ;
}
2008-12-02 16:01:14 +00:00
pcm - > private_free = platform - > pcm_free ;
2006-10-06 18:31:09 +02:00
printk ( KERN_INFO " asoc: %s <-> %s mapping ok \n " , codec_dai - > name ,
cpu_dai - > name ) ;
return ret ;
}
/* codec register dump */
2008-10-13 17:42:14 -07:00
static ssize_t soc_codec_reg_show ( struct snd_soc_device * devdata , char * buf )
2006-10-06 18:31:09 +02:00
{
struct snd_soc_codec * codec = devdata - > codec ;
int i , step = 1 , count = 0 ;
if ( ! codec - > reg_cache_size )
return 0 ;
if ( codec - > reg_cache_step )
step = codec - > reg_cache_step ;
count + = sprintf ( buf , " %s registers \n " , codec - > name ) ;
2008-07-29 11:42:25 +01:00
for ( i = 0 ; i < codec - > reg_cache_size ; i + = step ) {
count + = sprintf ( buf + count , " %2x: " , i ) ;
if ( count > = PAGE_SIZE - 1 )
break ;
if ( codec - > display_register )
count + = codec - > display_register ( codec , buf + count ,
PAGE_SIZE - count , i ) ;
else
count + = snprintf ( buf + count , PAGE_SIZE - count ,
" %4x " , codec - > read ( codec , i ) ) ;
if ( count > = PAGE_SIZE - 1 )
break ;
count + = snprintf ( buf + count , PAGE_SIZE - count , " \n " ) ;
if ( count > = PAGE_SIZE - 1 )
break ;
}
/* Truncate count; min() would cause a warning */
if ( count > = PAGE_SIZE )
count = PAGE_SIZE - 1 ;
2006-10-06 18:31:09 +02:00
return count ;
}
2008-10-13 17:42:14 -07:00
static ssize_t codec_reg_show ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
struct snd_soc_device * devdata = dev_get_drvdata ( dev ) ;
return soc_codec_reg_show ( devdata , buf ) ;
}
2006-10-06 18:31:09 +02:00
static DEVICE_ATTR ( codec_reg , 0444 , codec_reg_show , NULL ) ;
2008-10-13 17:42:14 -07:00
# ifdef CONFIG_DEBUG_FS
static int codec_reg_open_file ( struct inode * inode , struct file * file )
{
file - > private_data = inode - > i_private ;
return 0 ;
}
static ssize_t codec_reg_read_file ( struct file * file , char __user * user_buf ,
size_t count , loff_t * ppos )
{
ssize_t ret ;
struct snd_soc_device * devdata = file - > private_data ;
char * buf = kmalloc ( PAGE_SIZE , GFP_KERNEL ) ;
if ( ! buf )
return - ENOMEM ;
ret = soc_codec_reg_show ( devdata , buf ) ;
if ( ret > = 0 )
ret = simple_read_from_buffer ( user_buf , count , ppos , buf , ret ) ;
kfree ( buf ) ;
return ret ;
}
static ssize_t codec_reg_write_file ( struct file * file ,
const char __user * user_buf , size_t count , loff_t * ppos )
{
char buf [ 32 ] ;
int buf_size ;
char * start = buf ;
unsigned long reg , value ;
int step = 1 ;
struct snd_soc_device * devdata = file - > private_data ;
struct snd_soc_codec * codec = devdata - > codec ;
buf_size = min ( count , ( sizeof ( buf ) - 1 ) ) ;
if ( copy_from_user ( buf , user_buf , buf_size ) )
return - EFAULT ;
buf [ buf_size ] = 0 ;
if ( codec - > reg_cache_step )
step = codec - > reg_cache_step ;
while ( * start = = ' ' )
start + + ;
reg = simple_strtoul ( start , & start , 16 ) ;
if ( ( reg > = codec - > reg_cache_size ) | | ( reg % step ) )
return - EINVAL ;
while ( * start = = ' ' )
start + + ;
if ( strict_strtoul ( start , 16 , & value ) )
return - EINVAL ;
codec - > write ( codec , reg , value ) ;
return buf_size ;
}
static const struct file_operations codec_reg_fops = {
. open = codec_reg_open_file ,
. read = codec_reg_read_file ,
. write = codec_reg_write_file ,
} ;
static void soc_init_debugfs ( struct snd_soc_device * socdev )
{
struct dentry * root , * file ;
struct snd_soc_codec * codec = socdev - > codec ;
root = debugfs_create_dir ( dev_name ( socdev - > dev ) , NULL ) ;
if ( IS_ERR ( root ) | | ! root )
goto exit1 ;
file = debugfs_create_file ( " codec_reg " , 0644 ,
root , socdev , & codec_reg_fops ) ;
if ( ! file )
goto exit2 ;
file = debugfs_create_u32 ( " dapm_pop_time " , 0744 ,
root , & codec - > pop_time ) ;
if ( ! file )
goto exit2 ;
socdev - > debugfs_root = root ;
return ;
exit2 :
debugfs_remove_recursive ( root ) ;
exit1 :
dev_err ( socdev - > dev , " debugfs is not available \n " ) ;
}
static void soc_cleanup_debugfs ( struct snd_soc_device * socdev )
{
debugfs_remove_recursive ( socdev - > debugfs_root ) ;
socdev - > debugfs_root = NULL ;
}
# else
static inline void soc_init_debugfs ( struct snd_soc_device * socdev )
{
}
static inline void soc_cleanup_debugfs ( struct snd_soc_device * socdev )
{
}
# endif
2006-10-06 18:31:09 +02:00
/**
* snd_soc_new_ac97_codec - initailise AC97 device
* @ codec : audio codec
* @ ops : AC97 bus operations
* @ num : AC97 codec number
*
* Initialises AC97 codec resources for use by ad - hoc devices only .
*/
int snd_soc_new_ac97_codec ( struct snd_soc_codec * codec ,
struct snd_ac97_bus_ops * ops , int num )
{
mutex_lock ( & codec - > mutex ) ;
codec - > ac97 = kzalloc ( sizeof ( struct snd_ac97 ) , GFP_KERNEL ) ;
if ( codec - > ac97 = = NULL ) {
mutex_unlock ( & codec - > mutex ) ;
return - ENOMEM ;
}
codec - > ac97 - > bus = kzalloc ( sizeof ( struct snd_ac97_bus ) , GFP_KERNEL ) ;
if ( codec - > ac97 - > bus = = NULL ) {
kfree ( codec - > ac97 ) ;
codec - > ac97 = NULL ;
mutex_unlock ( & codec - > mutex ) ;
return - ENOMEM ;
}
codec - > ac97 - > bus - > ops = ops ;
codec - > ac97 - > num = num ;
mutex_unlock ( & codec - > mutex ) ;
return 0 ;
}
EXPORT_SYMBOL_GPL ( snd_soc_new_ac97_codec ) ;
/**
* snd_soc_free_ac97_codec - free AC97 codec device
* @ codec : audio codec
*
* Frees AC97 codec device resources .
*/
void snd_soc_free_ac97_codec ( struct snd_soc_codec * codec )
{
mutex_lock ( & codec - > mutex ) ;
kfree ( codec - > ac97 - > bus ) ;
kfree ( codec - > ac97 ) ;
codec - > ac97 = NULL ;
mutex_unlock ( & codec - > mutex ) ;
}
EXPORT_SYMBOL_GPL ( snd_soc_free_ac97_codec ) ;
/**
* snd_soc_update_bits - update codec register bits
* @ codec : audio codec
* @ reg : codec register
* @ mask : register mask
* @ value : new value
*
* Writes new register value .
*
* Returns 1 for change else 0.
*/
int snd_soc_update_bits ( struct snd_soc_codec * codec , unsigned short reg ,
unsigned short mask , unsigned short value )
{
int change ;
unsigned short old , new ;
mutex_lock ( & io_mutex ) ;
old = snd_soc_read ( codec , reg ) ;
new = ( old & ~ mask ) | value ;
change = old ! = new ;
if ( change )
snd_soc_write ( codec , reg , new ) ;
mutex_unlock ( & io_mutex ) ;
return change ;
}
EXPORT_SYMBOL_GPL ( snd_soc_update_bits ) ;
/**
* snd_soc_test_bits - test register for change
* @ codec : audio codec
* @ reg : codec register
* @ mask : register mask
* @ value : new value
*
* Tests a register with a new value and checks if the new value is
* different from the old value .
*
* Returns 1 for change else 0.
*/
int snd_soc_test_bits ( struct snd_soc_codec * codec , unsigned short reg ,
unsigned short mask , unsigned short value )
{
int change ;
unsigned short old , new ;
mutex_lock ( & io_mutex ) ;
old = snd_soc_read ( codec , reg ) ;
new = ( old & ~ mask ) | value ;
change = old ! = new ;
mutex_unlock ( & io_mutex ) ;
return change ;
}
EXPORT_SYMBOL_GPL ( snd_soc_test_bits ) ;
/**
* snd_soc_new_pcms - create new sound card and pcms
* @ socdev : the SoC audio device
*
* Create a new sound card based upon the codec and interface pcms .
*
* Returns 0 for success , else error .
*/
2007-02-01 12:26:07 +01:00
int snd_soc_new_pcms ( struct snd_soc_device * socdev , int idx , const char * xid )
2006-10-06 18:31:09 +02:00
{
struct snd_soc_codec * codec = socdev - > codec ;
2008-11-18 20:50:34 +00:00
struct snd_soc_card * card = socdev - > card ;
2006-10-06 18:31:09 +02:00
int ret = 0 , i ;
mutex_lock ( & codec - > mutex ) ;
/* register a sound card */
codec - > card = snd_card_new ( idx , xid , codec - > owner , 0 ) ;
if ( ! codec - > card ) {
printk ( KERN_ERR " asoc: can't create sound card for codec %s \n " ,
codec - > name ) ;
mutex_unlock ( & codec - > mutex ) ;
return - ENODEV ;
}
codec - > card - > dev = socdev - > dev ;
codec - > card - > private_data = codec ;
strncpy ( codec - > card - > driver , codec - > name , sizeof ( codec - > card - > driver ) ) ;
/* create the pcms */
2008-11-18 20:50:34 +00:00
for ( i = 0 ; i < card - > num_links ; i + + ) {
ret = soc_new_pcm ( socdev , & card - > dai_link [ i ] , i ) ;
2006-10-06 18:31:09 +02:00
if ( ret < 0 ) {
printk ( KERN_ERR " asoc: can't create pcm %s \n " ,
2008-11-18 20:50:34 +00:00
card - > dai_link [ i ] . stream_name ) ;
2006-10-06 18:31:09 +02:00
mutex_unlock ( & codec - > mutex ) ;
return ret ;
}
}
mutex_unlock ( & codec - > mutex ) ;
return ret ;
}
EXPORT_SYMBOL_GPL ( snd_soc_new_pcms ) ;
/**
2008-11-28 11:49:07 +00:00
* snd_soc_init_card - register sound card
2006-10-06 18:31:09 +02:00
* @ socdev : the SoC audio device
*
* Register a SoC sound card . Also registers an AC97 device if the
* codec is AC97 for ad hoc devices .
*
* Returns 0 for success , else error .
*/
2008-11-28 11:49:07 +00:00
int snd_soc_init_card ( struct snd_soc_device * socdev )
2006-10-06 18:31:09 +02:00
{
struct snd_soc_codec * codec = socdev - > codec ;
2008-11-18 20:50:34 +00:00
struct snd_soc_card * card = socdev - > card ;
2006-10-16 21:19:48 +02:00
int ret = 0 , i , ac97 = 0 , err = 0 ;
2006-10-06 18:31:09 +02:00
2008-11-18 20:50:34 +00:00
for ( i = 0 ; i < card - > num_links ; i + + ) {
if ( card - > dai_link [ i ] . init ) {
err = card - > dai_link [ i ] . init ( codec ) ;
2006-10-16 21:19:48 +02:00
if ( err < 0 ) {
printk ( KERN_ERR " asoc: failed to init %s \n " ,
2008-11-18 20:50:34 +00:00
card - > dai_link [ i ] . stream_name ) ;
2006-10-16 21:19:48 +02:00
continue ;
}
}
2008-11-24 18:01:05 +00:00
if ( card - > dai_link [ i ] . codec_dai - > ac97_control )
2006-10-06 18:31:09 +02:00
ac97 = 1 ;
}
snprintf ( codec - > card - > shortname , sizeof ( codec - > card - > shortname ) ,
2008-11-18 20:50:34 +00:00
" %s " , card - > name ) ;
2006-10-06 18:31:09 +02:00
snprintf ( codec - > card - > longname , sizeof ( codec - > card - > longname ) ,
2008-11-18 20:50:34 +00:00
" %s (%s) " , card - > name , codec - > name ) ;
2006-10-06 18:31:09 +02:00
ret = snd_card_register ( codec - > card ) ;
if ( ret < 0 ) {
2008-05-19 12:32:25 +02:00
printk ( KERN_ERR " asoc: failed to register soundcard for %s \n " ,
2006-10-06 18:31:09 +02:00
codec - > name ) ;
2006-10-16 21:19:48 +02:00
goto out ;
2006-10-06 18:31:09 +02:00
}
2008-01-21 14:33:37 +01:00
mutex_lock ( & codec - > mutex ) ;
2006-10-06 18:31:09 +02:00
# ifdef CONFIG_SND_SOC_AC97_BUS
2006-10-16 21:19:48 +02:00
if ( ac97 ) {
ret = soc_ac97_dev_register ( codec ) ;
if ( ret < 0 ) {
printk ( KERN_ERR " asoc: AC97 device register failed \n " ) ;
snd_card_free ( codec - > card ) ;
2008-01-21 14:33:37 +01:00
mutex_unlock ( & codec - > mutex ) ;
2006-10-16 21:19:48 +02:00
goto out ;
}
}
2006-10-06 18:31:09 +02:00
# endif
2006-10-16 21:19:48 +02:00
err = snd_soc_dapm_sys_add ( socdev - > dev ) ;
if ( err < 0 )
printk ( KERN_WARNING " asoc: failed to add dapm sysfs entries \n " ) ;
err = device_create_file ( socdev - > dev , & dev_attr_codec_reg ) ;
if ( err < 0 )
2008-05-19 12:32:25 +02:00
printk ( KERN_WARNING " asoc: failed to add codec sysfs files \n " ) ;
2008-01-21 14:33:37 +01:00
2008-10-13 17:42:14 -07:00
soc_init_debugfs ( socdev ) ;
2006-10-06 18:31:09 +02:00
mutex_unlock ( & codec - > mutex ) ;
2008-01-21 14:33:37 +01:00
out :
2006-10-06 18:31:09 +02:00
return ret ;
}
2008-11-28 11:49:07 +00:00
EXPORT_SYMBOL_GPL ( snd_soc_init_card ) ;
2006-10-06 18:31:09 +02:00
/**
* snd_soc_free_pcms - free sound card and pcms
* @ socdev : the SoC audio device
*
* Frees sound card and pcms associated with the socdev .
* Also unregister the codec if it is an AC97 device .
*/
void snd_soc_free_pcms ( struct snd_soc_device * socdev )
{
struct snd_soc_codec * codec = socdev - > codec ;
2007-05-10 19:27:27 +02:00
# ifdef CONFIG_SND_SOC_AC97_BUS
2008-07-07 16:07:17 +01:00
struct snd_soc_dai * codec_dai ;
2007-05-10 19:27:27 +02:00
int i ;
# endif
2006-10-06 18:31:09 +02:00
mutex_lock ( & codec - > mutex ) ;
2008-10-13 17:42:14 -07:00
soc_cleanup_debugfs ( socdev ) ;
2006-10-06 18:31:09 +02:00
# ifdef CONFIG_SND_SOC_AC97_BUS
2008-05-19 12:32:25 +02:00
for ( i = 0 ; i < codec - > num_dai ; i + + ) {
2007-05-10 19:27:27 +02:00
codec_dai = & codec - > dai [ i ] ;
2008-11-24 18:01:05 +00:00
if ( codec_dai - > ac97_control & & codec - > ac97 ) {
2007-05-10 19:27:27 +02:00
soc_ac97_dev_unregister ( codec ) ;
goto free_card ;
}
}
free_card :
2006-10-06 18:31:09 +02:00
# endif
if ( codec - > card )
snd_card_free ( codec - > card ) ;
device_remove_file ( socdev - > dev , & dev_attr_codec_reg ) ;
mutex_unlock ( & codec - > mutex ) ;
}
EXPORT_SYMBOL_GPL ( snd_soc_free_pcms ) ;
/**
* snd_soc_set_runtime_hwparams - set the runtime hardware parameters
* @ substream : the pcm substream
* @ hw : the hardware parameters
*
* Sets the substream runtime hardware parameters .
*/
int snd_soc_set_runtime_hwparams ( struct snd_pcm_substream * substream ,
const struct snd_pcm_hardware * hw )
{
struct snd_pcm_runtime * runtime = substream - > runtime ;
runtime - > hw . info = hw - > info ;
runtime - > hw . formats = hw - > formats ;
runtime - > hw . period_bytes_min = hw - > period_bytes_min ;
runtime - > hw . period_bytes_max = hw - > period_bytes_max ;
runtime - > hw . periods_min = hw - > periods_min ;
runtime - > hw . periods_max = hw - > periods_max ;
runtime - > hw . buffer_bytes_max = hw - > buffer_bytes_max ;
runtime - > hw . fifo_size = hw - > fifo_size ;
return 0 ;
}
EXPORT_SYMBOL_GPL ( snd_soc_set_runtime_hwparams ) ;
/**
* snd_soc_cnew - create new control
* @ _template : control template
* @ data : control private data
* @ lnng_name : control long name
*
* Create a new mixer control from a template control .
*
* Returns 0 for success , else error .
*/
struct snd_kcontrol * snd_soc_cnew ( const struct snd_kcontrol_new * _template ,
void * data , char * long_name )
{
struct snd_kcontrol_new template ;
memcpy ( & template , _template , sizeof ( template ) ) ;
if ( long_name )
template . name = long_name ;
template . index = 0 ;
return snd_ctl_new1 ( & template , data ) ;
}
EXPORT_SYMBOL_GPL ( snd_soc_cnew ) ;
/**
* snd_soc_info_enum_double - enumerated double mixer info callback
* @ kcontrol : mixer control
* @ uinfo : control element information
*
* Callback to provide information about a double enumerated
* mixer control .
*
* Returns 0 for success .
*/
int snd_soc_info_enum_double ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
{
struct soc_enum * e = ( struct soc_enum * ) kcontrol - > private_value ;
uinfo - > type = SNDRV_CTL_ELEM_TYPE_ENUMERATED ;
uinfo - > count = e - > shift_l = = e - > shift_r ? 1 : 2 ;
2008-07-29 11:42:27 +01:00
uinfo - > value . enumerated . items = e - > max ;
2006-10-06 18:31:09 +02:00
2008-07-29 11:42:27 +01:00
if ( uinfo - > value . enumerated . item > e - > max - 1 )
uinfo - > value . enumerated . item = e - > max - 1 ;
2006-10-06 18:31:09 +02:00
strcpy ( uinfo - > value . enumerated . name ,
e - > texts [ uinfo - > value . enumerated . item ] ) ;
return 0 ;
}
EXPORT_SYMBOL_GPL ( snd_soc_info_enum_double ) ;
/**
* snd_soc_get_enum_double - enumerated double mixer get callback
* @ kcontrol : mixer control
* @ uinfo : control element information
*
* Callback to get the value of a double enumerated mixer .
*
* Returns 0 for success .
*/
int snd_soc_get_enum_double ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
struct snd_soc_codec * codec = snd_kcontrol_chip ( kcontrol ) ;
struct soc_enum * e = ( struct soc_enum * ) kcontrol - > private_value ;
unsigned short val , bitmask ;
2008-07-29 11:42:27 +01:00
for ( bitmask = 1 ; bitmask < e - > max ; bitmask < < = 1 )
2006-10-06 18:31:09 +02:00
;
val = snd_soc_read ( codec , e - > reg ) ;
2008-05-19 12:32:25 +02:00
ucontrol - > value . enumerated . item [ 0 ]
= ( val > > e - > shift_l ) & ( bitmask - 1 ) ;
2006-10-06 18:31:09 +02:00
if ( e - > shift_l ! = e - > shift_r )
ucontrol - > value . enumerated . item [ 1 ] =
( val > > e - > shift_r ) & ( bitmask - 1 ) ;
return 0 ;
}
EXPORT_SYMBOL_GPL ( snd_soc_get_enum_double ) ;
/**
* snd_soc_put_enum_double - enumerated double mixer put callback
* @ kcontrol : mixer control
* @ uinfo : control element information
*
* Callback to set the value of a double enumerated mixer .
*
* Returns 0 for success .
*/
int snd_soc_put_enum_double ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
struct snd_soc_codec * codec = snd_kcontrol_chip ( kcontrol ) ;
struct soc_enum * e = ( struct soc_enum * ) kcontrol - > private_value ;
unsigned short val ;
unsigned short mask , bitmask ;
2008-07-29 11:42:27 +01:00
for ( bitmask = 1 ; bitmask < e - > max ; bitmask < < = 1 )
2006-10-06 18:31:09 +02:00
;
2008-07-29 11:42:27 +01:00
if ( ucontrol - > value . enumerated . item [ 0 ] > e - > max - 1 )
2006-10-06 18:31:09 +02:00
return - EINVAL ;
val = ucontrol - > value . enumerated . item [ 0 ] < < e - > shift_l ;
mask = ( bitmask - 1 ) < < e - > shift_l ;
if ( e - > shift_l ! = e - > shift_r ) {
2008-07-29 11:42:27 +01:00
if ( ucontrol - > value . enumerated . item [ 1 ] > e - > max - 1 )
2006-10-06 18:31:09 +02:00
return - EINVAL ;
val | = ucontrol - > value . enumerated . item [ 1 ] < < e - > shift_r ;
mask | = ( bitmask - 1 ) < < e - > shift_r ;
}
return snd_soc_update_bits ( codec , e - > reg , mask , val ) ;
}
EXPORT_SYMBOL_GPL ( snd_soc_put_enum_double ) ;
/**
* snd_soc_info_enum_ext - external enumerated single mixer info callback
* @ kcontrol : mixer control
* @ uinfo : control element information
*
* Callback to provide information about an external enumerated
* single mixer .
*
* Returns 0 for success .
*/
int snd_soc_info_enum_ext ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
{
struct soc_enum * e = ( struct soc_enum * ) kcontrol - > private_value ;
uinfo - > type = SNDRV_CTL_ELEM_TYPE_ENUMERATED ;
uinfo - > count = 1 ;
2008-07-29 11:42:27 +01:00
uinfo - > value . enumerated . items = e - > max ;
2006-10-06 18:31:09 +02:00
2008-07-29 11:42:27 +01:00
if ( uinfo - > value . enumerated . item > e - > max - 1 )
uinfo - > value . enumerated . item = e - > max - 1 ;
2006-10-06 18:31:09 +02:00
strcpy ( uinfo - > value . enumerated . name ,
e - > texts [ uinfo - > value . enumerated . item ] ) ;
return 0 ;
}
EXPORT_SYMBOL_GPL ( snd_soc_info_enum_ext ) ;
/**
* snd_soc_info_volsw_ext - external single mixer info callback
* @ kcontrol : mixer control
* @ uinfo : control element information
*
* Callback to provide information about a single external mixer control .
*
* Returns 0 for success .
*/
int snd_soc_info_volsw_ext ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
{
2008-01-10 14:37:42 +01:00
int max = kcontrol - > private_value ;
if ( max = = 1 )
uinfo - > type = SNDRV_CTL_ELEM_TYPE_BOOLEAN ;
else
uinfo - > type = SNDRV_CTL_ELEM_TYPE_INTEGER ;
2006-10-06 18:31:09 +02:00
uinfo - > count = 1 ;
uinfo - > value . integer . min = 0 ;
2008-01-10 14:37:42 +01:00
uinfo - > value . integer . max = max ;
2006-10-06 18:31:09 +02:00
return 0 ;
}
EXPORT_SYMBOL_GPL ( snd_soc_info_volsw_ext ) ;
/**
* snd_soc_info_volsw - single mixer info callback
* @ kcontrol : mixer control
* @ uinfo : control element information
*
* Callback to provide information about a single mixer control .
*
* Returns 0 for success .
*/
int snd_soc_info_volsw ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
{
2008-07-29 11:42:26 +01:00
struct soc_mixer_control * mc =
( struct soc_mixer_control * ) kcontrol - > private_value ;
int max = mc - > max ;
2008-10-30 12:37:08 +00:00
unsigned int shift = mc - > shift ;
2008-07-29 10:22:24 -04:00
unsigned int rshift = mc - > rshift ;
2006-10-06 18:31:09 +02:00
2008-01-10 14:37:42 +01:00
if ( max = = 1 )
uinfo - > type = SNDRV_CTL_ELEM_TYPE_BOOLEAN ;
else
uinfo - > type = SNDRV_CTL_ELEM_TYPE_INTEGER ;
2006-10-06 18:31:09 +02:00
uinfo - > count = shift = = rshift ? 1 : 2 ;
uinfo - > value . integer . min = 0 ;
2008-01-10 14:37:42 +01:00
uinfo - > value . integer . max = max ;
2006-10-06 18:31:09 +02:00
return 0 ;
}
EXPORT_SYMBOL_GPL ( snd_soc_info_volsw ) ;
/**
* snd_soc_get_volsw - single mixer get callback
* @ kcontrol : mixer control
* @ uinfo : control element information
*
* Callback to get the value of a single mixer control .
*
* Returns 0 for success .
*/
int snd_soc_get_volsw ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
2008-07-29 11:42:26 +01:00
struct soc_mixer_control * mc =
( struct soc_mixer_control * ) kcontrol - > private_value ;
2006-10-06 18:31:09 +02:00
struct snd_soc_codec * codec = snd_kcontrol_chip ( kcontrol ) ;
2008-07-29 10:22:24 -04:00
unsigned int reg = mc - > reg ;
unsigned int shift = mc - > shift ;
unsigned int rshift = mc - > rshift ;
2008-07-29 11:42:26 +01:00
int max = mc - > max ;
2008-07-29 10:22:24 -04:00
unsigned int mask = ( 1 < < fls ( max ) ) - 1 ;
unsigned int invert = mc - > invert ;
2006-10-06 18:31:09 +02:00
ucontrol - > value . integer . value [ 0 ] =
( snd_soc_read ( codec , reg ) > > shift ) & mask ;
if ( shift ! = rshift )
ucontrol - > value . integer . value [ 1 ] =
( snd_soc_read ( codec , reg ) > > rshift ) & mask ;
if ( invert ) {
ucontrol - > value . integer . value [ 0 ] =
2008-01-10 14:37:42 +01:00
max - ucontrol - > value . integer . value [ 0 ] ;
2006-10-06 18:31:09 +02:00
if ( shift ! = rshift )
ucontrol - > value . integer . value [ 1 ] =
2008-01-10 14:37:42 +01:00
max - ucontrol - > value . integer . value [ 1 ] ;
2006-10-06 18:31:09 +02:00
}
return 0 ;
}
EXPORT_SYMBOL_GPL ( snd_soc_get_volsw ) ;
/**
* snd_soc_put_volsw - single mixer put callback
* @ kcontrol : mixer control
* @ uinfo : control element information
*
* Callback to set the value of a single mixer control .
*
* Returns 0 for success .
*/
int snd_soc_put_volsw ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
2008-07-29 11:42:26 +01:00
struct soc_mixer_control * mc =
( struct soc_mixer_control * ) kcontrol - > private_value ;
2006-10-06 18:31:09 +02:00
struct snd_soc_codec * codec = snd_kcontrol_chip ( kcontrol ) ;
2008-07-29 10:22:24 -04:00
unsigned int reg = mc - > reg ;
unsigned int shift = mc - > shift ;
unsigned int rshift = mc - > rshift ;
2008-07-29 11:42:26 +01:00
int max = mc - > max ;
2008-07-29 10:22:24 -04:00
unsigned int mask = ( 1 < < fls ( max ) ) - 1 ;
unsigned int invert = mc - > invert ;
2006-10-06 18:31:09 +02:00
unsigned short val , val2 , val_mask ;
val = ( ucontrol - > value . integer . value [ 0 ] & mask ) ;
if ( invert )
2008-01-10 14:37:42 +01:00
val = max - val ;
2006-10-06 18:31:09 +02:00
val_mask = mask < < shift ;
val = val < < shift ;
if ( shift ! = rshift ) {
val2 = ( ucontrol - > value . integer . value [ 1 ] & mask ) ;
if ( invert )
2008-01-10 14:37:42 +01:00
val2 = max - val2 ;
2006-10-06 18:31:09 +02:00
val_mask | = mask < < rshift ;
val | = val2 < < rshift ;
}
2008-01-10 14:37:42 +01:00
return snd_soc_update_bits ( codec , reg , val_mask , val ) ;
2006-10-06 18:31:09 +02:00
}
EXPORT_SYMBOL_GPL ( snd_soc_put_volsw ) ;
/**
* snd_soc_info_volsw_2r - double mixer info callback
* @ kcontrol : mixer control
* @ uinfo : control element information
*
* Callback to provide information about a double mixer control that
* spans 2 codec registers .
*
* Returns 0 for success .
*/
int snd_soc_info_volsw_2r ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
{
2008-07-29 11:42:26 +01:00
struct soc_mixer_control * mc =
( struct soc_mixer_control * ) kcontrol - > private_value ;
int max = mc - > max ;
2008-01-10 14:37:42 +01:00
if ( max = = 1 )
uinfo - > type = SNDRV_CTL_ELEM_TYPE_BOOLEAN ;
else
uinfo - > type = SNDRV_CTL_ELEM_TYPE_INTEGER ;
2006-10-06 18:31:09 +02:00
uinfo - > count = 2 ;
uinfo - > value . integer . min = 0 ;
2008-01-10 14:37:42 +01:00
uinfo - > value . integer . max = max ;
2006-10-06 18:31:09 +02:00
return 0 ;
}
EXPORT_SYMBOL_GPL ( snd_soc_info_volsw_2r ) ;
/**
* snd_soc_get_volsw_2r - double mixer get callback
* @ kcontrol : mixer control
* @ uinfo : control element information
*
* Callback to get the value of a double mixer control that spans 2 registers .
*
* Returns 0 for success .
*/
int snd_soc_get_volsw_2r ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
2008-07-29 11:42:26 +01:00
struct soc_mixer_control * mc =
( struct soc_mixer_control * ) kcontrol - > private_value ;
2006-10-06 18:31:09 +02:00
struct snd_soc_codec * codec = snd_kcontrol_chip ( kcontrol ) ;
2008-07-29 10:22:24 -04:00
unsigned int reg = mc - > reg ;
unsigned int reg2 = mc - > rreg ;
unsigned int shift = mc - > shift ;
2008-07-29 11:42:26 +01:00
int max = mc - > max ;
2008-07-29 10:22:24 -04:00
unsigned int mask = ( 1 < < fls ( max ) ) - 1 ;
unsigned int invert = mc - > invert ;
2006-10-06 18:31:09 +02:00
ucontrol - > value . integer . value [ 0 ] =
( snd_soc_read ( codec , reg ) > > shift ) & mask ;
ucontrol - > value . integer . value [ 1 ] =
( snd_soc_read ( codec , reg2 ) > > shift ) & mask ;
if ( invert ) {
ucontrol - > value . integer . value [ 0 ] =
2008-01-10 14:37:42 +01:00
max - ucontrol - > value . integer . value [ 0 ] ;
2006-10-06 18:31:09 +02:00
ucontrol - > value . integer . value [ 1 ] =
2008-01-10 14:37:42 +01:00
max - ucontrol - > value . integer . value [ 1 ] ;
2006-10-06 18:31:09 +02:00
}
return 0 ;
}
EXPORT_SYMBOL_GPL ( snd_soc_get_volsw_2r ) ;
/**
* snd_soc_put_volsw_2r - double mixer set callback
* @ kcontrol : mixer control
* @ uinfo : control element information
*
* Callback to set the value of a double mixer control that spans 2 registers .
*
* Returns 0 for success .
*/
int snd_soc_put_volsw_2r ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
2008-07-29 11:42:26 +01:00
struct soc_mixer_control * mc =
( struct soc_mixer_control * ) kcontrol - > private_value ;
2006-10-06 18:31:09 +02:00
struct snd_soc_codec * codec = snd_kcontrol_chip ( kcontrol ) ;
2008-07-29 10:22:24 -04:00
unsigned int reg = mc - > reg ;
unsigned int reg2 = mc - > rreg ;
unsigned int shift = mc - > shift ;
2008-07-29 11:42:26 +01:00
int max = mc - > max ;
2008-07-29 10:22:24 -04:00
unsigned int mask = ( 1 < < fls ( max ) ) - 1 ;
unsigned int invert = mc - > invert ;
2006-10-06 18:31:09 +02:00
int err ;
unsigned short val , val2 , val_mask ;
val_mask = mask < < shift ;
val = ( ucontrol - > value . integer . value [ 0 ] & mask ) ;
val2 = ( ucontrol - > value . integer . value [ 1 ] & mask ) ;
if ( invert ) {
2008-01-10 14:37:42 +01:00
val = max - val ;
val2 = max - val2 ;
2006-10-06 18:31:09 +02:00
}
val = val < < shift ;
val2 = val2 < < shift ;
2008-05-19 12:32:25 +02:00
err = snd_soc_update_bits ( codec , reg , val_mask , val ) ;
if ( err < 0 )
2006-10-06 18:31:09 +02:00
return err ;
err = snd_soc_update_bits ( codec , reg2 , val_mask , val2 ) ;
return err ;
}
EXPORT_SYMBOL_GPL ( snd_soc_put_volsw_2r ) ;
2008-05-28 17:58:05 +01:00
/**
* snd_soc_info_volsw_s8 - signed mixer info callback
* @ kcontrol : mixer control
* @ uinfo : control element information
*
* Callback to provide information about a signed mixer control .
*
* Returns 0 for success .
*/
int snd_soc_info_volsw_s8 ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
{
2008-07-29 11:42:26 +01:00
struct soc_mixer_control * mc =
( struct soc_mixer_control * ) kcontrol - > private_value ;
int max = mc - > max ;
int min = mc - > min ;
2008-05-28 17:58:05 +01:00
uinfo - > type = SNDRV_CTL_ELEM_TYPE_INTEGER ;
uinfo - > count = 2 ;
uinfo - > value . integer . min = 0 ;
uinfo - > value . integer . max = max - min ;
return 0 ;
}
EXPORT_SYMBOL_GPL ( snd_soc_info_volsw_s8 ) ;
/**
* snd_soc_get_volsw_s8 - signed mixer get callback
* @ kcontrol : mixer control
* @ uinfo : control element information
*
* Callback to get the value of a signed mixer control .
*
* Returns 0 for success .
*/
int snd_soc_get_volsw_s8 ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
2008-07-29 11:42:26 +01:00
struct soc_mixer_control * mc =
( struct soc_mixer_control * ) kcontrol - > private_value ;
2008-05-28 17:58:05 +01:00
struct snd_soc_codec * codec = snd_kcontrol_chip ( kcontrol ) ;
2008-07-29 10:22:24 -04:00
unsigned int reg = mc - > reg ;
2008-07-29 11:42:26 +01:00
int min = mc - > min ;
2008-05-28 17:58:05 +01:00
int val = snd_soc_read ( codec , reg ) ;
ucontrol - > value . integer . value [ 0 ] =
( ( signed char ) ( val & 0xff ) ) - min ;
ucontrol - > value . integer . value [ 1 ] =
( ( signed char ) ( ( val > > 8 ) & 0xff ) ) - min ;
return 0 ;
}
EXPORT_SYMBOL_GPL ( snd_soc_get_volsw_s8 ) ;
/**
* snd_soc_put_volsw_sgn - signed mixer put callback
* @ kcontrol : mixer control
* @ uinfo : control element information
*
* Callback to set the value of a signed mixer control .
*
* Returns 0 for success .
*/
int snd_soc_put_volsw_s8 ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
2008-07-29 11:42:26 +01:00
struct soc_mixer_control * mc =
( struct soc_mixer_control * ) kcontrol - > private_value ;
2008-05-28 17:58:05 +01:00
struct snd_soc_codec * codec = snd_kcontrol_chip ( kcontrol ) ;
2008-07-29 10:22:24 -04:00
unsigned int reg = mc - > reg ;
2008-07-29 11:42:26 +01:00
int min = mc - > min ;
2008-05-28 17:58:05 +01:00
unsigned short val ;
val = ( ucontrol - > value . integer . value [ 0 ] + min ) & 0xff ;
val | = ( ( ucontrol - > value . integer . value [ 1 ] + min ) & 0xff ) < < 8 ;
return snd_soc_update_bits ( codec , reg , 0xffff , val ) ;
}
EXPORT_SYMBOL_GPL ( snd_soc_put_volsw_s8 ) ;
2008-07-08 13:19:13 +01:00
/**
* snd_soc_dai_set_sysclk - configure DAI system or master clock .
* @ dai : DAI
* @ clk_id : DAI specific clock ID
* @ freq : new clock frequency in Hz
* @ dir : new clock direction - input / output .
*
* Configures the DAI master ( MCLK ) or system ( SYSCLK ) clocking .
*/
int snd_soc_dai_set_sysclk ( struct snd_soc_dai * dai , int clk_id ,
unsigned int freq , int dir )
{
2008-11-18 22:11:38 +00:00
if ( dai - > ops . set_sysclk )
return dai - > ops . set_sysclk ( dai , clk_id , freq , dir ) ;
2008-07-08 13:19:13 +01:00
else
return - EINVAL ;
}
EXPORT_SYMBOL_GPL ( snd_soc_dai_set_sysclk ) ;
/**
* snd_soc_dai_set_clkdiv - configure DAI clock dividers .
* @ dai : DAI
* @ clk_id : DAI specific clock divider ID
* @ div : new clock divisor .
*
* Configures the clock dividers . This is used to derive the best DAI bit and
* frame clocks from the system or master clock . It ' s best to set the DAI bit
* and frame clocks as low as possible to save system power .
*/
int snd_soc_dai_set_clkdiv ( struct snd_soc_dai * dai ,
int div_id , int div )
{
2008-11-18 22:11:38 +00:00
if ( dai - > ops . set_clkdiv )
return dai - > ops . set_clkdiv ( dai , div_id , div ) ;
2008-07-08 13:19:13 +01:00
else
return - EINVAL ;
}
EXPORT_SYMBOL_GPL ( snd_soc_dai_set_clkdiv ) ;
/**
* snd_soc_dai_set_pll - configure DAI PLL .
* @ dai : DAI
* @ pll_id : DAI specific PLL ID
* @ freq_in : PLL input clock frequency in Hz
* @ freq_out : requested PLL output clock frequency in Hz
*
* Configures and enables PLL to generate output clock based on input clock .
*/
int snd_soc_dai_set_pll ( struct snd_soc_dai * dai ,
int pll_id , unsigned int freq_in , unsigned int freq_out )
{
2008-11-18 22:11:38 +00:00
if ( dai - > ops . set_pll )
return dai - > ops . set_pll ( dai , pll_id , freq_in , freq_out ) ;
2008-07-08 13:19:13 +01:00
else
return - EINVAL ;
}
EXPORT_SYMBOL_GPL ( snd_soc_dai_set_pll ) ;
/**
* snd_soc_dai_set_fmt - configure DAI hardware audio format .
* @ dai : DAI
* @ fmt : SND_SOC_DAIFMT_ format value .
*
* Configures the DAI hardware format and clocking .
*/
int snd_soc_dai_set_fmt ( struct snd_soc_dai * dai , unsigned int fmt )
{
2008-11-18 22:11:38 +00:00
if ( dai - > ops . set_fmt )
return dai - > ops . set_fmt ( dai , fmt ) ;
2008-07-08 13:19:13 +01:00
else
return - EINVAL ;
}
EXPORT_SYMBOL_GPL ( snd_soc_dai_set_fmt ) ;
/**
* snd_soc_dai_set_tdm_slot - configure DAI TDM .
* @ dai : DAI
* @ mask : DAI specific mask representing used slots .
* @ slots : Number of slots in use .
*
* Configures a DAI for TDM operation . Both mask and slots are codec and DAI
* specific .
*/
int snd_soc_dai_set_tdm_slot ( struct snd_soc_dai * dai ,
unsigned int mask , int slots )
{
2008-11-18 22:11:38 +00:00
if ( dai - > ops . set_sysclk )
return dai - > ops . set_tdm_slot ( dai , mask , slots ) ;
2008-07-08 13:19:13 +01:00
else
return - EINVAL ;
}
EXPORT_SYMBOL_GPL ( snd_soc_dai_set_tdm_slot ) ;
/**
* snd_soc_dai_set_tristate - configure DAI system or master clock .
* @ dai : DAI
* @ tristate : tristate enable
*
* Tristates the DAI so that others can use it .
*/
int snd_soc_dai_set_tristate ( struct snd_soc_dai * dai , int tristate )
{
2008-11-18 22:11:38 +00:00
if ( dai - > ops . set_sysclk )
return dai - > ops . set_tristate ( dai , tristate ) ;
2008-07-08 13:19:13 +01:00
else
return - EINVAL ;
}
EXPORT_SYMBOL_GPL ( snd_soc_dai_set_tristate ) ;
/**
* snd_soc_dai_digital_mute - configure DAI system or master clock .
* @ dai : DAI
* @ mute : mute enable
*
* Mutes the DAI DAC .
*/
int snd_soc_dai_digital_mute ( struct snd_soc_dai * dai , int mute )
{
2008-11-18 22:11:38 +00:00
if ( dai - > ops . digital_mute )
return dai - > ops . digital_mute ( dai , mute ) ;
2008-07-08 13:19:13 +01:00
else
return - EINVAL ;
}
EXPORT_SYMBOL_GPL ( snd_soc_dai_digital_mute ) ;
2006-10-06 18:31:09 +02:00
static int __devinit snd_soc_init ( void )
{
return platform_driver_register ( & soc_driver ) ;
}
2008-11-30 22:11:24 +00:00
static void __exit snd_soc_exit ( void )
2006-10-06 18:31:09 +02:00
{
2008-05-19 12:32:25 +02:00
platform_driver_unregister ( & soc_driver ) ;
2006-10-06 18:31:09 +02:00
}
module_init ( snd_soc_init ) ;
module_exit ( snd_soc_exit ) ;
/* Module information */
2008-10-12 13:17:36 +01:00
MODULE_AUTHOR ( " Liam Girdwood, lrg@slimlogic.co.uk " ) ;
2006-10-06 18:31:09 +02:00
MODULE_DESCRIPTION ( " ALSA SoC Core " ) ;
MODULE_LICENSE ( " GPL " ) ;
2008-04-14 13:33:36 +02:00
MODULE_ALIAS ( " platform:soc-audio " ) ;