2008-04-25 13:55:19 +02:00
/*
* omap - mcbsp . c - - OMAP ALSA SoC DAI driver using McBSP port
*
* Copyright ( C ) 2008 Nokia Corporation
*
* Contact : Jarkko Nikula < jarkko . nikula @ nokia . com >
*
* 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>
# include <sound/core.h>
# include <sound/pcm.h>
# include <sound/pcm_params.h>
# include <sound/initval.h>
# include <sound/soc.h>
2008-08-05 16:14:15 +01:00
# include <mach/control.h>
# include <mach/dma.h>
# include <mach/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
struct omap_mcbsp_data {
unsigned int bus_id ;
struct omap_mcbsp_reg_cfg regs ;
2008-10-20 15:29:59 +03:00
unsigned int fmt ;
2008-04-25 13:55:19 +02:00
/*
* Flags indicating is the bus already activated and configured by
* another substream
*/
int active ;
int configured ;
} ;
# define to_mcbsp(priv) container_of((priv), struct omap_mcbsp_data, bus_id)
static struct omap_mcbsp_data mcbsp_data [ NUM_LINKS ] ;
/*
* Stream DMA parameters . DMA request line and port address are set runtime
* since they are different between OMAP1 and later OMAPs
*/
2008-10-09 15:57:21 +03:00
static struct omap_pcm_dma_data omap_mcbsp_dai_dma_params [ NUM_LINKS ] [ 2 ] ;
2008-04-25 13:55:19 +02:00
# if defined(CONFIG_ARCH_OMAP15XX) || defined(CONFIG_ARCH_OMAP16XX)
static const int omap1_dma_reqs [ ] [ 2 ] = {
{ OMAP_DMA_MCBSP1_TX , OMAP_DMA_MCBSP1_RX } ,
{ OMAP_DMA_MCBSP2_TX , OMAP_DMA_MCBSP2_RX } ,
{ OMAP_DMA_MCBSP3_TX , OMAP_DMA_MCBSP3_RX } ,
} ;
static const unsigned long omap1_mcbsp_port [ ] [ 2 ] = {
{ OMAP1510_MCBSP1_BASE + OMAP_MCBSP_REG_DXR1 ,
OMAP1510_MCBSP1_BASE + OMAP_MCBSP_REG_DRR1 } ,
{ OMAP1510_MCBSP2_BASE + OMAP_MCBSP_REG_DXR1 ,
OMAP1510_MCBSP2_BASE + OMAP_MCBSP_REG_DRR1 } ,
{ OMAP1510_MCBSP3_BASE + OMAP_MCBSP_REG_DXR1 ,
OMAP1510_MCBSP3_BASE + OMAP_MCBSP_REG_DRR1 } ,
} ;
# else
static const int omap1_dma_reqs [ ] [ 2 ] = { } ;
static const unsigned long omap1_mcbsp_port [ ] [ 2 ] = { } ;
# endif
2008-10-09 15:57:20 +03:00
# if defined(CONFIG_ARCH_OMAP24XX) || defined(CONFIG_ARCH_OMAP34XX)
static const int omap24xx_dma_reqs [ ] [ 2 ] = {
2008-04-25 13:55:19 +02:00
{ OMAP24XX_DMA_MCBSP1_TX , OMAP24XX_DMA_MCBSP1_RX } ,
{ OMAP24XX_DMA_MCBSP2_TX , OMAP24XX_DMA_MCBSP2_RX } ,
2008-10-09 15:57:20 +03:00
# if defined(CONFIG_ARCH_OMAP2430) || defined(CONFIG_ARCH_OMAP34XX)
{ OMAP24XX_DMA_MCBSP3_TX , OMAP24XX_DMA_MCBSP3_RX } ,
{ OMAP24XX_DMA_MCBSP4_TX , OMAP24XX_DMA_MCBSP4_RX } ,
{ OMAP24XX_DMA_MCBSP5_TX , OMAP24XX_DMA_MCBSP5_RX } ,
# endif
2008-04-25 13:55:19 +02:00
} ;
2008-10-09 15:57:20 +03:00
# else
static const int omap24xx_dma_reqs [ ] [ 2 ] = { } ;
# endif
# if defined(CONFIG_ARCH_OMAP2420)
2008-04-25 13:55:19 +02:00
static const unsigned long omap2420_mcbsp_port [ ] [ 2 ] = {
{ OMAP24XX_MCBSP1_BASE + OMAP_MCBSP_REG_DXR1 ,
OMAP24XX_MCBSP1_BASE + OMAP_MCBSP_REG_DRR1 } ,
{ OMAP24XX_MCBSP2_BASE + OMAP_MCBSP_REG_DXR1 ,
OMAP24XX_MCBSP2_BASE + OMAP_MCBSP_REG_DRR1 } ,
} ;
# else
static const unsigned long omap2420_mcbsp_port [ ] [ 2 ] = { } ;
# endif
2008-10-09 15:57:20 +03:00
# if defined(CONFIG_ARCH_OMAP2430)
static const unsigned long omap2430_mcbsp_port [ ] [ 2 ] = {
{ OMAP24XX_MCBSP1_BASE + OMAP_MCBSP_REG_DXR ,
OMAP24XX_MCBSP1_BASE + OMAP_MCBSP_REG_DRR } ,
{ OMAP24XX_MCBSP2_BASE + OMAP_MCBSP_REG_DXR ,
OMAP24XX_MCBSP2_BASE + OMAP_MCBSP_REG_DRR } ,
{ OMAP2430_MCBSP3_BASE + OMAP_MCBSP_REG_DXR ,
OMAP2430_MCBSP3_BASE + OMAP_MCBSP_REG_DRR } ,
{ OMAP2430_MCBSP4_BASE + OMAP_MCBSP_REG_DXR ,
OMAP2430_MCBSP4_BASE + OMAP_MCBSP_REG_DRR } ,
{ OMAP2430_MCBSP5_BASE + OMAP_MCBSP_REG_DXR ,
OMAP2430_MCBSP5_BASE + OMAP_MCBSP_REG_DRR } ,
} ;
# else
static const unsigned long omap2430_mcbsp_port [ ] [ 2 ] = { } ;
# endif
# if defined(CONFIG_ARCH_OMAP34XX)
static const unsigned long omap34xx_mcbsp_port [ ] [ 2 ] = {
{ OMAP34XX_MCBSP1_BASE + OMAP_MCBSP_REG_DXR ,
OMAP34XX_MCBSP1_BASE + OMAP_MCBSP_REG_DRR } ,
{ OMAP34XX_MCBSP2_BASE + OMAP_MCBSP_REG_DXR ,
OMAP34XX_MCBSP2_BASE + OMAP_MCBSP_REG_DRR } ,
{ OMAP34XX_MCBSP3_BASE + OMAP_MCBSP_REG_DXR ,
OMAP34XX_MCBSP3_BASE + OMAP_MCBSP_REG_DRR } ,
{ OMAP34XX_MCBSP4_BASE + OMAP_MCBSP_REG_DXR ,
OMAP34XX_MCBSP4_BASE + OMAP_MCBSP_REG_DRR } ,
{ OMAP34XX_MCBSP5_BASE + OMAP_MCBSP_REG_DXR ,
OMAP34XX_MCBSP5_BASE + OMAP_MCBSP_REG_DRR } ,
} ;
# else
static const unsigned long omap34xx_mcbsp_port [ ] [ 2 ] = { } ;
# endif
2008-11-18 22:11:38 +00:00
static int omap_mcbsp_dai_startup ( struct snd_pcm_substream * substream ,
struct snd_soc_dai * dai )
2008-04-25 13:55:19 +02:00
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
2008-07-07 16:08:07 +01:00
struct snd_soc_dai * cpu_dai = rtd - > dai - > cpu_dai ;
2008-04-25 13:55:19 +02:00
struct omap_mcbsp_data * mcbsp_data = to_mcbsp ( cpu_dai - > private_data ) ;
int err = 0 ;
2009-03-27 15:32:01 +02:00
if ( cpu_is_omap343x ( ) & & mcbsp_data - > bus_id = = 1 ) {
/*
* McBSP2 in OMAP3 has 1024 * 32 - bit internal audio buffer .
* Set constraint for minimum buffer size to the same than FIFO
* size in order to avoid underruns in playback startup because
* HW is keeping the DMA request active until FIFO is filled .
*/
snd_pcm_hw_constraint_minmax ( substream - > runtime ,
SNDRV_PCM_HW_PARAM_BUFFER_BYTES , 4096 , UINT_MAX ) ;
}
2008-04-25 13:55:19 +02:00
if ( ! cpu_dai - > active )
err = omap_mcbsp_request ( mcbsp_data - > bus_id ) ;
return err ;
}
2008-11-18 22:11:38 +00:00
static void omap_mcbsp_dai_shutdown ( struct snd_pcm_substream * substream ,
struct snd_soc_dai * dai )
2008-04-25 13:55:19 +02:00
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
2008-07-07 16:08:07 +01:00
struct snd_soc_dai * cpu_dai = rtd - > dai - > cpu_dai ;
2008-04-25 13:55:19 +02:00
struct omap_mcbsp_data * mcbsp_data = to_mcbsp ( cpu_dai - > private_data ) ;
if ( ! cpu_dai - > active ) {
omap_mcbsp_free ( mcbsp_data - > bus_id ) ;
mcbsp_data - > configured = 0 ;
}
}
2008-11-18 22:11:38 +00:00
static int omap_mcbsp_dai_trigger ( struct snd_pcm_substream * substream , int cmd ,
struct snd_soc_dai * dai )
2008-04-25 13:55:19 +02:00
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
2008-07-07 16:08:07 +01:00
struct snd_soc_dai * cpu_dai = rtd - > dai - > cpu_dai ;
2008-04-25 13:55:19 +02:00
struct omap_mcbsp_data * mcbsp_data = to_mcbsp ( cpu_dai - > private_data ) ;
int err = 0 ;
switch ( cmd ) {
case SNDRV_PCM_TRIGGER_START :
case SNDRV_PCM_TRIGGER_RESUME :
case SNDRV_PCM_TRIGGER_PAUSE_RELEASE :
if ( ! mcbsp_data - > active + + )
omap_mcbsp_start ( mcbsp_data - > bus_id ) ;
break ;
case SNDRV_PCM_TRIGGER_STOP :
case SNDRV_PCM_TRIGGER_SUSPEND :
case SNDRV_PCM_TRIGGER_PAUSE_PUSH :
if ( ! - - mcbsp_data - > active )
omap_mcbsp_stop ( mcbsp_data - > bus_id ) ;
break ;
default :
err = - EINVAL ;
}
return err ;
}
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 ,
struct snd_soc_dai * dai )
2008-04-25 13:55:19 +02:00
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
2008-07-07 16:08:07 +01:00
struct snd_soc_dai * cpu_dai = rtd - > dai - > cpu_dai ;
2008-04-25 13:55:19 +02:00
struct omap_mcbsp_data * mcbsp_data = to_mcbsp ( cpu_dai - > private_data ) ;
struct omap_mcbsp_reg_cfg * regs = & mcbsp_data - > regs ;
int dma , bus_id = mcbsp_data - > bus_id , id = cpu_dai - > id ;
2009-04-15 15:38:55 +03:00
int wlen , channels , wpf ;
2008-04-25 13:55:19 +02:00
unsigned long port ;
2009-04-15 15:38:55 +03:00
unsigned int format ;
2008-04-25 13:55:19 +02:00
if ( cpu_class_is_omap1 ( ) ) {
dma = omap1_dma_reqs [ bus_id ] [ substream - > stream ] ;
port = omap1_mcbsp_port [ bus_id ] [ substream - > stream ] ;
} else if ( cpu_is_omap2420 ( ) ) {
2008-10-09 15:57:20 +03:00
dma = omap24xx_dma_reqs [ bus_id ] [ substream - > stream ] ;
2008-04-25 13:55:19 +02:00
port = omap2420_mcbsp_port [ bus_id ] [ substream - > stream ] ;
2008-10-09 15:57:20 +03:00
} else if ( cpu_is_omap2430 ( ) ) {
dma = omap24xx_dma_reqs [ bus_id ] [ substream - > stream ] ;
port = omap2430_mcbsp_port [ bus_id ] [ substream - > stream ] ;
} else if ( cpu_is_omap343x ( ) ) {
dma = omap24xx_dma_reqs [ bus_id ] [ substream - > stream ] ;
port = omap34xx_mcbsp_port [ bus_id ] [ substream - > stream ] ;
2008-04-25 13:55:19 +02:00
} else {
return - ENODEV ;
}
2008-10-09 15:57:21 +03:00
omap_mcbsp_dai_dma_params [ id ] [ substream - > stream ] . name =
substream - > stream ? " Audio Capture " : " Audio Playback " ;
2008-04-25 13:55:19 +02:00
omap_mcbsp_dai_dma_params [ id ] [ substream - > stream ] . dma_req = dma ;
omap_mcbsp_dai_dma_params [ id ] [ substream - > stream ] . port_addr = port ;
cpu_dai - > dma_data = & omap_mcbsp_dai_dma_params [ id ] [ substream - > stream ] ;
if ( mcbsp_data - > configured ) {
/* McBSP already configured by another stream */
return 0 ;
}
2009-04-15 15:38:55 +03:00
format = mcbsp_data - > fmt & SND_SOC_DAIFMT_FORMAT_MASK ;
wpf = channels = params_channels ( params ) ;
2008-11-25 12:45:09 +02:00
switch ( channels ) {
2008-04-25 13:55:19 +02:00
case 2 :
2009-04-15 15:38:55 +03:00
if ( format = = SND_SOC_DAIFMT_I2S ) {
/* 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-11-25 12:45:09 +02:00
case 1 :
2009-04-15 15:38:55 +03:00
/* Set word per (McBSP) frame for phase1 */
regs - > rcr1 | = RFRLEN1 ( wpf - 1 ) ;
regs - > xcr1 | = XFRLEN1 ( wpf - 1 ) ;
2008-04-25 13:55:19 +02:00
break ;
default :
/* Unsupported number of channels */
return - EINVAL ;
}
switch ( params_format ( params ) ) {
case SNDRV_PCM_FORMAT_S16_LE :
/* Set word lengths */
2008-10-20 15:29:59 +03:00
wlen = 16 ;
2008-04-25 13:55:19 +02:00
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 ;
default :
/* Unsupported PCM format */
return - EINVAL ;
}
2008-10-20 15:29:59 +03:00
/* Set FS period and length in terms of bit clock periods */
2009-04-15 15:38:55 +03:00
switch ( format ) {
2008-10-20 15:29:59 +03:00
case SND_SOC_DAIFMT_I2S :
2009-04-15 15:38:55 +03:00
regs - > srgr2 | = FPER ( wlen * channels - 1 ) ;
2008-10-20 15:29:59 +03:00
regs - > srgr1 | = FWID ( wlen - 1 ) ;
break ;
2008-12-22 10:21:36 +02:00
case SND_SOC_DAIFMT_DSP_B :
2008-11-25 12:45:09 +02:00
regs - > srgr2 | = FPER ( wlen * channels - 1 ) ;
regs - > srgr1 | = FWID ( wlen * channels - 2 ) ;
2008-10-20 15:29:59 +03:00
break ;
}
2008-04-25 13:55:19 +02:00
omap_mcbsp_config ( bus_id , & mcbsp_data - > regs ) ;
mcbsp_data - > configured = 1 ;
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 )
{
struct omap_mcbsp_data * mcbsp_data = to_mcbsp ( cpu_dai - > private_data ) ;
struct omap_mcbsp_reg_cfg * regs = & mcbsp_data - > regs ;
if ( mcbsp_data - > configured )
return 0 ;
2008-10-20 15:29:59 +03:00
mcbsp_data - > 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 ) ;
regs - > rcr2 | = RFIG ;
regs - > xcr2 | = XFIG ;
2009-01-29 13:29:46 +02:00
if ( cpu_is_omap2430 ( ) | | cpu_is_omap34xx ( ) ) {
regs - > xccr = DXENDLY ( 1 ) | XDMAEN ;
regs - > rccr = RFULL_CYCLE | RDMAEN ;
}
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 ;
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 ) ;
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 */
2008-10-22 15:00:29 +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 ;
}
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 )
{
struct omap_mcbsp_data * mcbsp_data = to_mcbsp ( cpu_dai - > private_data ) ;
struct omap_mcbsp_reg_cfg * regs = & mcbsp_data - > regs ;
if ( div_id ! = OMAP_MCBSP_CLKGDV )
return - ENODEV ;
regs - > srgr1 | = CLKGDV ( div - 1 ) ;
return 0 ;
}
static int omap_mcbsp_dai_set_clks_src ( struct omap_mcbsp_data * mcbsp_data ,
int clk_id )
{
int sel_bit ;
2008-10-09 15:57:20 +03:00
u16 reg , reg_devconf1 = OMAP243X_CONTROL_DEVCONF1 ;
2008-04-25 13:55:19 +02:00
if ( cpu_class_is_omap1 ( ) ) {
/* OMAP1's can use only external source clock */
if ( unlikely ( clk_id = = OMAP_MCBSP_SYSCLK_CLKS_FCLK ) )
return - EINVAL ;
else
return 0 ;
}
2008-10-09 15:57:20 +03:00
if ( cpu_is_omap2420 ( ) & & mcbsp_data - > bus_id > 1 )
return - EINVAL ;
if ( cpu_is_omap343x ( ) )
reg_devconf1 = OMAP343X_CONTROL_DEVCONF1 ;
2008-04-25 13:55:19 +02:00
switch ( mcbsp_data - > bus_id ) {
case 0 :
reg = OMAP2_CONTROL_DEVCONF0 ;
sel_bit = 2 ;
break ;
case 1 :
reg = OMAP2_CONTROL_DEVCONF0 ;
sel_bit = 6 ;
break ;
2008-10-09 15:57:20 +03:00
case 2 :
reg = reg_devconf1 ;
sel_bit = 0 ;
break ;
case 3 :
reg = reg_devconf1 ;
sel_bit = 2 ;
break ;
case 4 :
reg = reg_devconf1 ;
sel_bit = 4 ;
break ;
2008-04-25 13:55:19 +02:00
default :
return - EINVAL ;
}
2008-10-09 15:57:20 +03:00
if ( clk_id = = OMAP_MCBSP_SYSCLK_CLKS_FCLK )
omap_ctrl_writel ( omap_ctrl_readl ( reg ) & ~ ( 1 < < sel_bit ) , reg ) ;
else
omap_ctrl_writel ( omap_ctrl_readl ( reg ) | ( 1 < < sel_bit ) , reg ) ;
2008-04-25 13:55:19 +02:00
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 )
{
struct omap_mcbsp_data * mcbsp_data = to_mcbsp ( cpu_dai - > private_data ) ;
struct omap_mcbsp_reg_cfg * regs = & mcbsp_data - > regs ;
int err = 0 ;
switch ( clk_id ) {
case OMAP_MCBSP_SYSCLK_CLK :
regs - > srgr2 | = CLKSM ;
break ;
case OMAP_MCBSP_SYSCLK_CLKS_FCLK :
case OMAP_MCBSP_SYSCLK_CLKS_EXT :
err = omap_mcbsp_dai_set_clks_src ( mcbsp_data , clk_id ) ;
break ;
case OMAP_MCBSP_SYSCLK_CLKX_EXT :
regs - > srgr2 | = CLKSM ;
case OMAP_MCBSP_SYSCLK_CLKR_EXT :
regs - > pcr0 | = SCLKME ;
break ;
default :
err = - ENODEV ;
}
return err ;
}
2009-03-03 09:41:00 +08:00
static struct snd_soc_dai_ops omap_mcbsp_dai_ops = {
. startup = omap_mcbsp_dai_startup ,
. shutdown = omap_mcbsp_dai_shutdown ,
. trigger = omap_mcbsp_dai_trigger ,
. 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 ,
} ;
2008-10-09 15:57:22 +03:00
# define OMAP_MCBSP_DAI_BUILDER(link_id) \
{ \
2008-11-21 14:31:33 +02:00
. name = " omap-mcbsp-dai- " # link_id , \
2008-10-09 15:57:22 +03:00
. id = ( link_id ) , \
. playback = { \
2008-11-25 12:45:09 +02:00
. channels_min = 1 , \
2008-10-09 15:57:22 +03:00
. channels_max = 2 , \
. rates = OMAP_MCBSP_RATES , \
. formats = SNDRV_PCM_FMTBIT_S16_LE , \
} , \
. capture = { \
2008-11-25 12:45:09 +02:00
. channels_min = 1 , \
2008-10-09 15:57:22 +03:00
. channels_max = 2 , \
. rates = OMAP_MCBSP_RATES , \
. formats = SNDRV_PCM_FMTBIT_S16_LE , \
} , \
2009-03-03 09:41:00 +08:00
. ops = & omap_mcbsp_dai_ops , \
2008-10-09 15:57:22 +03:00
. private_data = & mcbsp_data [ ( link_id ) ] . bus_id , \
}
struct snd_soc_dai omap_mcbsp_dai [ ] = {
OMAP_MCBSP_DAI_BUILDER ( 0 ) ,
OMAP_MCBSP_DAI_BUILDER ( 1 ) ,
# if NUM_LINKS >= 3
OMAP_MCBSP_DAI_BUILDER ( 2 ) ,
# endif
# if NUM_LINKS == 5
OMAP_MCBSP_DAI_BUILDER ( 3 ) ,
OMAP_MCBSP_DAI_BUILDER ( 4 ) ,
# endif
2008-04-25 13:55:19 +02:00
} ;
2008-10-09 15:57:22 +03:00
2008-04-25 13:55:19 +02:00
EXPORT_SYMBOL_GPL ( omap_mcbsp_dai ) ;
2008-12-10 07:59:33 +01:00
static int __init snd_omap_mcbsp_init ( void )
2008-12-03 19:26:35 +00:00
{
return snd_soc_register_dais ( omap_mcbsp_dai ,
ARRAY_SIZE ( omap_mcbsp_dai ) ) ;
}
2008-12-10 07:59:33 +01:00
module_init ( snd_omap_mcbsp_init ) ;
2008-12-03 19:26:35 +00:00
2008-12-10 07:59:33 +01:00
static void __exit snd_omap_mcbsp_exit ( void )
2008-12-03 19:26:35 +00:00
{
snd_soc_unregister_dais ( omap_mcbsp_dai , ARRAY_SIZE ( omap_mcbsp_dai ) ) ;
}
2008-12-10 07:59:33 +01:00
module_exit ( snd_omap_mcbsp_exit ) ;
2008-12-03 19:26:35 +00:00
2008-04-25 13:55:19 +02:00
MODULE_AUTHOR ( " Jarkko Nikula <jarkko.nikula@nokia.com> " ) ;
MODULE_DESCRIPTION ( " OMAP I2S SoC Interface " ) ;
MODULE_LICENSE ( " GPL " ) ;