2008-07-23 14:03:07 +01:00
/*
* linux / sound / soc - dai . h - - ALSA SoC Layer
*
* Copyright : 2005 - 2008 Wolfson Microelectronics . PLC .
*
* 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 .
*
* Digital Audio Interface ( DAI ) API .
*/
# ifndef __LINUX_SND_SOC_DAI_H
# define __LINUX_SND_SOC_DAI_H
# include <linux/list.h>
2016-09-29 14:09:14 +08:00
# include <sound/asoc.h>
2008-07-23 14:03:07 +01:00
struct snd_pcm_substream ;
2012-02-16 19:37:51 -08:00
struct snd_soc_dapm_widget ;
2012-08-16 17:10:40 +05:30
struct snd_compr_stream ;
2008-07-23 14:03:07 +01:00
/*
* DAI hardware audio formats .
*
* Describes the physical PCM data formating and clocking . Add new formats
* to the end .
*/
2016-09-29 14:09:14 +08:00
# define SND_SOC_DAIFMT_I2S SND_SOC_DAI_FORMAT_I2S
# define SND_SOC_DAIFMT_RIGHT_J SND_SOC_DAI_FORMAT_RIGHT_J
# define SND_SOC_DAIFMT_LEFT_J SND_SOC_DAI_FORMAT_LEFT_J
# define SND_SOC_DAIFMT_DSP_A SND_SOC_DAI_FORMAT_DSP_A
# define SND_SOC_DAIFMT_DSP_B SND_SOC_DAI_FORMAT_DSP_B
# define SND_SOC_DAIFMT_AC97 SND_SOC_DAI_FORMAT_AC97
# define SND_SOC_DAIFMT_PDM SND_SOC_DAI_FORMAT_PDM
2008-07-23 14:03:07 +01:00
/* left and right justified also known as MSB and LSB respectively */
# define SND_SOC_DAIFMT_MSB SND_SOC_DAIFMT_LEFT_J
# define SND_SOC_DAIFMT_LSB SND_SOC_DAIFMT_RIGHT_J
/*
* DAI Clock gating .
*
2009-07-13 23:05:11 +01:00
* DAI bit clocks can be be gated ( disabled ) when the DAI is not
2008-07-23 14:03:07 +01:00
* sending or receiving PCM data in a frame . This can be used to save power .
*/
2011-09-27 16:41:01 +01:00
# define SND_SOC_DAIFMT_CONT (1 << 4) /* continuous clock */
2013-01-29 21:03:13 -08:00
# define SND_SOC_DAIFMT_GATED (0 << 4) /* clock is gated */
2008-07-23 14:03:07 +01:00
/*
2015-10-08 09:37:51 -07:00
* DAI hardware signal polarity .
2008-07-23 14:03:07 +01:00
*
* Specifies whether the DAI can also support inverted clocks for the specified
* format .
2015-10-08 09:37:51 -07:00
*
* BCLK :
* - " normal " polarity means signal is available at rising edge of BCLK
* - " inverted " polarity means signal is available at falling edge of BCLK
*
* FSYNC " normal " polarity depends on the frame format :
* - I2S : frame consists of left then right channel data . Left channel starts
* with falling FSYNC edge , right channel starts with rising FSYNC edge .
* - Left / Right Justified : frame consists of left then right channel data .
* Left channel starts with rising FSYNC edge , right channel starts with
* falling FSYNC edge .
* - DSP A / B : Frame starts with rising FSYNC edge .
* - AC97 : Frame starts with rising FSYNC edge .
*
* " Negative " FSYNC polarity is the one opposite of " normal " polarity .
2008-07-23 14:03:07 +01:00
*/
2013-01-15 20:18:23 -08:00
# define SND_SOC_DAIFMT_NB_NF (0 << 8) /* normal bit clock + frame */
2011-09-27 16:41:01 +01:00
# define SND_SOC_DAIFMT_NB_IF (2 << 8) /* normal BCLK + inv FRM */
# define SND_SOC_DAIFMT_IB_NF (3 << 8) /* invert BCLK + nor FRM */
# define SND_SOC_DAIFMT_IB_IF (4 << 8) /* invert BCLK + FRM */
2008-07-23 14:03:07 +01:00
/*
* DAI hardware clock masters .
*
* This is wrt the codec , the inverse is true for the interface
2009-07-13 23:05:11 +01:00
* i . e . if the codec is clk and FRM master then the interface is
2008-07-23 14:03:07 +01:00
* clk and frame slave .
*/
2011-09-27 16:41:01 +01:00
# define SND_SOC_DAIFMT_CBM_CFM (1 << 12) /* codec clk & FRM master */
# define SND_SOC_DAIFMT_CBS_CFM (2 << 12) /* codec clk slave & FRM master */
# define SND_SOC_DAIFMT_CBM_CFS (3 << 12) /* codec clk master & frame slave */
# define SND_SOC_DAIFMT_CBS_CFS (4 << 12) /* codec clk & FRM slave */
2008-07-23 14:03:07 +01:00
# define SND_SOC_DAIFMT_FORMAT_MASK 0x000f
# define SND_SOC_DAIFMT_CLOCK_MASK 0x00f0
# define SND_SOC_DAIFMT_INV_MASK 0x0f00
# define SND_SOC_DAIFMT_MASTER_MASK 0xf000
/*
* Master Clock Directions
*/
# define SND_SOC_CLOCK_IN 0
# define SND_SOC_CLOCK_OUT 1
2009-08-09 20:08:31 +01:00
# define SND_SOC_STD_AC97_FMTS (SNDRV_PCM_FMTBIT_S8 |\
SNDRV_PCM_FMTBIT_S16_LE | \
SNDRV_PCM_FMTBIT_S16_BE | \
SNDRV_PCM_FMTBIT_S20_3LE | \
SNDRV_PCM_FMTBIT_S20_3BE | \
SNDRV_PCM_FMTBIT_S24_3LE | \
SNDRV_PCM_FMTBIT_S24_3BE | \
2009-05-13 21:59:14 -04:00
SNDRV_PCM_FMTBIT_S32_LE | \
SNDRV_PCM_FMTBIT_S32_BE )
2009-05-02 12:24:55 +01:00
2010-03-17 20:15:21 +00:00
struct snd_soc_dai_driver ;
2008-07-23 14:03:07 +01:00
struct snd_soc_dai ;
struct snd_ac97_bus_ops ;
/* Digital Audio Interface clocking API.*/
int snd_soc_dai_set_sysclk ( struct snd_soc_dai * dai , int clk_id ,
unsigned int freq , int dir ) ;
int snd_soc_dai_set_clkdiv ( struct snd_soc_dai * dai ,
int div_id , int div ) ;
int snd_soc_dai_set_pll ( struct snd_soc_dai * dai ,
2009-09-05 18:52:16 +01:00
int pll_id , int source , unsigned int freq_in , unsigned int freq_out ) ;
2008-07-23 14:03:07 +01:00
2013-09-16 13:01:46 +01:00
int snd_soc_dai_set_bclk_ratio ( struct snd_soc_dai * dai , unsigned int ratio ) ;
2008-07-23 14:03:07 +01:00
/* Digital Audio interface formatting */
int snd_soc_dai_set_fmt ( struct snd_soc_dai * dai , unsigned int fmt ) ;
int snd_soc_dai_set_tdm_slot ( struct snd_soc_dai * dai ,
2009-06-15 21:44:31 -03:00
unsigned int tx_mask , unsigned int rx_mask , int slots , int slot_width ) ;
2008-07-23 14:03:07 +01:00
2009-09-12 01:16:29 +08:00
int snd_soc_dai_set_channel_map ( struct snd_soc_dai * dai ,
unsigned int tx_num , unsigned int * tx_slot ,
unsigned int rx_num , unsigned int * rx_slot ) ;
2008-07-23 14:03:07 +01:00
int snd_soc_dai_set_tristate ( struct snd_soc_dai * dai , int tristate ) ;
/* Digital Audio Interface mute */
2013-02-06 15:44:07 +00:00
int snd_soc_dai_digital_mute ( struct snd_soc_dai * dai , int mute ,
int direction ) ;
2008-07-23 14:03:07 +01:00
2014-01-08 10:40:18 +00:00
int snd_soc_dai_is_dummy ( struct snd_soc_dai * dai ) ;
2008-07-23 14:03:07 +01:00
struct snd_soc_dai_ops {
/*
* DAI clocking configuration , all optional .
* Called by soc_card drivers , normally in their hw_params .
*/
int ( * set_sysclk ) ( struct snd_soc_dai * dai ,
int clk_id , unsigned int freq , int dir ) ;
2009-09-05 18:52:16 +01:00
int ( * set_pll ) ( struct snd_soc_dai * dai , int pll_id , int source ,
unsigned int freq_in , unsigned int freq_out ) ;
2008-07-23 14:03:07 +01:00
int ( * set_clkdiv ) ( struct snd_soc_dai * dai , int div_id , int div ) ;
2013-09-16 13:01:46 +01:00
int ( * set_bclk_ratio ) ( struct snd_soc_dai * dai , unsigned int ratio ) ;
2008-07-23 14:03:07 +01:00
/*
* DAI format configuration
* Called by soc_card drivers , normally in their hw_params .
*/
int ( * set_fmt ) ( struct snd_soc_dai * dai , unsigned int fmt ) ;
2014-03-21 14:17:12 +08:00
int ( * xlate_tdm_slot_mask ) ( unsigned int slots ,
2014-02-14 09:34:35 +08:00
unsigned int * tx_mask , unsigned int * rx_mask ) ;
2008-07-23 14:03:07 +01:00
int ( * set_tdm_slot ) ( struct snd_soc_dai * dai ,
2009-06-15 21:44:31 -03:00
unsigned int tx_mask , unsigned int rx_mask ,
int slots , int slot_width ) ;
2009-09-12 01:16:29 +08:00
int ( * set_channel_map ) ( struct snd_soc_dai * dai ,
unsigned int tx_num , unsigned int * tx_slot ,
unsigned int rx_num , unsigned int * rx_slot ) ;
2008-07-23 14:03:07 +01:00
int ( * set_tristate ) ( struct snd_soc_dai * dai , int tristate ) ;
/*
* DAI digital mute - optional .
* Called by soc - core to minimise any pops .
*/
int ( * digital_mute ) ( struct snd_soc_dai * dai , int mute ) ;
2013-02-06 15:44:07 +00:00
int ( * mute_stream ) ( struct snd_soc_dai * dai , int mute , int stream ) ;
2008-11-18 22:11:38 +00:00
/*
* ALSA PCM audio operations - all optional .
* Called by soc - core during audio PCM operations .
*/
int ( * startup ) ( struct snd_pcm_substream * ,
struct snd_soc_dai * ) ;
void ( * shutdown ) ( struct snd_pcm_substream * ,
struct snd_soc_dai * ) ;
int ( * hw_params ) ( struct snd_pcm_substream * ,
struct snd_pcm_hw_params * , struct snd_soc_dai * ) ;
int ( * hw_free ) ( struct snd_pcm_substream * ,
struct snd_soc_dai * ) ;
int ( * prepare ) ( struct snd_pcm_substream * ,
struct snd_soc_dai * ) ;
2013-10-11 12:11:02 +02:00
/*
* NOTE : Commands passed to the trigger function are not necessarily
* compatible with the current state of the dai . For example this
* sequence of commands is possible : START STOP STOP .
* So do not unconditionally use refcounting functions in the trigger
* function , e . g . clk_enable / disable .
*/
2008-11-18 22:11:38 +00:00
int ( * trigger ) ( struct snd_pcm_substream * , int ,
struct snd_soc_dai * ) ;
2012-04-25 12:12:52 +01:00
int ( * bespoke_trigger ) ( struct snd_pcm_substream * , int ,
struct snd_soc_dai * ) ;
2010-03-03 15:08:07 +02:00
/*
* For hardware based FIFO caused delay reporting .
* Optional .
*/
snd_pcm_sframes_t ( * delay ) ( struct snd_pcm_substream * ,
struct snd_soc_dai * ) ;
2008-07-23 14:03:07 +01:00
} ;
2016-11-13 12:10:02 +05:30
struct snd_soc_cdai_ops {
/*
* for compress ops
*/
int ( * startup ) ( struct snd_compr_stream * ,
struct snd_soc_dai * ) ;
int ( * shutdown ) ( struct snd_compr_stream * ,
struct snd_soc_dai * ) ;
int ( * set_params ) ( struct snd_compr_stream * ,
struct snd_compr_params * , struct snd_soc_dai * ) ;
int ( * get_params ) ( struct snd_compr_stream * ,
struct snd_codec * , struct snd_soc_dai * ) ;
int ( * set_metadata ) ( struct snd_compr_stream * ,
struct snd_compr_metadata * , struct snd_soc_dai * ) ;
int ( * get_metadata ) ( struct snd_compr_stream * ,
struct snd_compr_metadata * , struct snd_soc_dai * ) ;
int ( * trigger ) ( struct snd_compr_stream * , int ,
struct snd_soc_dai * ) ;
int ( * pointer ) ( struct snd_compr_stream * ,
struct snd_compr_tstamp * , struct snd_soc_dai * ) ;
int ( * ack ) ( struct snd_compr_stream * , size_t ,
struct snd_soc_dai * ) ;
} ;
2008-07-23 14:03:07 +01:00
/*
2010-03-17 20:15:21 +00:00
* Digital Audio Interface Driver .
2008-07-23 14:03:07 +01:00
*
2010-03-17 20:15:21 +00:00
* Describes the Digital Audio Interface in terms of its ALSA , DAI and AC97
* operations and capabilities . Codec and platform drivers will register this
* structure for every DAI they have .
*
* This structure covers the clocking , formating and ALSA operations for each
* interface .
2008-07-23 14:03:07 +01:00
*/
2010-03-17 20:15:21 +00:00
struct snd_soc_dai_driver {
2008-07-23 14:03:07 +01:00
/* DAI description */
2010-03-17 20:15:21 +00:00
const char * name ;
2008-07-23 14:03:07 +01:00
unsigned int id ;
2012-05-01 20:03:32 +01:00
unsigned int base ;
2015-12-31 16:40:43 +08:00
struct snd_soc_dobj dobj ;
2008-07-23 14:03:07 +01:00
2010-03-17 20:15:21 +00:00
/* DAI driver callbacks */
int ( * probe ) ( struct snd_soc_dai * dai ) ;
int ( * remove ) ( struct snd_soc_dai * dai ) ;
2008-12-03 18:21:52 +00:00
int ( * suspend ) ( struct snd_soc_dai * dai ) ;
int ( * resume ) ( struct snd_soc_dai * dai ) ;
2012-08-16 17:10:40 +05:30
/* compress dai */
2015-10-13 23:41:00 +08:00
int ( * compress_new ) ( struct snd_soc_pcm_runtime * rtd , int num ) ;
2017-01-03 16:52:51 +01:00
/* Optional Callback used at pcm creation*/
int ( * pcm_new ) ( struct snd_soc_pcm_runtime * rtd ,
struct snd_soc_dai * dai ) ;
2014-11-10 22:41:52 +01:00
/* DAI is also used for the control bus */
bool bus_control ;
2008-07-23 14:03:07 +01:00
/* ops */
2010-12-02 16:10:09 +00:00
const struct snd_soc_dai_ops * ops ;
2016-11-13 12:10:02 +05:30
const struct snd_soc_cdai_ops * cops ;
2008-07-23 14:03:07 +01:00
/* DAI capabilities */
struct snd_soc_pcm_stream capture ;
struct snd_soc_pcm_stream playback ;
2009-04-07 18:10:13 +01:00
unsigned int symmetric_rates : 1 ;
2013-11-13 18:56:24 +08:00
unsigned int symmetric_channels : 1 ;
unsigned int symmetric_samplebits : 1 ;
2011-06-07 16:08:05 +01:00
/* probe ordering - for components with runtime dependencies */
int probe_order ;
int remove_order ;
2010-03-17 20:15:21 +00:00
} ;
/*
* Digital Audio Interface runtime data .
*
* Holds runtime data for a DAI .
*/
struct snd_soc_dai {
const char * name ;
int id ;
struct device * dev ;
/* driver ops */
struct snd_soc_dai_driver * driver ;
2008-07-23 14:03:07 +01:00
/* DAI runtime info */
2010-03-17 20:15:21 +00:00
unsigned int capture_active : 1 ; /* stream is in use */
unsigned int playback_active : 1 ; /* stream is in use */
unsigned int symmetric_rates : 1 ;
2013-11-13 18:56:24 +08:00
unsigned int symmetric_channels : 1 ;
unsigned int symmetric_samplebits : 1 ;
2016-10-24 05:25:19 +00:00
unsigned int probed : 1 ;
2008-07-23 14:03:07 +01:00
unsigned int active ;
2012-02-16 19:37:51 -08:00
struct snd_soc_dapm_widget * playback_widget ;
struct snd_soc_dapm_widget * capture_widget ;
2010-03-17 20:15:21 +00:00
/* DAI DMA data */
void * playback_dma_data ;
void * capture_dma_data ;
2008-07-23 14:03:07 +01:00
2011-08-29 17:15:14 +08:00
/* Symmetry data - only valid if symmetry is being enforced */
unsigned int rate ;
2013-11-13 18:56:24 +08:00
unsigned int channels ;
unsigned int sample_bits ;
2011-08-29 17:15:14 +08:00
2010-03-17 20:15:21 +00:00
/* parent platform/codec */
2012-03-02 16:18:30 +00:00
struct snd_soc_codec * codec ;
2014-03-05 13:17:46 +01:00
struct snd_soc_component * component ;
2012-03-02 16:18:30 +00:00
2014-07-08 23:19:34 +02:00
/* CODEC TDM slot masks and params (for fixup) */
unsigned int tx_mask ;
unsigned int rx_mask ;
2008-07-23 14:03:07 +01:00
struct list_head list ;
} ;
2010-03-19 14:52:55 +00:00
static inline void * snd_soc_dai_get_dma_data ( const struct snd_soc_dai * dai ,
const struct snd_pcm_substream * ss )
{
return ( ss - > stream = = SNDRV_PCM_STREAM_PLAYBACK ) ?
2010-03-17 20:15:21 +00:00
dai - > playback_dma_data : dai - > capture_dma_data ;
2010-03-19 14:52:55 +00:00
}
static inline void snd_soc_dai_set_dma_data ( struct snd_soc_dai * dai ,
const struct snd_pcm_substream * ss ,
void * data )
{
if ( ss - > stream = = SNDRV_PCM_STREAM_PLAYBACK )
2010-03-17 20:15:21 +00:00
dai - > playback_dma_data = data ;
2010-03-19 14:52:55 +00:00
else
2010-03-17 20:15:21 +00:00
dai - > capture_dma_data = data ;
}
2013-10-17 21:13:19 +01:00
static inline void snd_soc_dai_init_dma_data ( struct snd_soc_dai * dai ,
void * playback , void * capture )
{
dai - > playback_dma_data = playback ;
dai - > capture_dma_data = capture ;
}
2010-03-17 20:15:21 +00:00
static inline void snd_soc_dai_set_drvdata ( struct snd_soc_dai * dai ,
void * data )
{
dev_set_drvdata ( dai - > dev , data ) ;
}
static inline void * snd_soc_dai_get_drvdata ( struct snd_soc_dai * dai )
{
return dev_get_drvdata ( dai - > dev ) ;
2010-03-19 14:52:55 +00:00
}
2008-07-23 14:03:07 +01:00
# endif