2014-05-27 12:53:17 +04:00
/*
* Common code for ADAU1X61 and ADAU1X81 codecs
*
* Copyright 2011 - 2014 Analog Devices Inc .
* Author : Lars - Peter Clausen < lars @ metafoo . de >
*
* Licensed under the GPL - 2 or later .
*/
# include <linux/module.h>
# include <linux/init.h>
# include <linux/delay.h>
# include <linux/slab.h>
# include <sound/core.h>
# include <sound/pcm.h>
# include <sound/pcm_params.h>
# include <sound/soc.h>
# include <sound/tlv.h>
# include <linux/gcd.h>
# include <linux/i2c.h>
# include <linux/spi/spi.h>
# include <linux/regmap.h>
# include "sigmadsp.h"
# include "adau17x1.h"
static const char * const adau17x1_capture_mixer_boost_text [ ] = {
" Normal operation " , " Boost Level 1 " , " Boost Level 2 " , " Boost Level 3 " ,
} ;
static SOC_ENUM_SINGLE_DECL ( adau17x1_capture_boost_enum ,
ADAU17X1_REC_POWER_MGMT , 5 , adau17x1_capture_mixer_boost_text ) ;
static const char * const adau17x1_mic_bias_mode_text [ ] = {
" Normal operation " , " High performance " ,
} ;
static SOC_ENUM_SINGLE_DECL ( adau17x1_mic_bias_mode_enum ,
ADAU17X1_MICBIAS , 3 , adau17x1_mic_bias_mode_text ) ;
static const DECLARE_TLV_DB_MINMAX ( adau17x1_digital_tlv , - 9563 , 0 ) ;
static const struct snd_kcontrol_new adau17x1_controls [ ] = {
SOC_DOUBLE_R_TLV ( " Digital Capture Volume " ,
ADAU17X1_LEFT_INPUT_DIGITAL_VOL ,
ADAU17X1_RIGHT_INPUT_DIGITAL_VOL ,
0 , 0xff , 1 , adau17x1_digital_tlv ) ,
SOC_DOUBLE_R_TLV ( " Digital Playback Volume " , ADAU17X1_DAC_CONTROL1 ,
ADAU17X1_DAC_CONTROL2 , 0 , 0xff , 1 , adau17x1_digital_tlv ) ,
SOC_SINGLE ( " ADC High Pass Filter Switch " , ADAU17X1_ADC_CONTROL ,
5 , 1 , 0 ) ,
SOC_SINGLE ( " Playback De-emphasis Switch " , ADAU17X1_DAC_CONTROL0 ,
2 , 1 , 0 ) ,
SOC_ENUM ( " Capture Boost " , adau17x1_capture_boost_enum ) ,
SOC_ENUM ( " Mic Bias Mode " , adau17x1_mic_bias_mode_enum ) ,
} ;
static int adau17x1_pll_event ( struct snd_soc_dapm_widget * w ,
struct snd_kcontrol * kcontrol , int event )
{
2014-11-17 11:37:36 +03:00
struct snd_soc_codec * codec = snd_soc_dapm_to_codec ( w - > dapm ) ;
struct adau * adau = snd_soc_codec_get_drvdata ( codec ) ;
2014-05-27 12:53:17 +04:00
int ret ;
if ( SND_SOC_DAPM_EVENT_ON ( event ) ) {
adau - > pll_regs [ 5 ] = 1 ;
} else {
adau - > pll_regs [ 5 ] = 0 ;
/* Bypass the PLL when disabled, otherwise registers will become
* inaccessible . */
regmap_update_bits ( adau - > regmap , ADAU17X1_CLOCK_CONTROL ,
ADAU17X1_CLOCK_CONTROL_CORECLK_SRC_PLL , 0 ) ;
}
/* The PLL register is 6 bytes long and can only be written at once. */
ret = regmap_raw_write ( adau - > regmap , ADAU17X1_PLL_CONTROL ,
adau - > pll_regs , ARRAY_SIZE ( adau - > pll_regs ) ) ;
if ( SND_SOC_DAPM_EVENT_ON ( event ) ) {
mdelay ( 5 ) ;
regmap_update_bits ( adau - > regmap , ADAU17X1_CLOCK_CONTROL ,
ADAU17X1_CLOCK_CONTROL_CORECLK_SRC_PLL ,
ADAU17X1_CLOCK_CONTROL_CORECLK_SRC_PLL ) ;
}
return 0 ;
}
static const char * const adau17x1_mono_stereo_text [ ] = {
" Stereo " ,
" Mono Left Channel (L+R) " ,
" Mono Right Channel (L+R) " ,
" Mono (L+R) " ,
} ;
static SOC_ENUM_SINGLE_DECL ( adau17x1_dac_mode_enum ,
ADAU17X1_DAC_CONTROL0 , 6 , adau17x1_mono_stereo_text ) ;
static const struct snd_kcontrol_new adau17x1_dac_mode_mux =
SOC_DAPM_ENUM ( " DAC Mono-Stereo-Mode " , adau17x1_dac_mode_enum ) ;
static const struct snd_soc_dapm_widget adau17x1_dapm_widgets [ ] = {
SND_SOC_DAPM_SUPPLY_S ( " PLL " , 3 , SND_SOC_NOPM , 0 , 0 , adau17x1_pll_event ,
SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD ) ,
SND_SOC_DAPM_SUPPLY ( " AIFCLK " , SND_SOC_NOPM , 0 , 0 , NULL , 0 ) ,
SND_SOC_DAPM_SUPPLY ( " MICBIAS " , ADAU17X1_MICBIAS , 0 , 0 , NULL , 0 ) ,
SND_SOC_DAPM_SUPPLY ( " Left Playback Enable " , ADAU17X1_PLAY_POWER_MGMT ,
0 , 0 , NULL , 0 ) ,
SND_SOC_DAPM_SUPPLY ( " Right Playback Enable " , ADAU17X1_PLAY_POWER_MGMT ,
1 , 0 , NULL , 0 ) ,
SND_SOC_DAPM_MUX ( " Left DAC Mode Mux " , SND_SOC_NOPM , 0 , 0 ,
& adau17x1_dac_mode_mux ) ,
SND_SOC_DAPM_MUX ( " Right DAC Mode Mux " , SND_SOC_NOPM , 0 , 0 ,
& adau17x1_dac_mode_mux ) ,
SND_SOC_DAPM_ADC ( " Left Decimator " , NULL , ADAU17X1_ADC_CONTROL , 0 , 0 ) ,
SND_SOC_DAPM_ADC ( " Right Decimator " , NULL , ADAU17X1_ADC_CONTROL , 1 , 0 ) ,
SND_SOC_DAPM_DAC ( " Left DAC " , NULL , ADAU17X1_DAC_CONTROL0 , 0 , 0 ) ,
SND_SOC_DAPM_DAC ( " Right DAC " , NULL , ADAU17X1_DAC_CONTROL0 , 1 , 0 ) ,
} ;
static const struct snd_soc_dapm_route adau17x1_dapm_routes [ ] = {
{ " Left Decimator " , NULL , " SYSCLK " } ,
{ " Right Decimator " , NULL , " SYSCLK " } ,
{ " Left DAC " , NULL , " SYSCLK " } ,
{ " Right DAC " , NULL , " SYSCLK " } ,
{ " Capture " , NULL , " SYSCLK " } ,
{ " Playback " , NULL , " SYSCLK " } ,
{ " Left DAC " , NULL , " Left DAC Mode Mux " } ,
{ " Right DAC " , NULL , " Right DAC Mode Mux " } ,
{ " Capture " , NULL , " AIFCLK " } ,
{ " Playback " , NULL , " AIFCLK " } ,
} ;
static const struct snd_soc_dapm_route adau17x1_dapm_pll_route = {
" SYSCLK " , NULL , " PLL " ,
} ;
/*
* The MUX register for the Capture and Playback MUXs selects either DSP as
* source / destination or one of the TDM slots . The TDM slot is selected via
* snd_soc_dai_set_tdm_slot ( ) , so we only expose whether to go to the DSP or
* directly to the DAI interface with this control .
*/
static int adau17x1_dsp_mux_enum_put ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
struct snd_soc_codec * codec = snd_soc_dapm_kcontrol_codec ( kcontrol ) ;
2015-05-04 19:46:09 +03:00
struct snd_soc_dapm_context * dapm = snd_soc_codec_get_dapm ( codec ) ;
2014-05-27 12:53:17 +04:00
struct adau * adau = snd_soc_codec_get_drvdata ( codec ) ;
struct soc_enum * e = ( struct soc_enum * ) kcontrol - > private_value ;
struct snd_soc_dapm_update update ;
unsigned int stream = e - > shift_l ;
unsigned int val , change ;
int reg ;
if ( ucontrol - > value . enumerated . item [ 0 ] > = e - > items )
return - EINVAL ;
switch ( ucontrol - > value . enumerated . item [ 0 ] ) {
case 0 :
val = 0 ;
adau - > dsp_bypass [ stream ] = false ;
break ;
default :
val = ( adau - > tdm_slot [ stream ] * 2 ) + 1 ;
adau - > dsp_bypass [ stream ] = true ;
break ;
}
if ( stream = = SNDRV_PCM_STREAM_PLAYBACK )
reg = ADAU17X1_SERIAL_INPUT_ROUTE ;
else
reg = ADAU17X1_SERIAL_OUTPUT_ROUTE ;
change = snd_soc_test_bits ( codec , reg , 0xff , val ) ;
if ( change ) {
update . kcontrol = kcontrol ;
update . mask = 0xff ;
update . reg = reg ;
update . val = val ;
2015-05-04 19:46:09 +03:00
snd_soc_dapm_mux_update_power ( dapm , kcontrol ,
2014-05-27 12:53:17 +04:00
ucontrol - > value . enumerated . item [ 0 ] , e , & update ) ;
}
return change ;
}
static int adau17x1_dsp_mux_enum_get ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
{
struct snd_soc_codec * codec = snd_soc_dapm_kcontrol_codec ( kcontrol ) ;
struct adau * adau = snd_soc_codec_get_drvdata ( codec ) ;
struct soc_enum * e = ( struct soc_enum * ) kcontrol - > private_value ;
unsigned int stream = e - > shift_l ;
unsigned int reg , val ;
int ret ;
if ( stream = = SNDRV_PCM_STREAM_PLAYBACK )
reg = ADAU17X1_SERIAL_INPUT_ROUTE ;
else
reg = ADAU17X1_SERIAL_OUTPUT_ROUTE ;
ret = regmap_read ( adau - > regmap , reg , & val ) ;
if ( ret )
return ret ;
if ( val ! = 0 )
val = 1 ;
ucontrol - > value . enumerated . item [ 0 ] = val ;
return 0 ;
}
# define DECLARE_ADAU17X1_DSP_MUX_CTRL(_name, _label, _stream, _text) \
const struct snd_kcontrol_new _name = \
SOC_DAPM_ENUM_EXT ( _label , ( const struct soc_enum ) \
SOC_ENUM_SINGLE ( SND_SOC_NOPM , _stream , \
ARRAY_SIZE ( _text ) , _text ) , \
adau17x1_dsp_mux_enum_get , adau17x1_dsp_mux_enum_put )
static const char * const adau17x1_dac_mux_text [ ] = {
" DSP " ,
" AIFIN " ,
} ;
static const char * const adau17x1_capture_mux_text [ ] = {
" DSP " ,
" Decimator " ,
} ;
static DECLARE_ADAU17X1_DSP_MUX_CTRL ( adau17x1_dac_mux , " DAC Playback Mux " ,
SNDRV_PCM_STREAM_PLAYBACK , adau17x1_dac_mux_text ) ;
static DECLARE_ADAU17X1_DSP_MUX_CTRL ( adau17x1_capture_mux , " Capture Mux " ,
SNDRV_PCM_STREAM_CAPTURE , adau17x1_capture_mux_text ) ;
static const struct snd_soc_dapm_widget adau17x1_dsp_dapm_widgets [ ] = {
SND_SOC_DAPM_PGA ( " DSP " , ADAU17X1_DSP_RUN , 0 , 0 , NULL , 0 ) ,
SND_SOC_DAPM_SIGGEN ( " DSP Siggen " ) ,
SND_SOC_DAPM_MUX ( " DAC Playback Mux " , SND_SOC_NOPM , 0 , 0 ,
& adau17x1_dac_mux ) ,
SND_SOC_DAPM_MUX ( " Capture Mux " , SND_SOC_NOPM , 0 , 0 ,
& adau17x1_capture_mux ) ,
} ;
static const struct snd_soc_dapm_route adau17x1_dsp_dapm_routes [ ] = {
{ " DAC Playback Mux " , " DSP " , " DSP " } ,
{ " DAC Playback Mux " , " AIFIN " , " Playback " } ,
{ " Left DAC Mode Mux " , " Stereo " , " DAC Playback Mux " } ,
{ " Left DAC Mode Mux " , " Mono (L+R) " , " DAC Playback Mux " } ,
{ " Left DAC Mode Mux " , " Mono Left Channel (L+R) " , " DAC Playback Mux " } ,
{ " Right DAC Mode Mux " , " Stereo " , " DAC Playback Mux " } ,
{ " Right DAC Mode Mux " , " Mono (L+R) " , " DAC Playback Mux " } ,
{ " Right DAC Mode Mux " , " Mono Right Channel (L+R) " , " DAC Playback Mux " } ,
{ " Capture Mux " , " DSP " , " DSP " } ,
{ " Capture Mux " , " Decimator " , " Left Decimator " } ,
{ " Capture Mux " , " Decimator " , " Right Decimator " } ,
{ " Capture " , NULL , " Capture Mux " } ,
{ " DSP " , NULL , " DSP Siggen " } ,
{ " DSP " , NULL , " Left Decimator " } ,
{ " DSP " , NULL , " Right Decimator " } ,
} ;
static const struct snd_soc_dapm_route adau17x1_no_dsp_dapm_routes [ ] = {
{ " Left DAC Mode Mux " , " Stereo " , " Playback " } ,
{ " Left DAC Mode Mux " , " Mono (L+R) " , " Playback " } ,
{ " Left DAC Mode Mux " , " Mono Left Channel (L+R) " , " Playback " } ,
{ " Right DAC Mode Mux " , " Stereo " , " Playback " } ,
{ " Right DAC Mode Mux " , " Mono (L+R) " , " Playback " } ,
{ " Right DAC Mode Mux " , " Mono Right Channel (L+R) " , " Playback " } ,
{ " Capture " , NULL , " Left Decimator " } ,
{ " Capture " , NULL , " Right Decimator " } ,
} ;
bool adau17x1_has_dsp ( struct adau * adau )
{
switch ( adau - > type ) {
case ADAU1761 :
case ADAU1381 :
case ADAU1781 :
return true ;
default :
return false ;
}
}
EXPORT_SYMBOL_GPL ( adau17x1_has_dsp ) ;
static int adau17x1_hw_params ( struct snd_pcm_substream * substream ,
struct snd_pcm_hw_params * params , struct snd_soc_dai * dai )
{
struct snd_soc_codec * codec = dai - > codec ;
struct adau * adau = snd_soc_codec_get_drvdata ( codec ) ;
unsigned int val , div , dsp_div ;
unsigned int freq ;
2014-11-19 20:29:05 +03:00
int ret ;
2014-05-27 12:53:17 +04:00
if ( adau - > clk_src = = ADAU17X1_CLK_SRC_PLL )
freq = adau - > pll_freq ;
else
freq = adau - > sysclk ;
if ( freq % params_rate ( params ) ! = 0 )
return - EINVAL ;
switch ( freq / params_rate ( params ) ) {
case 1024 : /* fs */
div = 0 ;
dsp_div = 1 ;
break ;
case 6144 : /* fs / 6 */
div = 1 ;
dsp_div = 6 ;
break ;
case 4096 : /* fs / 4 */
div = 2 ;
dsp_div = 5 ;
break ;
case 3072 : /* fs / 3 */
div = 3 ;
dsp_div = 4 ;
break ;
case 2048 : /* fs / 2 */
div = 4 ;
dsp_div = 3 ;
break ;
case 1536 : /* fs / 1.5 */
div = 5 ;
dsp_div = 2 ;
break ;
case 512 : /* fs / 0.5 */
div = 6 ;
dsp_div = 0 ;
break ;
default :
return - EINVAL ;
}
regmap_update_bits ( adau - > regmap , ADAU17X1_CONVERTER0 ,
ADAU17X1_CONVERTER0_CONVSR_MASK , div ) ;
if ( adau17x1_has_dsp ( adau ) ) {
regmap_write ( adau - > regmap , ADAU17X1_SERIAL_SAMPLING_RATE , div ) ;
regmap_write ( adau - > regmap , ADAU17X1_DSP_SAMPLING_RATE , dsp_div ) ;
}
2014-11-19 20:29:05 +03:00
if ( adau - > sigmadsp ) {
ret = adau17x1_setup_firmware ( adau , params_rate ( params ) ) ;
if ( ret < 0 )
return ret ;
}
2014-05-27 12:53:17 +04:00
if ( adau - > dai_fmt ! = SND_SOC_DAIFMT_RIGHT_J )
return 0 ;
2014-07-31 15:28:24 +04:00
switch ( params_width ( params ) ) {
case 16 :
2014-05-27 12:53:17 +04:00
val = ADAU17X1_SERIAL_PORT1_DELAY16 ;
break ;
2014-07-31 15:28:24 +04:00
case 24 :
2014-05-27 12:53:17 +04:00
val = ADAU17X1_SERIAL_PORT1_DELAY8 ;
break ;
2014-07-31 15:28:24 +04:00
case 32 :
2014-05-27 12:53:17 +04:00
val = ADAU17X1_SERIAL_PORT1_DELAY0 ;
break ;
default :
return - EINVAL ;
}
return regmap_update_bits ( adau - > regmap , ADAU17X1_SERIAL_PORT1 ,
ADAU17X1_SERIAL_PORT1_DELAY_MASK , val ) ;
}
static int adau17x1_set_dai_pll ( struct snd_soc_dai * dai , int pll_id ,
int source , unsigned int freq_in , unsigned int freq_out )
{
struct snd_soc_codec * codec = dai - > codec ;
struct adau * adau = snd_soc_codec_get_drvdata ( codec ) ;
unsigned int r , n , m , i , j ;
unsigned int div ;
int ret ;
if ( freq_in < 8000000 | | freq_in > 27000000 )
return - EINVAL ;
if ( ! freq_out ) {
r = 0 ;
n = 0 ;
m = 0 ;
div = 0 ;
} else {
if ( freq_out % freq_in ! = 0 ) {
div = DIV_ROUND_UP ( freq_in , 13500000 ) ;
freq_in / = div ;
r = freq_out / freq_in ;
i = freq_out % freq_in ;
j = gcd ( i , freq_in ) ;
n = i / j ;
m = freq_in / j ;
div - - ;
} else {
r = freq_out / freq_in ;
n = 0 ;
m = 0 ;
div = 0 ;
}
if ( n > 0xffff | | m > 0xffff | | div > 3 | | r > 8 | | r < 2 )
return - EINVAL ;
}
adau - > pll_regs [ 0 ] = m > > 8 ;
adau - > pll_regs [ 1 ] = m & 0xff ;
adau - > pll_regs [ 2 ] = n > > 8 ;
adau - > pll_regs [ 3 ] = n & 0xff ;
adau - > pll_regs [ 4 ] = ( r < < 3 ) | ( div < < 1 ) ;
if ( m ! = 0 )
adau - > pll_regs [ 4 ] | = 1 ; /* Fractional mode */
/* The PLL register is 6 bytes long and can only be written at once. */
ret = regmap_raw_write ( adau - > regmap , ADAU17X1_PLL_CONTROL ,
adau - > pll_regs , ARRAY_SIZE ( adau - > pll_regs ) ) ;
if ( ret )
return ret ;
adau - > pll_freq = freq_out ;
return 0 ;
}
static int adau17x1_set_dai_sysclk ( struct snd_soc_dai * dai ,
int clk_id , unsigned int freq , int dir )
{
2015-05-04 19:46:09 +03:00
struct snd_soc_dapm_context * dapm = snd_soc_codec_get_dapm ( dai - > codec ) ;
2014-05-27 12:53:17 +04:00
struct adau * adau = snd_soc_codec_get_drvdata ( dai - > codec ) ;
switch ( clk_id ) {
case ADAU17X1_CLK_SRC_MCLK :
case ADAU17X1_CLK_SRC_PLL :
break ;
default :
return - EINVAL ;
}
adau - > sysclk = freq ;
if ( adau - > clk_src ! = clk_id ) {
if ( clk_id = = ADAU17X1_CLK_SRC_PLL ) {
snd_soc_dapm_add_routes ( dapm ,
& adau17x1_dapm_pll_route , 1 ) ;
} else {
snd_soc_dapm_del_routes ( dapm ,
& adau17x1_dapm_pll_route , 1 ) ;
}
}
adau - > clk_src = clk_id ;
return 0 ;
}
static int adau17x1_set_dai_fmt ( struct snd_soc_dai * dai ,
unsigned int fmt )
{
struct adau * adau = snd_soc_codec_get_drvdata ( dai - > codec ) ;
unsigned int ctrl0 , ctrl1 ;
int lrclk_pol ;
switch ( fmt & SND_SOC_DAIFMT_MASTER_MASK ) {
case SND_SOC_DAIFMT_CBM_CFM :
ctrl0 = ADAU17X1_SERIAL_PORT0_MASTER ;
adau - > master = true ;
break ;
case SND_SOC_DAIFMT_CBS_CFS :
ctrl0 = 0 ;
adau - > master = false ;
break ;
default :
return - EINVAL ;
}
switch ( fmt & SND_SOC_DAIFMT_FORMAT_MASK ) {
case SND_SOC_DAIFMT_I2S :
lrclk_pol = 0 ;
ctrl1 = ADAU17X1_SERIAL_PORT1_DELAY1 ;
break ;
case SND_SOC_DAIFMT_LEFT_J :
case SND_SOC_DAIFMT_RIGHT_J :
lrclk_pol = 1 ;
ctrl1 = ADAU17X1_SERIAL_PORT1_DELAY0 ;
break ;
case SND_SOC_DAIFMT_DSP_A :
lrclk_pol = 1 ;
ctrl0 | = ADAU17X1_SERIAL_PORT0_PULSE_MODE ;
ctrl1 = ADAU17X1_SERIAL_PORT1_DELAY1 ;
break ;
case SND_SOC_DAIFMT_DSP_B :
lrclk_pol = 1 ;
ctrl0 | = ADAU17X1_SERIAL_PORT0_PULSE_MODE ;
ctrl1 = ADAU17X1_SERIAL_PORT1_DELAY0 ;
break ;
default :
return - EINVAL ;
}
switch ( fmt & SND_SOC_DAIFMT_INV_MASK ) {
case SND_SOC_DAIFMT_NB_NF :
break ;
case SND_SOC_DAIFMT_IB_NF :
ctrl0 | = ADAU17X1_SERIAL_PORT0_BCLK_POL ;
break ;
case SND_SOC_DAIFMT_NB_IF :
lrclk_pol = ! lrclk_pol ;
break ;
case SND_SOC_DAIFMT_IB_IF :
ctrl0 | = ADAU17X1_SERIAL_PORT0_BCLK_POL ;
lrclk_pol = ! lrclk_pol ;
break ;
default :
return - EINVAL ;
}
if ( lrclk_pol )
ctrl0 | = ADAU17X1_SERIAL_PORT0_LRCLK_POL ;
regmap_write ( adau - > regmap , ADAU17X1_SERIAL_PORT0 , ctrl0 ) ;
regmap_write ( adau - > regmap , ADAU17X1_SERIAL_PORT1 , ctrl1 ) ;
adau - > dai_fmt = fmt & SND_SOC_DAIFMT_FORMAT_MASK ;
return 0 ;
}
static int adau17x1_set_dai_tdm_slot ( struct snd_soc_dai * dai ,
unsigned int tx_mask , unsigned int rx_mask , int slots , int slot_width )
{
struct adau * adau = snd_soc_codec_get_drvdata ( dai - > codec ) ;
unsigned int ser_ctrl0 , ser_ctrl1 ;
unsigned int conv_ctrl0 , conv_ctrl1 ;
/* I2S mode */
if ( slots = = 0 ) {
slots = 2 ;
rx_mask = 3 ;
tx_mask = 3 ;
slot_width = 32 ;
}
switch ( slots ) {
case 2 :
ser_ctrl0 = ADAU17X1_SERIAL_PORT0_STEREO ;
break ;
case 4 :
ser_ctrl0 = ADAU17X1_SERIAL_PORT0_TDM4 ;
break ;
case 8 :
if ( adau - > type = = ADAU1361 )
return - EINVAL ;
ser_ctrl0 = ADAU17X1_SERIAL_PORT0_TDM8 ;
break ;
default :
return - EINVAL ;
}
switch ( slot_width * slots ) {
case 32 :
if ( adau - > type = = ADAU1761 )
return - EINVAL ;
ser_ctrl1 = ADAU17X1_SERIAL_PORT1_BCLK32 ;
break ;
case 64 :
ser_ctrl1 = ADAU17X1_SERIAL_PORT1_BCLK64 ;
break ;
case 48 :
ser_ctrl1 = ADAU17X1_SERIAL_PORT1_BCLK48 ;
break ;
case 128 :
ser_ctrl1 = ADAU17X1_SERIAL_PORT1_BCLK128 ;
break ;
case 256 :
if ( adau - > type = = ADAU1361 )
return - EINVAL ;
ser_ctrl1 = ADAU17X1_SERIAL_PORT1_BCLK256 ;
break ;
default :
return - EINVAL ;
}
switch ( rx_mask ) {
case 0x03 :
conv_ctrl1 = ADAU17X1_CONVERTER1_ADC_PAIR ( 1 ) ;
adau - > tdm_slot [ SNDRV_PCM_STREAM_CAPTURE ] = 0 ;
break ;
case 0x0c :
conv_ctrl1 = ADAU17X1_CONVERTER1_ADC_PAIR ( 2 ) ;
adau - > tdm_slot [ SNDRV_PCM_STREAM_CAPTURE ] = 1 ;
break ;
case 0x30 :
conv_ctrl1 = ADAU17X1_CONVERTER1_ADC_PAIR ( 3 ) ;
adau - > tdm_slot [ SNDRV_PCM_STREAM_CAPTURE ] = 2 ;
break ;
case 0xc0 :
conv_ctrl1 = ADAU17X1_CONVERTER1_ADC_PAIR ( 4 ) ;
adau - > tdm_slot [ SNDRV_PCM_STREAM_CAPTURE ] = 3 ;
break ;
default :
return - EINVAL ;
}
switch ( tx_mask ) {
case 0x03 :
conv_ctrl0 = ADAU17X1_CONVERTER0_DAC_PAIR ( 1 ) ;
adau - > tdm_slot [ SNDRV_PCM_STREAM_PLAYBACK ] = 0 ;
break ;
case 0x0c :
conv_ctrl0 = ADAU17X1_CONVERTER0_DAC_PAIR ( 2 ) ;
adau - > tdm_slot [ SNDRV_PCM_STREAM_PLAYBACK ] = 1 ;
break ;
case 0x30 :
conv_ctrl0 = ADAU17X1_CONVERTER0_DAC_PAIR ( 3 ) ;
adau - > tdm_slot [ SNDRV_PCM_STREAM_PLAYBACK ] = 2 ;
break ;
case 0xc0 :
conv_ctrl0 = ADAU17X1_CONVERTER0_DAC_PAIR ( 4 ) ;
adau - > tdm_slot [ SNDRV_PCM_STREAM_PLAYBACK ] = 3 ;
break ;
default :
return - EINVAL ;
}
regmap_update_bits ( adau - > regmap , ADAU17X1_CONVERTER0 ,
ADAU17X1_CONVERTER0_DAC_PAIR_MASK , conv_ctrl0 ) ;
regmap_update_bits ( adau - > regmap , ADAU17X1_CONVERTER1 ,
ADAU17X1_CONVERTER1_ADC_PAIR_MASK , conv_ctrl1 ) ;
regmap_update_bits ( adau - > regmap , ADAU17X1_SERIAL_PORT0 ,
ADAU17X1_SERIAL_PORT0_TDM_MASK , ser_ctrl0 ) ;
regmap_update_bits ( adau - > regmap , ADAU17X1_SERIAL_PORT1 ,
ADAU17X1_SERIAL_PORT1_BCLK_MASK , ser_ctrl1 ) ;
if ( ! adau17x1_has_dsp ( adau ) )
return 0 ;
if ( adau - > dsp_bypass [ SNDRV_PCM_STREAM_PLAYBACK ] ) {
regmap_write ( adau - > regmap , ADAU17X1_SERIAL_INPUT_ROUTE ,
( adau - > tdm_slot [ SNDRV_PCM_STREAM_PLAYBACK ] * 2 ) + 1 ) ;
}
if ( adau - > dsp_bypass [ SNDRV_PCM_STREAM_CAPTURE ] ) {
regmap_write ( adau - > regmap , ADAU17X1_SERIAL_OUTPUT_ROUTE ,
( adau - > tdm_slot [ SNDRV_PCM_STREAM_CAPTURE ] * 2 ) + 1 ) ;
}
return 0 ;
}
2014-11-19 20:29:05 +03:00
static int adau17x1_startup ( struct snd_pcm_substream * substream ,
struct snd_soc_dai * dai )
{
struct adau * adau = snd_soc_codec_get_drvdata ( dai - > codec ) ;
if ( adau - > sigmadsp )
return sigmadsp_restrict_params ( adau - > sigmadsp , substream ) ;
return 0 ;
}
2014-05-27 12:53:17 +04:00
const struct snd_soc_dai_ops adau17x1_dai_ops = {
. hw_params = adau17x1_hw_params ,
. set_sysclk = adau17x1_set_dai_sysclk ,
. set_fmt = adau17x1_set_dai_fmt ,
. set_pll = adau17x1_set_dai_pll ,
. set_tdm_slot = adau17x1_set_dai_tdm_slot ,
2014-11-19 20:29:05 +03:00
. startup = adau17x1_startup ,
2014-05-27 12:53:17 +04:00
} ;
EXPORT_SYMBOL_GPL ( adau17x1_dai_ops ) ;
int adau17x1_set_micbias_voltage ( struct snd_soc_codec * codec ,
enum adau17x1_micbias_voltage micbias )
{
struct adau * adau = snd_soc_codec_get_drvdata ( codec ) ;
switch ( micbias ) {
case ADAU17X1_MICBIAS_0_90_AVDD :
case ADAU17X1_MICBIAS_0_65_AVDD :
break ;
default :
return - EINVAL ;
}
return regmap_write ( adau - > regmap , ADAU17X1_MICBIAS , micbias < < 2 ) ;
}
EXPORT_SYMBOL_GPL ( adau17x1_set_micbias_voltage ) ;
2014-11-21 20:53:51 +03:00
bool adau17x1_precious_register ( struct device * dev , unsigned int reg )
{
/* SigmaDSP parameter memory */
if ( reg < 0x400 )
return true ;
return false ;
}
EXPORT_SYMBOL_GPL ( adau17x1_precious_register ) ;
2014-05-27 12:53:17 +04:00
bool adau17x1_readable_register ( struct device * dev , unsigned int reg )
{
2014-11-21 20:53:51 +03:00
/* SigmaDSP parameter memory */
if ( reg < 0x400 )
return true ;
2014-05-27 12:53:17 +04:00
switch ( reg ) {
case ADAU17X1_CLOCK_CONTROL :
case ADAU17X1_PLL_CONTROL :
case ADAU17X1_REC_POWER_MGMT :
case ADAU17X1_MICBIAS :
case ADAU17X1_SERIAL_PORT0 :
case ADAU17X1_SERIAL_PORT1 :
case ADAU17X1_CONVERTER0 :
case ADAU17X1_CONVERTER1 :
case ADAU17X1_LEFT_INPUT_DIGITAL_VOL :
case ADAU17X1_RIGHT_INPUT_DIGITAL_VOL :
case ADAU17X1_ADC_CONTROL :
case ADAU17X1_PLAY_POWER_MGMT :
case ADAU17X1_DAC_CONTROL0 :
case ADAU17X1_DAC_CONTROL1 :
case ADAU17X1_DAC_CONTROL2 :
case ADAU17X1_SERIAL_PORT_PAD :
case ADAU17X1_CONTROL_PORT_PAD0 :
case ADAU17X1_CONTROL_PORT_PAD1 :
case ADAU17X1_DSP_SAMPLING_RATE :
case ADAU17X1_SERIAL_INPUT_ROUTE :
case ADAU17X1_SERIAL_OUTPUT_ROUTE :
case ADAU17X1_DSP_ENABLE :
case ADAU17X1_DSP_RUN :
case ADAU17X1_SERIAL_SAMPLING_RATE :
return true ;
default :
break ;
}
return false ;
}
EXPORT_SYMBOL_GPL ( adau17x1_readable_register ) ;
bool adau17x1_volatile_register ( struct device * dev , unsigned int reg )
{
/* SigmaDSP parameter and program memory */
if ( reg < 0x4000 )
return true ;
switch ( reg ) {
/* The PLL register is 6 bytes long */
case ADAU17X1_PLL_CONTROL :
case ADAU17X1_PLL_CONTROL + 1 :
case ADAU17X1_PLL_CONTROL + 2 :
case ADAU17X1_PLL_CONTROL + 3 :
case ADAU17X1_PLL_CONTROL + 4 :
case ADAU17X1_PLL_CONTROL + 5 :
return true ;
default :
break ;
}
return false ;
}
EXPORT_SYMBOL_GPL ( adau17x1_volatile_register ) ;
2014-11-19 20:29:05 +03:00
int adau17x1_setup_firmware ( struct adau * adau , unsigned int rate )
2014-05-27 12:53:17 +04:00
{
int ret ;
int dspsr ;
ret = regmap_read ( adau - > regmap , ADAU17X1_DSP_SAMPLING_RATE , & dspsr ) ;
if ( ret )
return ret ;
regmap_write ( adau - > regmap , ADAU17X1_DSP_ENABLE , 1 ) ;
regmap_write ( adau - > regmap , ADAU17X1_DSP_SAMPLING_RATE , 0xf ) ;
2014-11-19 20:29:05 +03:00
ret = sigmadsp_setup ( adau - > sigmadsp , rate ) ;
2014-05-27 12:53:17 +04:00
if ( ret ) {
regmap_write ( adau - > regmap , ADAU17X1_DSP_ENABLE , 0 ) ;
return ret ;
}
regmap_write ( adau - > regmap , ADAU17X1_DSP_SAMPLING_RATE , dspsr ) ;
return 0 ;
}
2014-11-19 20:29:05 +03:00
EXPORT_SYMBOL_GPL ( adau17x1_setup_firmware ) ;
2014-05-27 12:53:17 +04:00
int adau17x1_add_widgets ( struct snd_soc_codec * codec )
{
2015-05-04 19:46:09 +03:00
struct snd_soc_dapm_context * dapm = snd_soc_codec_get_dapm ( codec ) ;
2014-05-27 12:53:17 +04:00
struct adau * adau = snd_soc_codec_get_drvdata ( codec ) ;
int ret ;
ret = snd_soc_add_codec_controls ( codec , adau17x1_controls ,
ARRAY_SIZE ( adau17x1_controls ) ) ;
if ( ret )
return ret ;
2015-05-04 19:46:09 +03:00
ret = snd_soc_dapm_new_controls ( dapm , adau17x1_dapm_widgets ,
2014-05-27 12:53:17 +04:00
ARRAY_SIZE ( adau17x1_dapm_widgets ) ) ;
if ( ret )
return ret ;
if ( adau17x1_has_dsp ( adau ) ) {
2015-05-04 19:46:09 +03:00
ret = snd_soc_dapm_new_controls ( dapm , adau17x1_dsp_dapm_widgets ,
2014-05-27 12:53:17 +04:00
ARRAY_SIZE ( adau17x1_dsp_dapm_widgets ) ) ;
2014-11-19 20:29:05 +03:00
if ( ret )
return ret ;
if ( ! adau - > sigmadsp )
return 0 ;
ret = sigmadsp_attach ( adau - > sigmadsp , & codec - > component ) ;
if ( ret ) {
dev_err ( codec - > dev , " Failed to attach firmware: %d \n " ,
ret ) ;
return ret ;
}
2014-05-27 12:53:17 +04:00
}
2014-11-19 20:29:05 +03:00
return 0 ;
2014-05-27 12:53:17 +04:00
}
EXPORT_SYMBOL_GPL ( adau17x1_add_widgets ) ;
int adau17x1_add_routes ( struct snd_soc_codec * codec )
{
2015-05-04 19:46:09 +03:00
struct snd_soc_dapm_context * dapm = snd_soc_codec_get_dapm ( codec ) ;
2014-05-27 12:53:17 +04:00
struct adau * adau = snd_soc_codec_get_drvdata ( codec ) ;
int ret ;
2015-05-04 19:46:09 +03:00
ret = snd_soc_dapm_add_routes ( dapm , adau17x1_dapm_routes ,
2014-05-27 12:53:17 +04:00
ARRAY_SIZE ( adau17x1_dapm_routes ) ) ;
if ( ret )
return ret ;
if ( adau17x1_has_dsp ( adau ) ) {
2015-05-04 19:46:09 +03:00
ret = snd_soc_dapm_add_routes ( dapm , adau17x1_dsp_dapm_routes ,
2014-05-27 12:53:17 +04:00
ARRAY_SIZE ( adau17x1_dsp_dapm_routes ) ) ;
} else {
2015-05-04 19:46:09 +03:00
ret = snd_soc_dapm_add_routes ( dapm , adau17x1_no_dsp_dapm_routes ,
2014-05-27 12:53:17 +04:00
ARRAY_SIZE ( adau17x1_no_dsp_dapm_routes ) ) ;
}
return ret ;
}
EXPORT_SYMBOL_GPL ( adau17x1_add_routes ) ;
int adau17x1_resume ( struct snd_soc_codec * codec )
{
struct adau * adau = snd_soc_codec_get_drvdata ( codec ) ;
if ( adau - > switch_mode )
adau - > switch_mode ( codec - > dev ) ;
regcache_sync ( adau - > regmap ) ;
return 0 ;
}
EXPORT_SYMBOL_GPL ( adau17x1_resume ) ;
int adau17x1_probe ( struct device * dev , struct regmap * regmap ,
2014-11-19 20:29:05 +03:00
enum adau17x1_type type , void ( * switch_mode ) ( struct device * dev ) ,
const char * firmware_name )
2014-05-27 12:53:17 +04:00
{
struct adau * adau ;
if ( IS_ERR ( regmap ) )
return PTR_ERR ( regmap ) ;
adau = devm_kzalloc ( dev , sizeof ( * adau ) , GFP_KERNEL ) ;
if ( ! adau )
return - ENOMEM ;
adau - > regmap = regmap ;
adau - > switch_mode = switch_mode ;
adau - > type = type ;
dev_set_drvdata ( dev , adau ) ;
2014-11-19 20:29:05 +03:00
if ( firmware_name ) {
adau - > sigmadsp = devm_sigmadsp_init_regmap ( dev , regmap , NULL ,
firmware_name ) ;
if ( IS_ERR ( adau - > sigmadsp ) ) {
dev_warn ( dev , " Could not find firmware file: %ld \n " ,
PTR_ERR ( adau - > sigmadsp ) ) ;
adau - > sigmadsp = NULL ;
}
}
2014-05-27 12:53:17 +04:00
if ( switch_mode )
switch_mode ( dev ) ;
return 0 ;
}
EXPORT_SYMBOL_GPL ( adau17x1_probe ) ;
MODULE_DESCRIPTION ( " ASoC ADAU1X61/ADAU1X81 common code " ) ;
MODULE_AUTHOR ( " Lars-Peter Clausen <lars@metafoo.de> " ) ;
MODULE_LICENSE ( " GPL " ) ;