2008-04-25 13:55:19 +02:00
/*
* omap - mcbsp . c - - OMAP ALSA SoC DAI driver using McBSP port
*
* Copyright ( C ) 2008 Nokia Corporation
*
2011-08-11 15:44:57 +03:00
* Contact : Jarkko Nikula < jarkko . nikula @ bitmer . com >
2011-05-03 18:14:06 +03:00
* Peter Ujfalusi < peter . ujfalusi @ ti . com >
2008-04-25 13:55:19 +02:00
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* version 2 as published by the Free Software Foundation .
*
* This program is distributed in the hope that it will be useful , but
* WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
* General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 51 Franklin St , Fifth Floor , Boston , MA
* 02110 - 1301 USA
*
*/
# include <linux/init.h>
# include <linux/module.h>
# include <linux/device.h>
2012-02-14 14:52:42 +02:00
# include <linux/pm_runtime.h>
2008-04-25 13:55:19 +02:00
# include <sound/core.h>
# include <sound/pcm.h>
# include <sound/pcm_params.h>
# include <sound/initval.h>
# include <sound/soc.h>
2012-08-27 17:43:01 -07:00
# include <plat/cpu.h>
2009-10-20 09:40:47 -07:00
# include <plat/dma.h>
# include <plat/mcbsp.h>
2012-02-03 13:11:47 +02:00
# include "mcbsp.h"
2008-04-25 13:55:19 +02:00
# include "omap-mcbsp.h"
# include "omap-pcm.h"
2008-11-12 17:05:51 +02:00
# define OMAP_MCBSP_RATES (SNDRV_PCM_RATE_8000_96000)
2008-04-25 13:55:19 +02:00
2010-02-22 12:21:12 +00:00
# define OMAP_MCBSP_SOC_SINGLE_S16_EXT(xname, xmin, xmax, \
xhandler_get , xhandler_put ) \
{ . iface = SNDRV_CTL_ELEM_IFACE_MIXER , . name = xname , \
. info = omap_mcbsp_st_info_volsw , \
. get = xhandler_get , . put = xhandler_put , \
. private_value = ( unsigned long ) & ( struct soc_mixer_control ) \
{ . min = xmin , . max = xmax } }
2012-02-03 13:11:47 +02:00
enum {
OMAP_MCBSP_WORD_8 = 0 ,
OMAP_MCBSP_WORD_12 ,
OMAP_MCBSP_WORD_16 ,
OMAP_MCBSP_WORD_20 ,
OMAP_MCBSP_WORD_24 ,
OMAP_MCBSP_WORD_32 ,
} ;
2008-04-25 13:55:19 +02:00
/*
* Stream DMA parameters . DMA request line and port address are set runtime
* since they are different between OMAP1 and later OMAPs
*/
2009-08-20 16:18:25 +03:00
static void omap_mcbsp_set_threshold ( 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 * cpu_dai = rtd - > cpu_dai ;
2012-02-14 18:20:58 +02:00
struct omap_mcbsp * mcbsp = snd_soc_dai_get_drvdata ( cpu_dai ) ;
2010-07-29 09:51:27 +03:00
struct omap_pcm_dma_data * dma_data ;
2010-06-03 07:39:35 +03:00
int words ;
2009-08-20 16:18:26 +03:00
2010-03-17 20:15:21 +00:00
dma_data = snd_soc_dai_get_dma_data ( rtd - > cpu_dai , substream ) ;
2010-07-29 09:51:27 +03:00
2012-03-15 12:20:32 +02:00
/*
* Configure McBSP threshold based on either :
* packet_size , when the sDMA is in packet mode , or based on the
* period size in THRESHOLD mode , otherwise use McBSP threshold = 1
* for mono streams .
*/
if ( dma_data - > packet_size )
words = dma_data - > packet_size ;
else if ( mcbsp - > dma_op_mode = = MCBSP_DMA_MODE_THRESHOLD )
words = snd_pcm_lib_period_bytes ( substream ) /
( mcbsp - > wlen / 8 ) ;
2009-08-20 16:18:26 +03:00
else
2010-06-03 07:39:35 +03:00
words = 1 ;
2009-08-20 16:18:25 +03:00
/* Configure McBSP internal buffer usage */
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK )
2012-02-14 18:20:58 +02:00
omap_mcbsp_set_tx_threshold ( mcbsp , words ) ;
2009-08-20 16:18:25 +03:00
else
2012-02-14 18:20:58 +02:00
omap_mcbsp_set_rx_threshold ( mcbsp , words ) ;
2009-08-20 16:18:25 +03:00
}
2010-06-03 07:39:36 +03:00
static int omap_mcbsp_hwrule_min_buffersize ( struct snd_pcm_hw_params * params ,
struct snd_pcm_hw_rule * rule )
{
struct snd_interval * buffer_size = hw_param_interval ( params ,
SNDRV_PCM_HW_PARAM_BUFFER_SIZE ) ;
struct snd_interval * channels = hw_param_interval ( params ,
SNDRV_PCM_HW_PARAM_CHANNELS ) ;
2012-02-14 18:20:58 +02:00
struct omap_mcbsp * mcbsp = rule - > private ;
2010-06-03 07:39:36 +03:00
struct snd_interval frames ;
int size ;
snd_interval_any ( & frames ) ;
2012-02-13 16:26:54 +02:00
size = mcbsp - > pdata - > buffer_size ;
2010-06-03 07:39:36 +03:00
frames . min = size / channels - > min ;
frames . integer = 1 ;
return snd_interval_refine ( buffer_size , & frames ) ;
}
2008-11-18 22:11:38 +00:00
static int omap_mcbsp_dai_startup ( struct snd_pcm_substream * substream ,
2010-03-17 20:15:21 +00:00
struct snd_soc_dai * cpu_dai )
2008-04-25 13:55:19 +02:00
{
2012-02-14 18:20:58 +02:00
struct omap_mcbsp * mcbsp = snd_soc_dai_get_drvdata ( cpu_dai ) ;
2008-04-25 13:55:19 +02:00
int err = 0 ;
2009-08-20 16:18:25 +03:00
if ( ! cpu_dai - > active )
2012-02-14 18:20:58 +02:00
err = omap_mcbsp_request ( mcbsp ) ;
2009-08-20 16:18:25 +03:00
2010-06-03 07:39:36 +03:00
/*
* OMAP3 McBSP FIFO is word structured .
* McBSP2 has 1024 + 256 = 1280 word long buffer ,
* McBSP1 , 3 , 4 , 5 has 128 word long buffer
* This means that the size of the FIFO depends on the sample format .
* For example on McBSP3 :
* 16 bit samples : size is 128 * 2 = 256 bytes
* 32 bit samples : size is 128 * 4 = 512 bytes
* It is simpler to place constraint for buffer and period based on
* channels .
* McBSP3 as example again ( 16 or 32 bit samples ) :
* 1 channel ( mono ) : size is 128 frames ( 128 words )
* 2 channels ( stereo ) : size is 128 / 2 = 64 frames ( 2 * 64 words )
* 4 channels : size is 128 / 4 = 32 frames ( 4 * 32 words )
*/
2012-02-14 18:20:58 +02:00
if ( mcbsp - > pdata - > buffer_size ) {
2009-03-27 15:32:01 +02:00
/*
2010-07-29 09:51:28 +03:00
* Rule for the buffer size . We should not allow
2012-03-20 11:47:36 +02:00
* smaller buffer than the FIFO size to avoid underruns .
* This applies only for the playback stream .
2010-06-03 07:39:36 +03:00
*/
2012-03-20 11:47:36 +02:00
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK )
snd_pcm_hw_rule_add ( substream - > runtime , 0 ,
SNDRV_PCM_HW_PARAM_BUFFER_SIZE ,
omap_mcbsp_hwrule_min_buffersize ,
mcbsp ,
SNDRV_PCM_HW_PARAM_CHANNELS , - 1 ) ;
2009-08-20 16:18:25 +03:00
2010-07-29 09:51:28 +03:00
/* Make sure, that the period size is always even */
snd_pcm_hw_constraint_step ( substream - > runtime , 0 ,
SNDRV_PCM_HW_PARAM_PERIOD_SIZE , 2 ) ;
2009-03-27 15:32:01 +02:00
}
2008-04-25 13:55:19 +02:00
return err ;
}
2008-11-18 22:11:38 +00:00
static void omap_mcbsp_dai_shutdown ( struct snd_pcm_substream * substream ,
2010-03-17 20:15:21 +00:00
struct snd_soc_dai * cpu_dai )
2008-04-25 13:55:19 +02:00
{
2012-02-14 18:20:58 +02:00
struct omap_mcbsp * mcbsp = snd_soc_dai_get_drvdata ( cpu_dai ) ;
2008-04-25 13:55:19 +02:00
if ( ! cpu_dai - > active ) {
2012-02-14 18:20:58 +02:00
omap_mcbsp_free ( mcbsp ) ;
2012-02-14 15:23:15 +02:00
mcbsp - > configured = 0 ;
2008-04-25 13:55:19 +02:00
}
}
2008-11-18 22:11:38 +00:00
static int omap_mcbsp_dai_trigger ( struct snd_pcm_substream * substream , int cmd ,
2010-03-17 20:15:21 +00:00
struct snd_soc_dai * cpu_dai )
2008-04-25 13:55:19 +02:00
{
2012-02-14 18:20:58 +02:00
struct omap_mcbsp * mcbsp = snd_soc_dai_get_drvdata ( cpu_dai ) ;
2009-08-07 09:59:47 +03:00
int err = 0 , play = ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK ) ;
2008-04-25 13:55:19 +02:00
switch ( cmd ) {
case SNDRV_PCM_TRIGGER_START :
case SNDRV_PCM_TRIGGER_RESUME :
case SNDRV_PCM_TRIGGER_PAUSE_RELEASE :
2012-02-14 15:23:15 +02:00
mcbsp - > active + + ;
2012-02-14 18:20:58 +02:00
omap_mcbsp_start ( mcbsp , play , ! play ) ;
2008-04-25 13:55:19 +02:00
break ;
case SNDRV_PCM_TRIGGER_STOP :
case SNDRV_PCM_TRIGGER_SUSPEND :
case SNDRV_PCM_TRIGGER_PAUSE_PUSH :
2012-02-14 18:20:58 +02:00
omap_mcbsp_stop ( mcbsp , play , ! play ) ;
2012-02-14 15:23:15 +02:00
mcbsp - > active - - ;
2008-04-25 13:55:19 +02:00
break ;
default :
err = - EINVAL ;
}
return err ;
}
2010-03-03 15:08:09 +02:00
static snd_pcm_sframes_t omap_mcbsp_dai_delay (
struct snd_pcm_substream * substream ,
struct snd_soc_dai * dai )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
2010-03-17 20:15:21 +00:00
struct snd_soc_dai * cpu_dai = rtd - > cpu_dai ;
2012-02-14 18:20:58 +02:00
struct omap_mcbsp * mcbsp = snd_soc_dai_get_drvdata ( cpu_dai ) ;
2010-03-03 15:08:09 +02:00
u16 fifo_use ;
snd_pcm_sframes_t delay ;
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK )
2012-02-14 18:20:58 +02:00
fifo_use = omap_mcbsp_get_tx_delay ( mcbsp ) ;
2010-03-03 15:08:09 +02:00
else
2012-02-14 18:20:58 +02:00
fifo_use = omap_mcbsp_get_rx_delay ( mcbsp ) ;
2010-03-03 15:08:09 +02:00
/*
* Divide the used locations with the channel count to get the
* FIFO usage in samples ( don ' t care about partial samples in the
* buffer ) .
*/
delay = fifo_use / substream - > runtime - > channels ;
return delay ;
}
2008-04-25 13:55:19 +02:00
static int omap_mcbsp_dai_hw_params ( struct snd_pcm_substream * substream ,
2008-11-18 22:11:38 +00:00
struct snd_pcm_hw_params * params ,
2010-03-17 20:15:21 +00:00
struct snd_soc_dai * cpu_dai )
2008-04-25 13:55:19 +02:00
{
2012-02-14 18:20:58 +02:00
struct omap_mcbsp * mcbsp = snd_soc_dai_get_drvdata ( cpu_dai ) ;
2012-02-14 15:23:15 +02:00
struct omap_mcbsp_reg_cfg * regs = & mcbsp - > cfg_regs ;
2010-07-29 09:51:26 +03:00
struct omap_pcm_dma_data * dma_data ;
2009-08-20 16:18:25 +03:00
int wlen , channels , wpf , sync_mode = OMAP_DMA_SYNC_ELEMENT ;
2010-07-29 09:51:27 +03:00
int pkt_size = 0 ;
2009-11-09 19:02:15 +00:00
unsigned int format , div , framesize , master ;
2008-04-25 13:55:19 +02:00
2012-02-14 15:23:15 +02:00
dma_data = & mcbsp - > dma_data [ substream - > stream ] ;
2012-03-15 12:20:32 +02:00
channels = params_channels ( params ) ;
2011-02-24 15:16:56 +05:30
2010-05-13 19:48:16 +04:00
switch ( params_format ( params ) ) {
case SNDRV_PCM_FORMAT_S16_LE :
2010-07-29 09:51:26 +03:00
dma_data - > data_type = OMAP_DMA_DATA_TYPE_S16 ;
2010-07-29 09:51:27 +03:00
wlen = 16 ;
2010-05-13 19:48:16 +04:00
break ;
case SNDRV_PCM_FORMAT_S32_LE :
2010-07-29 09:51:26 +03:00
dma_data - > data_type = OMAP_DMA_DATA_TYPE_S32 ;
2010-07-29 09:51:27 +03:00
wlen = 32 ;
2010-05-13 19:48:16 +04:00
break ;
default :
return - EINVAL ;
}
2012-02-14 18:20:58 +02:00
if ( mcbsp - > pdata - > buffer_size ) {
2010-07-29 09:51:25 +03:00
dma_data - > set_threshold = omap_mcbsp_set_threshold ;
2012-02-13 16:26:54 +02:00
if ( mcbsp - > dma_op_mode = = MCBSP_DMA_MODE_THRESHOLD ) {
2010-07-29 09:51:27 +03:00
int period_words , max_thrsh ;
period_words = params_period_bytes ( params ) / ( wlen / 8 ) ;
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK )
2012-02-13 16:26:54 +02:00
max_thrsh = mcbsp - > max_tx_thres ;
2010-07-29 09:51:27 +03:00
else
2012-02-13 16:26:54 +02:00
max_thrsh = mcbsp - > max_rx_thres ;
2010-07-29 09:51:27 +03:00
/*
* If the period contains less or equal number of words ,
* we are using the original threshold mode setup :
* McBSP threshold = sDMA frame size = period_size
* Otherwise we switch to sDMA packet mode :
* McBSP threshold = sDMA packet size
* sDMA frame size = period size
*/
if ( period_words > max_thrsh ) {
int divider = 0 ;
/*
* Look for the biggest threshold value , which
* divides the period size evenly .
*/
divider = period_words / max_thrsh ;
if ( period_words % max_thrsh )
divider + + ;
while ( period_words % divider & &
divider < period_words )
divider + + ;
if ( divider = = period_words )
return - EINVAL ;
pkt_size = period_words / divider ;
sync_mode = OMAP_DMA_SYNC_PACKET ;
} else {
sync_mode = OMAP_DMA_SYNC_FRAME ;
}
2012-03-15 12:20:32 +02:00
} else if ( channels > 1 ) {
/* Use packet mode for non mono streams */
pkt_size = channels ;
sync_mode = OMAP_DMA_SYNC_PACKET ;
2010-07-29 09:51:27 +03:00
}
2010-07-29 09:51:25 +03:00
}
dma_data - > sync_mode = sync_mode ;
2010-07-29 09:51:27 +03:00
dma_data - > packet_size = pkt_size ;
2010-03-19 14:52:55 +00:00
2010-07-29 09:51:26 +03:00
snd_soc_dai_set_dma_data ( cpu_dai , substream , dma_data ) ;
2008-04-25 13:55:19 +02:00
2012-02-14 15:23:15 +02:00
if ( mcbsp - > configured ) {
2008-04-25 13:55:19 +02:00
/* McBSP already configured by another stream */
return 0 ;
}
2011-09-30 16:07:44 +03:00
regs - > rcr2 & = ~ ( RPHASE | RFRLEN2 ( 0x7f ) | RWDLEN2 ( 7 ) ) ;
regs - > xcr2 & = ~ ( RPHASE | XFRLEN2 ( 0x7f ) | XWDLEN2 ( 7 ) ) ;
regs - > rcr1 & = ~ ( RFRLEN1 ( 0x7f ) | RWDLEN1 ( 7 ) ) ;
regs - > xcr1 & = ~ ( XFRLEN1 ( 0x7f ) | XWDLEN1 ( 7 ) ) ;
2012-02-14 15:23:15 +02:00
format = mcbsp - > fmt & SND_SOC_DAIFMT_FORMAT_MASK ;
2012-03-15 12:20:32 +02:00
wpf = channels ;
2010-03-19 12:27:31 +02:00
if ( channels = = 2 & & ( format = = SND_SOC_DAIFMT_I2S | |
format = = SND_SOC_DAIFMT_LEFT_J ) ) {
2009-11-09 19:02:15 +00:00
/* Use dual-phase frames */
regs - > rcr2 | = RPHASE ;
regs - > xcr2 | = XPHASE ;
/* Set 1 word per (McBSP) frame for phase1 and phase2 */
wpf - - ;
regs - > rcr2 | = RFRLEN2 ( wpf - 1 ) ;
regs - > xcr2 | = XFRLEN2 ( wpf - 1 ) ;
2008-04-25 13:55:19 +02:00
}
2009-11-09 19:02:15 +00:00
regs - > rcr1 | = RFRLEN1 ( wpf - 1 ) ;
regs - > xcr1 | = XFRLEN1 ( wpf - 1 ) ;
2008-04-25 13:55:19 +02:00
switch ( params_format ( params ) ) {
case SNDRV_PCM_FORMAT_S16_LE :
/* Set word lengths */
regs - > rcr2 | = RWDLEN2 ( OMAP_MCBSP_WORD_16 ) ;
regs - > rcr1 | = RWDLEN1 ( OMAP_MCBSP_WORD_16 ) ;
regs - > xcr2 | = XWDLEN2 ( OMAP_MCBSP_WORD_16 ) ;
regs - > xcr1 | = XWDLEN1 ( OMAP_MCBSP_WORD_16 ) ;
break ;
2010-05-13 19:48:16 +04:00
case SNDRV_PCM_FORMAT_S32_LE :
/* Set word lengths */
regs - > rcr2 | = RWDLEN2 ( OMAP_MCBSP_WORD_32 ) ;
regs - > rcr1 | = RWDLEN1 ( OMAP_MCBSP_WORD_32 ) ;
regs - > xcr2 | = XWDLEN2 ( OMAP_MCBSP_WORD_32 ) ;
regs - > xcr1 | = XWDLEN1 ( OMAP_MCBSP_WORD_32 ) ;
break ;
2008-04-25 13:55:19 +02:00
default :
/* Unsupported PCM format */
return - EINVAL ;
}
2009-11-09 19:02:15 +00:00
/* In McBSP master modes, FRAME (i.e. sample rate) is generated
* by _counting_ BCLKs . Calculate frame size in BCLKs */
2012-02-14 15:23:15 +02:00
master = mcbsp - > fmt & SND_SOC_DAIFMT_MASTER_MASK ;
2009-11-09 19:02:15 +00:00
if ( master = = SND_SOC_DAIFMT_CBS_CFS ) {
2012-02-14 15:23:15 +02:00
div = mcbsp - > clk_div ? mcbsp - > clk_div : 1 ;
framesize = ( mcbsp - > in_freq / div ) / params_rate ( params ) ;
2009-11-09 19:02:15 +00:00
if ( framesize < wlen * channels ) {
printk ( KERN_ERR " %s: not enough bandwidth for desired rate and "
" channels \n " , __func__ ) ;
return - EINVAL ;
}
} else
framesize = wlen * channels ;
2008-10-20 15:29:59 +03:00
/* Set FS period and length in terms of bit clock periods */
2011-09-30 16:07:44 +03:00
regs - > srgr2 & = ~ FPER ( 0xfff ) ;
regs - > srgr1 & = ~ FWID ( 0xff ) ;
2009-04-15 15:38:55 +03:00
switch ( format ) {
2008-10-20 15:29:59 +03:00
case SND_SOC_DAIFMT_I2S :
2010-03-19 12:27:31 +02:00
case SND_SOC_DAIFMT_LEFT_J :
2009-11-09 19:02:15 +00:00
regs - > srgr2 | = FPER ( framesize - 1 ) ;
regs - > srgr1 | = FWID ( ( framesize > > 1 ) - 1 ) ;
2008-10-20 15:29:59 +03:00
break ;
2009-04-15 15:38:56 +03:00
case SND_SOC_DAIFMT_DSP_A :
2008-12-22 10:21:36 +02:00
case SND_SOC_DAIFMT_DSP_B :
2009-11-09 19:02:15 +00:00
regs - > srgr2 | = FPER ( framesize - 1 ) ;
2009-04-15 13:48:16 +03:00
regs - > srgr1 | = FWID ( 0 ) ;
2008-10-20 15:29:59 +03:00
break ;
}
2012-02-14 15:23:15 +02:00
omap_mcbsp_config ( mcbsp , & mcbsp - > cfg_regs ) ;
mcbsp - > wlen = wlen ;
mcbsp - > configured = 1 ;
2008-04-25 13:55:19 +02:00
return 0 ;
}
/*
* This must be called before _set_clkdiv and _set_sysclk since McBSP register
* cache is initialized here
*/
2008-07-07 16:08:07 +01:00
static int omap_mcbsp_dai_set_dai_fmt ( struct snd_soc_dai * cpu_dai ,
2008-04-25 13:55:19 +02:00
unsigned int fmt )
{
2012-02-14 18:20:58 +02:00
struct omap_mcbsp * mcbsp = snd_soc_dai_get_drvdata ( cpu_dai ) ;
2012-02-14 15:23:15 +02:00
struct omap_mcbsp_reg_cfg * regs = & mcbsp - > cfg_regs ;
2011-09-30 10:55:32 +03:00
bool inv_fs = false ;
2008-04-25 13:55:19 +02:00
2012-02-14 15:23:15 +02:00
if ( mcbsp - > configured )
2008-04-25 13:55:19 +02:00
return 0 ;
2012-02-14 15:23:15 +02:00
mcbsp - > fmt = fmt ;
2008-04-25 13:55:19 +02:00
memset ( regs , 0 , sizeof ( * regs ) ) ;
/* Generic McBSP register settings */
regs - > spcr2 | = XINTM ( 3 ) | FREE ;
regs - > spcr1 | = RINTM ( 3 ) ;
2009-08-20 16:18:23 +03:00
/* RFIG and XFIG are not defined in 34xx */
2010-12-20 11:32:47 -06:00
if ( ! cpu_is_omap34xx ( ) & & ! cpu_is_omap44xx ( ) ) {
2009-08-20 16:18:23 +03:00
regs - > rcr2 | = RFIG ;
regs - > xcr2 | = XFIG ;
}
2010-12-20 11:32:47 -06:00
if ( cpu_is_omap2430 ( ) | | cpu_is_omap34xx ( ) | | cpu_is_omap44xx ( ) ) {
2009-08-23 12:24:26 +03:00
regs - > xccr = DXENDLY ( 1 ) | XDMAEN | XDISABLE ;
regs - > rccr = RFULL_CYCLE | RDMAEN | RDISABLE ;
2009-01-29 13:29:46 +02:00
}
2008-04-25 13:55:19 +02:00
switch ( fmt & SND_SOC_DAIFMT_FORMAT_MASK ) {
case SND_SOC_DAIFMT_I2S :
/* 1-bit data delay */
regs - > rcr2 | = RDATDLY ( 1 ) ;
regs - > xcr2 | = XDATDLY ( 1 ) ;
break ;
2010-03-19 12:27:31 +02:00
case SND_SOC_DAIFMT_LEFT_J :
/* 0-bit data delay */
regs - > rcr2 | = RDATDLY ( 0 ) ;
regs - > xcr2 | = XDATDLY ( 0 ) ;
regs - > spcr1 | = RJUST ( 2 ) ;
/* Invert FS polarity configuration */
2011-09-30 10:55:32 +03:00
inv_fs = true ;
2010-03-19 12:27:31 +02:00
break ;
2009-04-15 15:38:56 +03:00
case SND_SOC_DAIFMT_DSP_A :
/* 1-bit data delay */
regs - > rcr2 | = RDATDLY ( 1 ) ;
regs - > xcr2 | = XDATDLY ( 1 ) ;
/* Invert FS polarity configuration */
2011-09-30 10:55:32 +03:00
inv_fs = true ;
2009-04-15 15:38:56 +03:00
break ;
2008-12-22 10:21:36 +02:00
case SND_SOC_DAIFMT_DSP_B :
2008-10-02 15:07:06 +05:30
/* 0-bit data delay */
regs - > rcr2 | = RDATDLY ( 0 ) ;
regs - > xcr2 | = XDATDLY ( 0 ) ;
2009-04-15 13:48:16 +03:00
/* Invert FS polarity configuration */
2011-09-30 10:55:32 +03:00
inv_fs = true ;
2008-10-02 15:07:06 +05:30
break ;
2008-04-25 13:55:19 +02:00
default :
/* Unsupported data format */
return - EINVAL ;
}
switch ( fmt & SND_SOC_DAIFMT_MASTER_MASK ) {
case SND_SOC_DAIFMT_CBS_CFS :
/* McBSP master. Set FS and bit clocks as outputs */
regs - > pcr0 | = FSXM | FSRM |
CLKXM | CLKRM ;
/* Sample rate generator drives the FS */
regs - > srgr2 | = FSGM ;
break ;
case SND_SOC_DAIFMT_CBM_CFM :
/* McBSP slave */
break ;
default :
/* Unsupported master/slave configuration */
return - EINVAL ;
}
/* Set bit clock (CLKX/CLKR) and FS polarities */
2011-09-30 10:55:32 +03:00
switch ( fmt & SND_SOC_DAIFMT_INV_MASK ) {
2008-04-25 13:55:19 +02:00
case SND_SOC_DAIFMT_NB_NF :
/*
* Normal BCLK + FS .
* FS active low . TX data driven on falling edge of bit clock
* and RX data sampled on rising edge of bit clock .
*/
regs - > pcr0 | = FSXP | FSRP |
CLKXP | CLKRP ;
break ;
case SND_SOC_DAIFMT_NB_IF :
regs - > pcr0 | = CLKXP | CLKRP ;
break ;
case SND_SOC_DAIFMT_IB_NF :
regs - > pcr0 | = FSXP | FSRP ;
break ;
case SND_SOC_DAIFMT_IB_IF :
break ;
default :
return - EINVAL ;
}
2011-09-30 10:55:32 +03:00
if ( inv_fs = = true )
regs - > pcr0 ^ = FSXP | FSRP ;
2008-04-25 13:55:19 +02:00
return 0 ;
}
2008-07-07 16:08:07 +01:00
static int omap_mcbsp_dai_set_clkdiv ( struct snd_soc_dai * cpu_dai ,
2008-04-25 13:55:19 +02:00
int div_id , int div )
{
2012-02-14 18:20:58 +02:00
struct omap_mcbsp * mcbsp = snd_soc_dai_get_drvdata ( cpu_dai ) ;
2012-02-14 15:23:15 +02:00
struct omap_mcbsp_reg_cfg * regs = & mcbsp - > cfg_regs ;
2008-04-25 13:55:19 +02:00
if ( div_id ! = OMAP_MCBSP_CLKGDV )
return - ENODEV ;
2012-02-14 15:23:15 +02:00
mcbsp - > clk_div = div ;
2011-09-30 16:07:44 +03:00
regs - > srgr1 & = ~ CLKGDV ( 0xff ) ;
2008-04-25 13:55:19 +02:00
regs - > srgr1 | = CLKGDV ( div - 1 ) ;
return 0 ;
}
2008-07-07 16:08:07 +01:00
static int omap_mcbsp_dai_set_dai_sysclk ( struct snd_soc_dai * cpu_dai ,
2008-04-25 13:55:19 +02:00
int clk_id , unsigned int freq ,
int dir )
{
2012-02-14 18:20:58 +02:00
struct omap_mcbsp * mcbsp = snd_soc_dai_get_drvdata ( cpu_dai ) ;
2012-02-14 15:23:15 +02:00
struct omap_mcbsp_reg_cfg * regs = & mcbsp - > cfg_regs ;
2008-04-25 13:55:19 +02:00
int err = 0 ;
2012-02-14 15:23:15 +02:00
if ( mcbsp - > active ) {
if ( freq = = mcbsp - > in_freq )
2011-09-23 11:19:13 +03:00
return 0 ;
else
return - EBUSY ;
2011-09-26 10:56:42 +03:00
}
2011-09-23 11:19:13 +03:00
2012-03-08 13:34:16 +02:00
if ( clk_id = = OMAP_MCBSP_SYSCLK_CLK | |
clk_id = = OMAP_MCBSP_SYSCLK_CLKS_FCLK | |
clk_id = = OMAP_MCBSP_SYSCLK_CLKS_EXT | |
clk_id = = OMAP_MCBSP_SYSCLK_CLKX_EXT | |
clk_id = = OMAP_MCBSP_SYSCLK_CLKR_EXT ) {
mcbsp - > in_freq = freq ;
regs - > srgr2 & = ~ CLKSM ;
regs - > pcr0 & = ~ SCLKME ;
} else if ( cpu_class_is_omap1 ( ) ) {
/*
* McBSP CLKR / FSR signal muxing functions are only available on
* OMAP2 or newer versions
*/
return - EINVAL ;
}
2009-11-09 19:02:15 +00:00
2008-04-25 13:55:19 +02:00
switch ( clk_id ) {
case OMAP_MCBSP_SYSCLK_CLK :
regs - > srgr2 | = CLKSM ;
break ;
case OMAP_MCBSP_SYSCLK_CLKS_FCLK :
2010-10-08 11:40:19 -06:00
if ( cpu_class_is_omap1 ( ) ) {
err = - EINVAL ;
break ;
}
2012-02-14 18:20:58 +02:00
err = omap2_mcbsp_set_clks_src ( mcbsp ,
2010-10-08 11:40:19 -06:00
MCBSP_CLKS_PRCM_SRC ) ;
break ;
2008-04-25 13:55:19 +02:00
case OMAP_MCBSP_SYSCLK_CLKS_EXT :
2010-10-08 11:40:19 -06:00
if ( cpu_class_is_omap1 ( ) ) {
err = 0 ;
break ;
}
2012-02-14 18:20:58 +02:00
err = omap2_mcbsp_set_clks_src ( mcbsp ,
2010-10-08 11:40:19 -06:00
MCBSP_CLKS_PAD_SRC ) ;
2008-04-25 13:55:19 +02:00
break ;
case OMAP_MCBSP_SYSCLK_CLKX_EXT :
regs - > srgr2 | = CLKSM ;
case OMAP_MCBSP_SYSCLK_CLKR_EXT :
regs - > pcr0 | = SCLKME ;
break ;
2009-08-28 15:35:35 +03:00
OMAP: McBSP: implement McBSP CLKR and FSR signal muxing via mach-omap2/mcbsp.c
The OMAP ASoC McBSP code implemented CLKR and FSR signal muxing via
direct System Control Module writes on OMAP2+. This required the
omap_ctrl_{read,write}l() functions to be exported, which is against
policy: the only code that should call those functions directly is
OMAP core code, not device drivers. omap_ctrl_{read,write}*() are no
longer exported, so the driver no longer builds as a module.
Fix the pinmuxing part of the problem by removing calls to
omap_ctrl_{read,write}l() from the OMAP ASoC McBSP code and
implementing signal muxing functions in arch/arm/mach-omap2/mcbsp.c.
Due to the unfortunate way that McBSP support is implemented in ASoC
and the OMAP tree, these symbols must be exported for use by
sound/soc/omap/omap-mcbsp.c.
Going forward, the McBSP device driver should be moved from
arch/arm/*omap* into drivers/ or sound/soc/*, and the CPU DAI driver
should be implemented as a platform_driver as many other ASoC CPU DAI
drivers are. These two steps should resolve many of the layering
problems, which will rapidly reappear during a McBSP hwmod/PM runtime
conversion.
Signed-off-by: Paul Walmsley <paul@pwsan.com>
Acked-by: Jarkko Nikula <jhnikula@gmail.com>
Acked-by: Peter Ujfalusi <peter.ujfalusi@nokia.com>
Acked-by: Liam Girdwood <lrg@slimlogic.co.uk>
Acked-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
2010-10-08 11:40:19 -06:00
2009-08-28 15:35:35 +03:00
case OMAP_MCBSP_CLKR_SRC_CLKR :
2012-03-08 11:01:37 +02:00
err = omap_mcbsp_6pin_src_mux ( mcbsp , CLKR_SRC_CLKR ) ;
OMAP: McBSP: implement McBSP CLKR and FSR signal muxing via mach-omap2/mcbsp.c
The OMAP ASoC McBSP code implemented CLKR and FSR signal muxing via
direct System Control Module writes on OMAP2+. This required the
omap_ctrl_{read,write}l() functions to be exported, which is against
policy: the only code that should call those functions directly is
OMAP core code, not device drivers. omap_ctrl_{read,write}*() are no
longer exported, so the driver no longer builds as a module.
Fix the pinmuxing part of the problem by removing calls to
omap_ctrl_{read,write}l() from the OMAP ASoC McBSP code and
implementing signal muxing functions in arch/arm/mach-omap2/mcbsp.c.
Due to the unfortunate way that McBSP support is implemented in ASoC
and the OMAP tree, these symbols must be exported for use by
sound/soc/omap/omap-mcbsp.c.
Going forward, the McBSP device driver should be moved from
arch/arm/*omap* into drivers/ or sound/soc/*, and the CPU DAI driver
should be implemented as a platform_driver as many other ASoC CPU DAI
drivers are. These two steps should resolve many of the layering
problems, which will rapidly reappear during a McBSP hwmod/PM runtime
conversion.
Signed-off-by: Paul Walmsley <paul@pwsan.com>
Acked-by: Jarkko Nikula <jhnikula@gmail.com>
Acked-by: Peter Ujfalusi <peter.ujfalusi@nokia.com>
Acked-by: Liam Girdwood <lrg@slimlogic.co.uk>
Acked-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
2010-10-08 11:40:19 -06:00
break ;
2009-08-28 15:35:35 +03:00
case OMAP_MCBSP_CLKR_SRC_CLKX :
2012-03-08 11:01:37 +02:00
err = omap_mcbsp_6pin_src_mux ( mcbsp , CLKR_SRC_CLKX ) ;
OMAP: McBSP: implement McBSP CLKR and FSR signal muxing via mach-omap2/mcbsp.c
The OMAP ASoC McBSP code implemented CLKR and FSR signal muxing via
direct System Control Module writes on OMAP2+. This required the
omap_ctrl_{read,write}l() functions to be exported, which is against
policy: the only code that should call those functions directly is
OMAP core code, not device drivers. omap_ctrl_{read,write}*() are no
longer exported, so the driver no longer builds as a module.
Fix the pinmuxing part of the problem by removing calls to
omap_ctrl_{read,write}l() from the OMAP ASoC McBSP code and
implementing signal muxing functions in arch/arm/mach-omap2/mcbsp.c.
Due to the unfortunate way that McBSP support is implemented in ASoC
and the OMAP tree, these symbols must be exported for use by
sound/soc/omap/omap-mcbsp.c.
Going forward, the McBSP device driver should be moved from
arch/arm/*omap* into drivers/ or sound/soc/*, and the CPU DAI driver
should be implemented as a platform_driver as many other ASoC CPU DAI
drivers are. These two steps should resolve many of the layering
problems, which will rapidly reappear during a McBSP hwmod/PM runtime
conversion.
Signed-off-by: Paul Walmsley <paul@pwsan.com>
Acked-by: Jarkko Nikula <jhnikula@gmail.com>
Acked-by: Peter Ujfalusi <peter.ujfalusi@nokia.com>
Acked-by: Liam Girdwood <lrg@slimlogic.co.uk>
Acked-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
2010-10-08 11:40:19 -06:00
break ;
2009-08-28 15:35:35 +03:00
case OMAP_MCBSP_FSR_SRC_FSR :
2012-03-08 11:01:37 +02:00
err = omap_mcbsp_6pin_src_mux ( mcbsp , FSR_SRC_FSR ) ;
OMAP: McBSP: implement McBSP CLKR and FSR signal muxing via mach-omap2/mcbsp.c
The OMAP ASoC McBSP code implemented CLKR and FSR signal muxing via
direct System Control Module writes on OMAP2+. This required the
omap_ctrl_{read,write}l() functions to be exported, which is against
policy: the only code that should call those functions directly is
OMAP core code, not device drivers. omap_ctrl_{read,write}*() are no
longer exported, so the driver no longer builds as a module.
Fix the pinmuxing part of the problem by removing calls to
omap_ctrl_{read,write}l() from the OMAP ASoC McBSP code and
implementing signal muxing functions in arch/arm/mach-omap2/mcbsp.c.
Due to the unfortunate way that McBSP support is implemented in ASoC
and the OMAP tree, these symbols must be exported for use by
sound/soc/omap/omap-mcbsp.c.
Going forward, the McBSP device driver should be moved from
arch/arm/*omap* into drivers/ or sound/soc/*, and the CPU DAI driver
should be implemented as a platform_driver as many other ASoC CPU DAI
drivers are. These two steps should resolve many of the layering
problems, which will rapidly reappear during a McBSP hwmod/PM runtime
conversion.
Signed-off-by: Paul Walmsley <paul@pwsan.com>
Acked-by: Jarkko Nikula <jhnikula@gmail.com>
Acked-by: Peter Ujfalusi <peter.ujfalusi@nokia.com>
Acked-by: Liam Girdwood <lrg@slimlogic.co.uk>
Acked-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
2010-10-08 11:40:19 -06:00
break ;
2009-08-28 15:35:35 +03:00
case OMAP_MCBSP_FSR_SRC_FSX :
2012-03-08 11:01:37 +02:00
err = omap_mcbsp_6pin_src_mux ( mcbsp , FSR_SRC_FSX ) ;
2009-08-28 15:35:35 +03:00
break ;
2008-04-25 13:55:19 +02:00
default :
err = - ENODEV ;
}
return err ;
}
2011-11-23 11:40:40 +01:00
static const struct snd_soc_dai_ops mcbsp_dai_ops = {
2009-03-03 09:41:00 +08:00
. startup = omap_mcbsp_dai_startup ,
. shutdown = omap_mcbsp_dai_shutdown ,
. trigger = omap_mcbsp_dai_trigger ,
2010-03-03 15:08:09 +02:00
. delay = omap_mcbsp_dai_delay ,
2009-03-03 09:41:00 +08:00
. hw_params = omap_mcbsp_dai_hw_params ,
. set_fmt = omap_mcbsp_dai_set_dai_fmt ,
. set_clkdiv = omap_mcbsp_dai_set_clkdiv ,
. set_sysclk = omap_mcbsp_dai_set_dai_sysclk ,
} ;
2012-02-14 14:52:42 +02:00
static int omap_mcbsp_probe ( struct snd_soc_dai * dai )
{
struct omap_mcbsp * mcbsp = snd_soc_dai_get_drvdata ( dai ) ;
pm_runtime_enable ( mcbsp - > dev ) ;
return 0 ;
}
static int omap_mcbsp_remove ( struct snd_soc_dai * dai )
{
struct omap_mcbsp * mcbsp = snd_soc_dai_get_drvdata ( dai ) ;
pm_runtime_disable ( mcbsp - > dev ) ;
return 0 ;
}
2011-10-10 07:07:08 +02:00
static struct snd_soc_dai_driver omap_mcbsp_dai = {
2012-02-14 14:52:42 +02:00
. probe = omap_mcbsp_probe ,
. remove = omap_mcbsp_remove ,
2010-03-17 20:15:21 +00:00
. playback = {
. channels_min = 1 ,
. channels_max = 16 ,
. rates = OMAP_MCBSP_RATES ,
. formats = SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S32_LE ,
} ,
. capture = {
. channels_min = 1 ,
. channels_max = 16 ,
. rates = OMAP_MCBSP_RATES ,
. formats = SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S32_LE ,
} ,
. ops = & mcbsp_dai_ops ,
2008-04-25 13:55:19 +02:00
} ;
2008-10-09 15:57:22 +03:00
2010-09-08 08:53:43 +05:30
static int omap_mcbsp_st_info_volsw ( struct snd_kcontrol * kcontrol ,
2010-02-22 12:21:12 +00:00
struct snd_ctl_elem_info * uinfo )
{
struct soc_mixer_control * mc =
( struct soc_mixer_control * ) kcontrol - > private_value ;
int max = mc - > max ;
int min = mc - > min ;
uinfo - > type = SNDRV_CTL_ELEM_TYPE_INTEGER ;
uinfo - > count = 1 ;
uinfo - > value . integer . min = min ;
uinfo - > value . integer . max = max ;
return 0 ;
}
2012-02-14 18:20:58 +02:00
# define OMAP_MCBSP_ST_SET_CHANNEL_VOLUME(channel) \
2010-02-22 12:21:12 +00:00
static int \
2012-02-14 18:20:58 +02:00
omap_mcbsp_set_st_ch # # channel # # _volume ( struct snd_kcontrol * kc , \
2010-02-22 12:21:12 +00:00
struct snd_ctl_elem_value * uc ) \
{ \
2012-02-14 18:20:58 +02:00
struct snd_soc_dai * cpu_dai = snd_kcontrol_chip ( kc ) ; \
struct omap_mcbsp * mcbsp = snd_soc_dai_get_drvdata ( cpu_dai ) ; \
2010-02-22 12:21:12 +00:00
struct soc_mixer_control * mc = \
( struct soc_mixer_control * ) kc - > private_value ; \
int max = mc - > max ; \
int min = mc - > min ; \
int val = uc - > value . integer . value [ 0 ] ; \
\
if ( val < min | | val > max ) \
return - EINVAL ; \
\
/* OMAP McBSP implementation uses index values 0..4 */ \
2012-02-14 18:20:58 +02:00
return omap_st_set_chgain ( mcbsp , channel , val ) ; \
2010-02-22 12:21:12 +00:00
}
2012-02-14 18:20:58 +02:00
# define OMAP_MCBSP_ST_GET_CHANNEL_VOLUME(channel) \
2010-02-22 12:21:12 +00:00
static int \
2012-02-14 18:20:58 +02:00
omap_mcbsp_get_st_ch # # channel # # _volume ( struct snd_kcontrol * kc , \
2010-02-22 12:21:12 +00:00
struct snd_ctl_elem_value * uc ) \
{ \
2012-02-14 18:20:58 +02:00
struct snd_soc_dai * cpu_dai = snd_kcontrol_chip ( kc ) ; \
struct omap_mcbsp * mcbsp = snd_soc_dai_get_drvdata ( cpu_dai ) ; \
2010-02-22 12:21:12 +00:00
s16 chgain ; \
\
2012-02-14 18:20:58 +02:00
if ( omap_st_get_chgain ( mcbsp , channel , & chgain ) ) \
2010-02-22 12:21:12 +00:00
return - EAGAIN ; \
\
uc - > value . integer . value [ 0 ] = chgain ; \
return 0 ; \
}
2012-02-14 18:20:58 +02:00
OMAP_MCBSP_ST_SET_CHANNEL_VOLUME ( 0 )
OMAP_MCBSP_ST_SET_CHANNEL_VOLUME ( 1 )
OMAP_MCBSP_ST_GET_CHANNEL_VOLUME ( 0 )
OMAP_MCBSP_ST_GET_CHANNEL_VOLUME ( 1 )
2010-02-22 12:21:12 +00:00
static int omap_mcbsp_st_put_mode ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
2012-02-14 18:20:58 +02:00
struct snd_soc_dai * cpu_dai = snd_kcontrol_chip ( kcontrol ) ;
struct omap_mcbsp * mcbsp = snd_soc_dai_get_drvdata ( cpu_dai ) ;
2010-02-22 12:21:12 +00:00
u8 value = ucontrol - > value . integer . value [ 0 ] ;
2012-02-14 18:20:58 +02:00
if ( value = = omap_st_is_enabled ( mcbsp ) )
2010-02-22 12:21:12 +00:00
return 0 ;
if ( value )
2012-02-14 18:20:58 +02:00
omap_st_enable ( mcbsp ) ;
2010-02-22 12:21:12 +00:00
else
2012-02-14 18:20:58 +02:00
omap_st_disable ( mcbsp ) ;
2010-02-22 12:21:12 +00:00
return 1 ;
}
static int omap_mcbsp_st_get_mode ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
2012-02-14 18:20:58 +02:00
struct snd_soc_dai * cpu_dai = snd_kcontrol_chip ( kcontrol ) ;
struct omap_mcbsp * mcbsp = snd_soc_dai_get_drvdata ( cpu_dai ) ;
2010-02-22 12:21:12 +00:00
2012-02-14 18:20:58 +02:00
ucontrol - > value . integer . value [ 0 ] = omap_st_is_enabled ( mcbsp ) ;
2010-02-22 12:21:12 +00:00
return 0 ;
}
static const struct snd_kcontrol_new omap_mcbsp2_st_controls [ ] = {
SOC_SINGLE_EXT ( " McBSP2 Sidetone Switch " , 1 , 0 , 1 , 0 ,
omap_mcbsp_st_get_mode , omap_mcbsp_st_put_mode ) ,
OMAP_MCBSP_SOC_SINGLE_S16_EXT ( " McBSP2 Sidetone Channel 0 Volume " ,
- 32768 , 32767 ,
2012-02-14 18:20:58 +02:00
omap_mcbsp_get_st_ch0_volume ,
omap_mcbsp_set_st_ch0_volume ) ,
2010-02-22 12:21:12 +00:00
OMAP_MCBSP_SOC_SINGLE_S16_EXT ( " McBSP2 Sidetone Channel 1 Volume " ,
- 32768 , 32767 ,
2012-02-14 18:20:58 +02:00
omap_mcbsp_get_st_ch1_volume ,
omap_mcbsp_set_st_ch1_volume ) ,
2010-02-22 12:21:12 +00:00
} ;
static const struct snd_kcontrol_new omap_mcbsp3_st_controls [ ] = {
SOC_SINGLE_EXT ( " McBSP3 Sidetone Switch " , 2 , 0 , 1 , 0 ,
omap_mcbsp_st_get_mode , omap_mcbsp_st_put_mode ) ,
OMAP_MCBSP_SOC_SINGLE_S16_EXT ( " McBSP3 Sidetone Channel 0 Volume " ,
- 32768 , 32767 ,
2012-02-14 18:20:58 +02:00
omap_mcbsp_get_st_ch0_volume ,
omap_mcbsp_set_st_ch0_volume ) ,
2010-02-22 12:21:12 +00:00
OMAP_MCBSP_SOC_SINGLE_S16_EXT ( " McBSP3 Sidetone Channel 1 Volume " ,
- 32768 , 32767 ,
2012-02-14 18:20:58 +02:00
omap_mcbsp_get_st_ch1_volume ,
omap_mcbsp_set_st_ch1_volume ) ,
2010-02-22 12:21:12 +00:00
} ;
2012-02-14 18:20:58 +02:00
int omap_mcbsp_st_add_controls ( struct snd_soc_pcm_runtime * rtd )
2010-02-22 12:21:12 +00:00
{
2012-02-14 18:20:58 +02:00
struct snd_soc_dai * cpu_dai = rtd - > cpu_dai ;
struct omap_mcbsp * mcbsp = snd_soc_dai_get_drvdata ( cpu_dai ) ;
if ( ! mcbsp - > st_data )
2010-02-22 12:21:12 +00:00
return - ENODEV ;
2012-02-14 18:20:58 +02:00
switch ( cpu_dai - > id ) {
case 2 : /* McBSP 2 */
return snd_soc_add_dai_controls ( cpu_dai ,
omap_mcbsp2_st_controls ,
2010-02-22 12:21:12 +00:00
ARRAY_SIZE ( omap_mcbsp2_st_controls ) ) ;
2012-02-14 18:20:58 +02:00
case 3 : /* McBSP 3 */
return snd_soc_add_dai_controls ( cpu_dai ,
omap_mcbsp3_st_controls ,
2010-02-22 12:21:12 +00:00
ARRAY_SIZE ( omap_mcbsp3_st_controls ) ) ;
default :
break ;
}
return - EINVAL ;
}
EXPORT_SYMBOL_GPL ( omap_mcbsp_st_add_controls ) ;
2010-03-17 20:15:21 +00:00
static __devinit int asoc_mcbsp_probe ( struct platform_device * pdev )
{
2012-02-14 14:52:42 +02:00
struct omap_mcbsp_platform_data * pdata = dev_get_platdata ( & pdev - > dev ) ;
struct omap_mcbsp * mcbsp ;
2012-02-14 18:20:58 +02:00
int ret ;
2012-02-14 14:52:42 +02:00
if ( ! pdata ) {
dev_err ( & pdev - > dev , " missing platform data. \n " ) ;
return - EINVAL ;
}
mcbsp = devm_kzalloc ( & pdev - > dev , sizeof ( struct omap_mcbsp ) , GFP_KERNEL ) ;
if ( ! mcbsp )
return - ENOMEM ;
mcbsp - > id = pdev - > id ;
mcbsp - > pdata = pdata ;
mcbsp - > dev = & pdev - > dev ;
platform_set_drvdata ( pdev , mcbsp ) ;
ret = omap_mcbsp_init ( pdev ) ;
2012-02-14 18:20:58 +02:00
if ( ! ret )
return snd_soc_register_dai ( & pdev - > dev , & omap_mcbsp_dai ) ;
return ret ;
2010-03-17 20:15:21 +00:00
}
static int __devexit asoc_mcbsp_remove ( struct platform_device * pdev )
{
2012-02-14 14:52:42 +02:00
struct omap_mcbsp * mcbsp = platform_get_drvdata ( pdev ) ;
2010-03-17 20:15:21 +00:00
snd_soc_unregister_dai ( & pdev - > dev ) ;
2012-02-14 14:52:42 +02:00
if ( mcbsp - > pdata - > ops & & mcbsp - > pdata - > ops - > free )
mcbsp - > pdata - > ops - > free ( mcbsp - > id ) ;
omap_mcbsp_sysfs_remove ( mcbsp ) ;
clk_put ( mcbsp - > fclk ) ;
platform_set_drvdata ( pdev , NULL ) ;
2010-03-17 20:15:21 +00:00
return 0 ;
}
static struct platform_driver asoc_mcbsp_driver = {
. driver = {
2012-02-14 18:20:58 +02:00
. name = " omap-mcbsp " ,
2010-03-17 20:15:21 +00:00
. owner = THIS_MODULE ,
} ,
. probe = asoc_mcbsp_probe ,
. remove = __devexit_p ( asoc_mcbsp_remove ) ,
} ;
2011-11-25 10:12:16 +08:00
module_platform_driver ( asoc_mcbsp_driver ) ;
2008-12-03 19:26:35 +00:00
2011-08-11 15:44:57 +03:00
MODULE_AUTHOR ( " Jarkko Nikula <jarkko.nikula@bitmer.com> " ) ;
2008-04-25 13:55:19 +02:00
MODULE_DESCRIPTION ( " OMAP I2S SoC Interface " ) ;
MODULE_LICENSE ( " GPL " ) ;
2012-07-12 15:08:16 +02:00
MODULE_ALIAS ( " platform:omap-mcbsp " ) ;