2019-05-24 12:04:09 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2014-05-27 10:53:17 +02:00
/*
* Common code for ADAU1X61 and ADAU1X81 codecs
*
* Copyright 2011 - 2014 Analog Devices Inc .
* Author : Lars - Peter Clausen < lars @ metafoo . de >
*/
# include <linux/module.h>
# include <linux/init.h>
2016-06-15 15:07:27 +02:00
# include <linux/clk.h>
2014-05-27 10:53:17 +02:00
# 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>
2018-08-21 13:07:49 +02:00
# include <asm/unaligned.h>
2014-05-27 10:53:17 +02:00
# include "sigmadsp.h"
# include "adau17x1.h"
2016-06-09 19:39:06 +02:00
# include "adau-utils.h"
2014-05-27 10:53:17 +02:00
2018-08-21 13:07:49 +02:00
# define ADAU17X1_SAFELOAD_TARGET_ADDRESS 0x0006
# define ADAU17X1_SAFELOAD_TRIGGER 0x0007
# define ADAU17X1_SAFELOAD_DATA 0x0001
# define ADAU17X1_SAFELOAD_DATA_SIZE 20
# define ADAU17X1_WORD_SIZE 4
2014-05-27 10:53:17 +02:00
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 ) ,
} ;
2018-08-13 09:33:58 +02:00
static int adau17x1_setup_firmware ( struct snd_soc_component * component ,
unsigned int rate ) ;
2014-05-27 10:53:17 +02:00
static int adau17x1_pll_event ( struct snd_soc_dapm_widget * w ,
struct snd_kcontrol * kcontrol , int event )
{
2018-01-29 04:12:24 +00:00
struct snd_soc_component * component = snd_soc_dapm_to_component ( w - > dapm ) ;
struct adau * adau = snd_soc_component_get_drvdata ( component ) ;
2014-05-27 10:53:17 +02:00
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. */
2016-12-08 23:01:31 +05:30
regmap_raw_write ( adau - > regmap , ADAU17X1_PLL_CONTROL ,
2014-05-27 10:53:17 +02:00
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 ;
}
2017-09-07 15:31:38 +02:00
static int adau17x1_adc_fixup ( struct snd_soc_dapm_widget * w ,
struct snd_kcontrol * kcontrol , int event )
{
2018-01-29 04:12:24 +00:00
struct snd_soc_component * component = snd_soc_dapm_to_component ( w - > dapm ) ;
struct adau * adau = snd_soc_component_get_drvdata ( component ) ;
2017-09-07 15:31:38 +02:00
/*
* If we are capturing , toggle the ADOSR bit in Converter Control 0 to
* avoid losing SNR ( workaround from ADI ) . This must be done after
* the ADC ( s ) have been enabled . According to the data sheet , it is
* normally illegal to set this bit when the sampling rate is 96 kHz ,
* but according to ADI it is acceptable for this workaround .
*/
regmap_update_bits ( adau - > regmap , ADAU17X1_CONVERTER0 ,
ADAU17X1_CONVERTER0_ADOSR , ADAU17X1_CONVERTER0_ADOSR ) ;
regmap_update_bits ( adau - > regmap , ADAU17X1_CONVERTER0 ,
ADAU17X1_CONVERTER0_ADOSR , 0 ) ;
return 0 ;
}
2014-05-27 10:53:17 +02:00
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 ) ,
2017-09-07 15:31:38 +02:00
SND_SOC_DAPM_ADC_E ( " Left Decimator " , NULL , ADAU17X1_ADC_CONTROL , 0 , 0 ,
adau17x1_adc_fixup , SND_SOC_DAPM_POST_PMU ) ,
2014-05-27 10:53:17 +02:00
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 )
{
2018-01-29 04:12:24 +00:00
struct snd_soc_component * component = snd_soc_dapm_kcontrol_component ( kcontrol ) ;
struct snd_soc_dapm_context * dapm = snd_soc_component_get_dapm ( component ) ;
struct adau * adau = snd_soc_component_get_drvdata ( component ) ;
2014-05-27 10:53:17 +02:00
struct soc_enum * e = ( struct soc_enum * ) kcontrol - > private_value ;
2018-02-14 13:39:03 -02:00
struct snd_soc_dapm_update update = { } ;
2014-05-27 10:53:17 +02:00
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 ;
2018-01-29 04:12:24 +00:00
change = snd_soc_component_test_bits ( component , reg , 0xff , val ) ;
2014-05-27 10:53:17 +02:00
if ( change ) {
update . kcontrol = kcontrol ;
update . mask = 0xff ;
update . reg = reg ;
update . val = val ;
2015-05-04 18:46:09 +02:00
snd_soc_dapm_mux_update_power ( dapm , kcontrol ,
2014-05-27 10:53:17 +02: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 )
{
2018-01-29 04:12:24 +00:00
struct snd_soc_component * component = snd_soc_dapm_kcontrol_component ( kcontrol ) ;
struct adau * adau = snd_soc_component_get_drvdata ( component ) ;
2014-05-27 10:53:17 +02:00
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 " } ,
2018-07-06 16:19:14 +03:00
{ " DSP " , NULL , " Playback " } ,
2014-05-27 10:53:17 +02:00
} ;
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 " } ,
} ;
2018-08-13 09:33:58 +02:00
static bool adau17x1_has_dsp ( struct adau * adau )
2014-05-27 10:53:17 +02:00
{
switch ( adau - > type ) {
case ADAU1761 :
case ADAU1381 :
case ADAU1781 :
return true ;
default :
return false ;
}
}
2022-04-28 18:46:35 +02:00
/* Chip has a DSP but we're pretending it doesn't. */
static bool adau17x1_has_disused_dsp ( struct adau * adau )
{
switch ( adau - > type ) {
case ADAU1761_AS_1361 :
return true ;
default :
return false ;
}
}
2018-08-21 13:07:49 +02:00
static bool adau17x1_has_safeload ( struct adau * adau )
{
switch ( adau - > type ) {
case ADAU1761 :
case ADAU1781 :
return true ;
default :
return false ;
}
}
2016-06-15 15:07:27 +02:00
static int adau17x1_set_dai_pll ( struct snd_soc_dai * dai , int pll_id ,
int source , unsigned int freq_in , unsigned int freq_out )
{
2018-01-29 04:12:24 +00:00
struct snd_soc_component * component = dai - > component ;
struct adau * adau = snd_soc_component_get_drvdata ( component ) ;
2016-06-15 15:07:27 +02:00
int ret ;
if ( freq_in < 8000000 | | freq_in > 27000000 )
return - EINVAL ;
ret = adau_calc_pll_cfg ( freq_in , freq_out , adau - > pll_regs ) ;
if ( ret < 0 )
return ret ;
/* 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 )
{
2018-01-29 04:12:24 +00:00
struct snd_soc_dapm_context * dapm = snd_soc_component_get_dapm ( dai - > component ) ;
struct adau * adau = snd_soc_component_get_drvdata ( dai - > component ) ;
2016-06-15 15:07:27 +02:00
bool is_pll ;
bool was_pll ;
switch ( clk_id ) {
case ADAU17X1_CLK_SRC_MCLK :
is_pll = false ;
break ;
case ADAU17X1_CLK_SRC_PLL_AUTO :
if ( ! adau - > mclk )
return - EINVAL ;
2020-07-08 20:03:59 -05:00
fallthrough ;
2016-06-15 15:07:27 +02:00
case ADAU17X1_CLK_SRC_PLL :
is_pll = true ;
break ;
default :
return - EINVAL ;
}
switch ( adau - > clk_src ) {
case ADAU17X1_CLK_SRC_MCLK :
was_pll = false ;
break ;
case ADAU17X1_CLK_SRC_PLL :
case ADAU17X1_CLK_SRC_PLL_AUTO :
was_pll = true ;
break ;
default :
return - EINVAL ;
}
adau - > sysclk = freq ;
if ( is_pll ! = was_pll ) {
if ( is_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_auto_pll ( struct snd_soc_dai * dai ,
struct snd_pcm_hw_params * params )
{
struct adau * adau = snd_soc_dai_get_drvdata ( dai ) ;
unsigned int pll_rate ;
switch ( params_rate ( params ) ) {
case 48000 :
case 8000 :
case 12000 :
case 16000 :
case 24000 :
case 32000 :
case 96000 :
pll_rate = 48000 * 1024 ;
break ;
case 44100 :
case 7350 :
case 11025 :
case 14700 :
case 22050 :
case 29400 :
case 88200 :
pll_rate = 44100 * 1024 ;
break ;
default :
return - EINVAL ;
}
return adau17x1_set_dai_pll ( dai , ADAU17X1_PLL , ADAU17X1_PLL_SRC_MCLK ,
clk_get_rate ( adau - > mclk ) , pll_rate ) ;
}
2014-05-27 10:53:17 +02:00
static int adau17x1_hw_params ( struct snd_pcm_substream * substream ,
struct snd_pcm_hw_params * params , struct snd_soc_dai * dai )
{
2018-01-29 04:12:24 +00:00
struct snd_soc_component * component = dai - > component ;
struct adau * adau = snd_soc_component_get_drvdata ( component ) ;
2014-05-27 10:53:17 +02:00
unsigned int val , div , dsp_div ;
unsigned int freq ;
2014-11-19 18:29:05 +01:00
int ret ;
2014-05-27 10:53:17 +02:00
2016-06-15 15:07:27 +02:00
switch ( adau - > clk_src ) {
case ADAU17X1_CLK_SRC_PLL_AUTO :
ret = adau17x1_auto_pll ( dai , params ) ;
if ( ret )
return ret ;
2020-07-08 20:03:59 -05:00
fallthrough ;
2016-06-15 15:07:27 +02:00
case ADAU17X1_CLK_SRC_PLL :
2014-05-27 10:53:17 +02:00
freq = adau - > pll_freq ;
2016-06-15 15:07:27 +02:00
break ;
default :
2014-05-27 10:53:17 +02:00
freq = adau - > sysclk ;
2016-06-15 15:07:27 +02:00
break ;
}
2014-05-27 10:53:17 +02:00
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 ) ;
2022-04-28 18:46:35 +02:00
if ( adau17x1_has_dsp ( adau ) | | adau17x1_has_disused_dsp ( adau ) )
2014-05-27 10:53:17 +02:00
regmap_write ( adau - > regmap , ADAU17X1_SERIAL_SAMPLING_RATE , div ) ;
2022-04-28 18:46:35 +02:00
if ( adau17x1_has_dsp ( adau ) )
2014-05-27 10:53:17 +02:00
regmap_write ( adau - > regmap , ADAU17X1_DSP_SAMPLING_RATE , dsp_div ) ;
2014-11-19 18:29:05 +01:00
if ( adau - > sigmadsp ) {
2018-04-09 15:13:35 +02:00
ret = adau17x1_setup_firmware ( component , params_rate ( params ) ) ;
2014-11-19 18:29:05 +01:00
if ( ret < 0 )
return ret ;
}
2014-05-27 10:53:17 +02:00
if ( adau - > dai_fmt ! = SND_SOC_DAIFMT_RIGHT_J )
return 0 ;
2014-07-31 12:28:24 +01:00
switch ( params_width ( params ) ) {
case 16 :
2014-05-27 10:53:17 +02:00
val = ADAU17X1_SERIAL_PORT1_DELAY16 ;
break ;
2014-07-31 12:28:24 +01:00
case 24 :
2014-05-27 10:53:17 +02:00
val = ADAU17X1_SERIAL_PORT1_DELAY8 ;
break ;
2014-07-31 12:28:24 +01:00
case 32 :
2014-05-27 10:53:17 +02: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_fmt ( struct snd_soc_dai * dai ,
unsigned int fmt )
{
2018-01-29 04:12:24 +00:00
struct adau * adau = snd_soc_component_get_drvdata ( dai - > component ) ;
2014-05-27 10:53:17 +02:00
unsigned int ctrl0 , ctrl1 ;
2021-04-22 15:02:26 +02:00
unsigned int ctrl0_mask ;
2014-05-27 10:53:17 +02:00
int lrclk_pol ;
2021-09-16 16:18:04 +01:00
switch ( fmt & SND_SOC_DAIFMT_CLOCK_PROVIDER_MASK ) {
case SND_SOC_DAIFMT_CBP_CFP :
2014-05-27 10:53:17 +02:00
ctrl0 = ADAU17X1_SERIAL_PORT0_MASTER ;
adau - > master = true ;
break ;
2021-09-16 16:18:04 +01:00
case SND_SOC_DAIFMT_CBC_CFC :
2014-05-27 10:53:17 +02:00
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 ;
2021-04-22 15:02:26 +02:00
/* Set the mask to update all relevant bits in ADAU17X1_SERIAL_PORT0 */
ctrl0_mask = ADAU17X1_SERIAL_PORT0_MASTER |
ADAU17X1_SERIAL_PORT0_LRCLK_POL |
ADAU17X1_SERIAL_PORT0_BCLK_POL |
ADAU17X1_SERIAL_PORT0_PULSE_MODE ;
regmap_update_bits ( adau - > regmap , ADAU17X1_SERIAL_PORT0 , ctrl0_mask ,
ctrl0 ) ;
regmap_update_bits ( adau - > regmap , ADAU17X1_SERIAL_PORT1 ,
ADAU17X1_SERIAL_PORT1_DELAY_MASK , ctrl1 ) ;
2014-05-27 10:53:17 +02:00
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 )
{
2018-01-29 04:12:24 +00:00
struct adau * adau = snd_soc_component_get_drvdata ( dai - > component ) ;
2014-05-27 10:53:17 +02:00
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 :
2022-04-28 18:46:35 +02:00
if ( adau - > type = = ADAU1761 | | adau - > type = = ADAU1761_AS_1361 )
2014-05-27 10:53:17 +02:00
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 ) ;
2022-04-28 18:46:35 +02:00
if ( ! adau17x1_has_dsp ( adau ) & & ! adau17x1_has_disused_dsp ( adau ) )
2014-05-27 10:53:17 +02:00
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 18:29:05 +01:00
static int adau17x1_startup ( struct snd_pcm_substream * substream ,
struct snd_soc_dai * dai )
{
2018-01-29 04:12:24 +00:00
struct adau * adau = snd_soc_component_get_drvdata ( dai - > component ) ;
2014-11-19 18:29:05 +01:00
if ( adau - > sigmadsp )
return sigmadsp_restrict_params ( adau - > sigmadsp , substream ) ;
return 0 ;
}
2014-05-27 10:53:17 +02: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 18:29:05 +01:00
. startup = adau17x1_startup ,
2014-05-27 10:53:17 +02:00
} ;
EXPORT_SYMBOL_GPL ( adau17x1_dai_ops ) ;
2018-01-29 04:12:24 +00:00
int adau17x1_set_micbias_voltage ( struct snd_soc_component * component ,
2014-05-27 10:53:17 +02:00
enum adau17x1_micbias_voltage micbias )
{
2018-01-29 04:12:24 +00:00
struct adau * adau = snd_soc_component_get_drvdata ( component ) ;
2014-05-27 10:53:17 +02:00
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 18:53:51 +01: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 10:53:17 +02:00
bool adau17x1_readable_register ( struct device * dev , unsigned int reg )
{
2014-11-21 18:53:51 +01:00
/* SigmaDSP parameter memory */
if ( reg < 0x400 )
return true ;
2014-05-27 10:53:17 +02: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 ) ;
2018-08-13 09:33:58 +02:00
static int adau17x1_setup_firmware ( struct snd_soc_component * component ,
2018-04-09 15:13:35 +02:00
unsigned int rate )
2014-05-27 10:53:17 +02:00
{
int ret ;
2018-04-09 15:13:35 +02:00
int dspsr , dsp_run ;
struct adau * adau = snd_soc_component_get_drvdata ( component ) ;
struct snd_soc_dapm_context * dapm = snd_soc_component_get_dapm ( component ) ;
2018-04-09 15:13:36 +02:00
/* Check if sample rate is the same as before. If it is there is no
* point in performing the below steps as the call to
* sigmadsp_setup ( . . . ) will return directly when it finds the sample
* rate to be the same as before . By checking this we can prevent an
* audiable popping noise which occours when toggling DSP_RUN .
*/
if ( adau - > sigmadsp - > current_samplerate = = rate )
return 0 ;
2018-04-09 15:13:35 +02:00
snd_soc_dapm_mutex_lock ( dapm ) ;
2014-05-27 10:53:17 +02:00
ret = regmap_read ( adau - > regmap , ADAU17X1_DSP_SAMPLING_RATE , & dspsr ) ;
if ( ret )
2018-04-09 15:13:35 +02:00
goto err ;
ret = regmap_read ( adau - > regmap , ADAU17X1_DSP_RUN , & dsp_run ) ;
if ( ret )
goto err ;
2014-05-27 10:53:17 +02:00
regmap_write ( adau - > regmap , ADAU17X1_DSP_ENABLE , 1 ) ;
regmap_write ( adau - > regmap , ADAU17X1_DSP_SAMPLING_RATE , 0xf ) ;
2018-04-09 15:13:35 +02:00
regmap_write ( adau - > regmap , ADAU17X1_DSP_RUN , 0 ) ;
2014-05-27 10:53:17 +02:00
2014-11-19 18:29:05 +01:00
ret = sigmadsp_setup ( adau - > sigmadsp , rate ) ;
2014-05-27 10:53:17 +02:00
if ( ret ) {
regmap_write ( adau - > regmap , ADAU17X1_DSP_ENABLE , 0 ) ;
2018-04-09 15:13:35 +02:00
goto err ;
2014-05-27 10:53:17 +02:00
}
regmap_write ( adau - > regmap , ADAU17X1_DSP_SAMPLING_RATE , dspsr ) ;
2018-04-09 15:13:35 +02:00
regmap_write ( adau - > regmap , ADAU17X1_DSP_RUN , dsp_run ) ;
2014-05-27 10:53:17 +02:00
2018-04-09 15:13:35 +02:00
err :
snd_soc_dapm_mutex_unlock ( dapm ) ;
return ret ;
2014-05-27 10:53:17 +02:00
}
2018-01-29 04:12:24 +00:00
int adau17x1_add_widgets ( struct snd_soc_component * component )
2014-05-27 10:53:17 +02:00
{
2018-01-29 04:12:24 +00:00
struct snd_soc_dapm_context * dapm = snd_soc_component_get_dapm ( component ) ;
struct adau * adau = snd_soc_component_get_drvdata ( component ) ;
2014-05-27 10:53:17 +02:00
int ret ;
2018-01-29 04:12:24 +00:00
ret = snd_soc_add_component_controls ( component , adau17x1_controls ,
2014-05-27 10:53:17 +02:00
ARRAY_SIZE ( adau17x1_controls ) ) ;
if ( ret )
return ret ;
2015-05-04 18:46:09 +02:00
ret = snd_soc_dapm_new_controls ( dapm , adau17x1_dapm_widgets ,
2014-05-27 10:53:17 +02:00
ARRAY_SIZE ( adau17x1_dapm_widgets ) ) ;
if ( ret )
return ret ;
if ( adau17x1_has_dsp ( adau ) ) {
2015-05-04 18:46:09 +02:00
ret = snd_soc_dapm_new_controls ( dapm , adau17x1_dsp_dapm_widgets ,
2014-05-27 10:53:17 +02:00
ARRAY_SIZE ( adau17x1_dsp_dapm_widgets ) ) ;
2014-11-19 18:29:05 +01:00
if ( ret )
return ret ;
if ( ! adau - > sigmadsp )
return 0 ;
2018-01-29 04:12:24 +00:00
ret = sigmadsp_attach ( adau - > sigmadsp , component ) ;
2014-11-19 18:29:05 +01:00
if ( ret ) {
2018-01-29 04:12:24 +00:00
dev_err ( component - > dev , " Failed to attach firmware: %d \n " ,
2014-11-19 18:29:05 +01:00
ret ) ;
return ret ;
}
2014-05-27 10:53:17 +02:00
}
2014-11-19 18:29:05 +01:00
return 0 ;
2014-05-27 10:53:17 +02:00
}
EXPORT_SYMBOL_GPL ( adau17x1_add_widgets ) ;
2018-01-29 04:12:24 +00:00
int adau17x1_add_routes ( struct snd_soc_component * component )
2014-05-27 10:53:17 +02:00
{
2018-01-29 04:12:24 +00:00
struct snd_soc_dapm_context * dapm = snd_soc_component_get_dapm ( component ) ;
struct adau * adau = snd_soc_component_get_drvdata ( component ) ;
2014-05-27 10:53:17 +02:00
int ret ;
2015-05-04 18:46:09 +02:00
ret = snd_soc_dapm_add_routes ( dapm , adau17x1_dapm_routes ,
2014-05-27 10:53:17 +02:00
ARRAY_SIZE ( adau17x1_dapm_routes ) ) ;
if ( ret )
return ret ;
if ( adau17x1_has_dsp ( adau ) ) {
2015-05-04 18:46:09 +02:00
ret = snd_soc_dapm_add_routes ( dapm , adau17x1_dsp_dapm_routes ,
2014-05-27 10:53:17 +02:00
ARRAY_SIZE ( adau17x1_dsp_dapm_routes ) ) ;
} else {
2015-05-04 18:46:09 +02:00
ret = snd_soc_dapm_add_routes ( dapm , adau17x1_no_dsp_dapm_routes ,
2014-05-27 10:53:17 +02:00
ARRAY_SIZE ( adau17x1_no_dsp_dapm_routes ) ) ;
}
2016-06-15 15:07:27 +02:00
if ( adau - > clk_src ! = ADAU17X1_CLK_SRC_MCLK )
snd_soc_dapm_add_routes ( dapm , & adau17x1_dapm_pll_route , 1 ) ;
2014-05-27 10:53:17 +02:00
return ret ;
}
EXPORT_SYMBOL_GPL ( adau17x1_add_routes ) ;
2018-01-29 04:12:24 +00:00
int adau17x1_resume ( struct snd_soc_component * component )
2014-05-27 10:53:17 +02:00
{
2018-01-29 04:12:24 +00:00
struct adau * adau = snd_soc_component_get_drvdata ( component ) ;
2014-05-27 10:53:17 +02:00
if ( adau - > switch_mode )
2018-01-29 04:12:24 +00:00
adau - > switch_mode ( component - > dev ) ;
2014-05-27 10:53:17 +02:00
regcache_sync ( adau - > regmap ) ;
return 0 ;
}
EXPORT_SYMBOL_GPL ( adau17x1_resume ) ;
2018-08-21 13:07:49 +02:00
static int adau17x1_safeload ( struct sigmadsp * sigmadsp , unsigned int addr ,
const uint8_t bytes [ ] , size_t len )
{
uint8_t buf [ ADAU17X1_WORD_SIZE ] ;
uint8_t data [ ADAU17X1_SAFELOAD_DATA_SIZE ] ;
unsigned int addr_offset ;
unsigned int nbr_words ;
int ret ;
/* write data to safeload addresses. Check if len is not a multiple of
* 4 bytes , if so we need to zero pad .
*/
nbr_words = len / ADAU17X1_WORD_SIZE ;
if ( ( len - nbr_words * ADAU17X1_WORD_SIZE ) = = 0 ) {
ret = regmap_raw_write ( sigmadsp - > control_data ,
ADAU17X1_SAFELOAD_DATA , bytes , len ) ;
} else {
nbr_words + + ;
memset ( data , 0 , ADAU17X1_SAFELOAD_DATA_SIZE ) ;
memcpy ( data , bytes , len ) ;
ret = regmap_raw_write ( sigmadsp - > control_data ,
ADAU17X1_SAFELOAD_DATA , data ,
nbr_words * ADAU17X1_WORD_SIZE ) ;
}
if ( ret < 0 )
return ret ;
/* Write target address, target address is offset by 1 */
addr_offset = addr - 1 ;
put_unaligned_be32 ( addr_offset , buf ) ;
ret = regmap_raw_write ( sigmadsp - > control_data ,
ADAU17X1_SAFELOAD_TARGET_ADDRESS , buf , ADAU17X1_WORD_SIZE ) ;
if ( ret < 0 )
return ret ;
/* write nbr of words to trigger address */
put_unaligned_be32 ( nbr_words , buf ) ;
ret = regmap_raw_write ( sigmadsp - > control_data ,
ADAU17X1_SAFELOAD_TRIGGER , buf , ADAU17X1_WORD_SIZE ) ;
if ( ret < 0 )
return ret ;
return 0 ;
}
static const struct sigmadsp_ops adau17x1_sigmadsp_ops = {
. safeload = adau17x1_safeload ,
} ;
2014-05-27 10:53:17 +02:00
int adau17x1_probe ( struct device * dev , struct regmap * regmap ,
2014-11-19 18:29:05 +01:00
enum adau17x1_type type , void ( * switch_mode ) ( struct device * dev ) ,
const char * firmware_name )
2014-05-27 10:53:17 +02:00
{
struct adau * adau ;
2016-06-15 15:07:27 +02:00
int ret ;
2014-05-27 10:53:17 +02:00
if ( IS_ERR ( regmap ) )
return PTR_ERR ( regmap ) ;
adau = devm_kzalloc ( dev , sizeof ( * adau ) , GFP_KERNEL ) ;
if ( ! adau )
return - ENOMEM ;
2016-06-15 15:07:27 +02:00
adau - > mclk = devm_clk_get ( dev , " mclk " ) ;
if ( IS_ERR ( adau - > mclk ) ) {
if ( PTR_ERR ( adau - > mclk ) ! = - ENOENT )
return PTR_ERR ( adau - > mclk ) ;
/* Clock is optional (for the driver) */
adau - > mclk = NULL ;
} else if ( adau - > mclk ) {
adau - > clk_src = ADAU17X1_CLK_SRC_PLL_AUTO ;
/*
* Any valid PLL output rate will work at this point , use one
* that is likely to be chosen later as well . The register will
* be written when the PLL is powered up for the first time .
*/
ret = adau_calc_pll_cfg ( clk_get_rate ( adau - > mclk ) , 48000 * 1024 ,
adau - > pll_regs ) ;
if ( ret < 0 )
return ret ;
ret = clk_prepare_enable ( adau - > mclk ) ;
if ( ret )
return ret ;
}
2014-05-27 10:53:17 +02:00
adau - > regmap = regmap ;
adau - > switch_mode = switch_mode ;
adau - > type = type ;
dev_set_drvdata ( dev , adau ) ;
2014-11-19 18:29:05 +01:00
if ( firmware_name ) {
2018-08-21 13:07:49 +02:00
if ( adau17x1_has_safeload ( adau ) ) {
adau - > sigmadsp = devm_sigmadsp_init_regmap ( dev , regmap ,
& adau17x1_sigmadsp_ops , firmware_name ) ;
} else {
adau - > sigmadsp = devm_sigmadsp_init_regmap ( dev , regmap ,
NULL , firmware_name ) ;
}
2014-11-19 18:29:05 +01:00
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 10:53:17 +02:00
if ( switch_mode )
switch_mode ( dev ) ;
return 0 ;
}
EXPORT_SYMBOL_GPL ( adau17x1_probe ) ;
2016-06-15 15:07:27 +02:00
void adau17x1_remove ( struct device * dev )
{
struct adau * adau = dev_get_drvdata ( dev ) ;
2021-01-08 08:44:56 +00:00
clk_disable_unprepare ( adau - > mclk ) ;
2016-06-15 15:07:27 +02:00
}
EXPORT_SYMBOL_GPL ( adau17x1_remove ) ;
2014-05-27 10:53:17 +02:00
MODULE_DESCRIPTION ( " ASoC ADAU1X61/ADAU1X81 common code " ) ;
MODULE_AUTHOR ( " Lars-Peter Clausen <lars@metafoo.de> " ) ;
MODULE_LICENSE ( " GPL " ) ;