2010-03-19 14:48:33 +08:00
/*
* AD193X Audio Codec driver supporting AD1936 / 7 / 8 / 9
*
* Copyright 2010 Analog Devices Inc .
*
* Licensed under the GPL - 2 or later .
*/
# include <linux/init.h>
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/device.h>
# include <linux/i2c.h>
# include <linux/spi/spi.h>
2010-03-29 15:32:18 +11:00
# include <linux/slab.h>
2010-03-19 14:48:33 +08:00
# include <sound/core.h>
# include <sound/pcm.h>
# include <sound/pcm_params.h>
# include <sound/initval.h>
# include <sound/soc.h>
# include <sound/tlv.h>
# include "ad193x.h"
/* codec private data */
struct ad193x_priv {
2011-09-05 20:46:33 +02:00
struct regmap * regmap ;
2010-03-17 20:15:21 +00:00
int sysclk ;
2010-03-19 14:48:33 +08:00
} ;
/*
* AD193X volume / mute / de - emphasis etc . controls
*/
2011-11-28 17:28:08 +01:00
static const char * const ad193x_deemp [ ] = { " None " , " 48kHz " , " 44.1kHz " , " 32kHz " } ;
2010-03-19 14:48:33 +08:00
static const struct soc_enum ad193x_deemp_enum =
SOC_ENUM_SINGLE ( AD193X_DAC_CTRL2 , 1 , 4 , ad193x_deemp ) ;
2011-11-28 17:28:07 +01:00
static const DECLARE_TLV_DB_MINMAX ( adau193x_tlv , - 9563 , 0 ) ;
2010-03-19 14:48:33 +08:00
static const struct snd_kcontrol_new ad193x_snd_controls [ ] = {
/* DAC volume control */
2011-11-28 17:28:07 +01:00
SOC_DOUBLE_R_TLV ( " DAC1 Volume " , AD193X_DAC_L1_VOL ,
AD193X_DAC_R1_VOL , 0 , 0xFF , 1 , adau193x_tlv ) ,
SOC_DOUBLE_R_TLV ( " DAC2 Volume " , AD193X_DAC_L2_VOL ,
AD193X_DAC_R2_VOL , 0 , 0xFF , 1 , adau193x_tlv ) ,
SOC_DOUBLE_R_TLV ( " DAC3 Volume " , AD193X_DAC_L3_VOL ,
AD193X_DAC_R3_VOL , 0 , 0xFF , 1 , adau193x_tlv ) ,
SOC_DOUBLE_R_TLV ( " DAC4 Volume " , AD193X_DAC_L4_VOL ,
AD193X_DAC_R4_VOL , 0 , 0xFF , 1 , adau193x_tlv ) ,
2010-03-19 14:48:33 +08:00
/* ADC switch control */
SOC_DOUBLE ( " ADC1 Switch " , AD193X_ADC_CTRL0 , AD193X_ADCL1_MUTE ,
AD193X_ADCR1_MUTE , 1 , 1 ) ,
SOC_DOUBLE ( " ADC2 Switch " , AD193X_ADC_CTRL0 , AD193X_ADCL2_MUTE ,
AD193X_ADCR2_MUTE , 1 , 1 ) ,
/* DAC switch control */
SOC_DOUBLE ( " DAC1 Switch " , AD193X_DAC_CHNL_MUTE , AD193X_DACL1_MUTE ,
AD193X_DACR1_MUTE , 1 , 1 ) ,
SOC_DOUBLE ( " DAC2 Switch " , AD193X_DAC_CHNL_MUTE , AD193X_DACL2_MUTE ,
AD193X_DACR2_MUTE , 1 , 1 ) ,
SOC_DOUBLE ( " DAC3 Switch " , AD193X_DAC_CHNL_MUTE , AD193X_DACL3_MUTE ,
AD193X_DACR3_MUTE , 1 , 1 ) ,
SOC_DOUBLE ( " DAC4 Switch " , AD193X_DAC_CHNL_MUTE , AD193X_DACL4_MUTE ,
AD193X_DACR4_MUTE , 1 , 1 ) ,
/* ADC high-pass filter */
SOC_SINGLE ( " ADC High Pass Filter Switch " , AD193X_ADC_CTRL0 ,
AD193X_ADC_HIGHPASS_FILTER , 1 , 0 ) ,
/* DAC de-emphasis */
SOC_ENUM ( " Playback Deemphasis " , ad193x_deemp_enum ) ,
} ;
static const struct snd_soc_dapm_widget ad193x_dapm_widgets [ ] = {
SND_SOC_DAPM_DAC ( " DAC " , " Playback " , AD193X_DAC_CTRL0 , 0 , 1 ) ,
SND_SOC_DAPM_ADC ( " ADC " , " Capture " , SND_SOC_NOPM , 0 , 0 ) ,
SND_SOC_DAPM_SUPPLY ( " PLL_PWR " , AD193X_PLL_CLK_CTRL0 , 0 , 1 , NULL , 0 ) ,
SND_SOC_DAPM_SUPPLY ( " ADC_PWR " , AD193X_ADC_CTRL0 , 0 , 1 , NULL , 0 ) ,
2011-11-28 17:28:10 +01:00
SND_SOC_DAPM_SUPPLY ( " SYSCLK " , AD193X_PLL_CLK_CTRL0 , 7 , 0 , NULL , 0 ) ,
2010-03-19 14:48:33 +08:00
SND_SOC_DAPM_OUTPUT ( " DAC1OUT " ) ,
SND_SOC_DAPM_OUTPUT ( " DAC2OUT " ) ,
SND_SOC_DAPM_OUTPUT ( " DAC3OUT " ) ,
SND_SOC_DAPM_OUTPUT ( " DAC4OUT " ) ,
SND_SOC_DAPM_INPUT ( " ADC1IN " ) ,
SND_SOC_DAPM_INPUT ( " ADC2IN " ) ,
} ;
static const struct snd_soc_dapm_route audio_paths [ ] = {
2011-11-28 17:28:10 +01:00
{ " DAC " , NULL , " SYSCLK " } ,
{ " ADC " , NULL , " SYSCLK " } ,
2010-03-19 14:48:33 +08:00
{ " DAC " , NULL , " ADC_PWR " } ,
{ " ADC " , NULL , " ADC_PWR " } ,
2011-11-28 17:28:09 +01:00
{ " DAC1OUT " , NULL , " DAC " } ,
{ " DAC2OUT " , NULL , " DAC " } ,
{ " DAC3OUT " , NULL , " DAC " } ,
{ " DAC4OUT " , NULL , " DAC " } ,
{ " ADC " , NULL , " ADC1IN " } ,
{ " ADC " , NULL , " ADC2IN " } ,
2011-11-28 17:28:10 +01:00
{ " SYSCLK " , NULL , " PLL_PWR " } ,
2010-03-19 14:48:33 +08:00
} ;
/*
* DAI ops entries
*/
static int ad193x_mute ( struct snd_soc_dai * dai , int mute )
{
2011-11-28 17:28:12 +01:00
struct ad193x_priv * ad193x = snd_soc_codec_get_drvdata ( dai - > codec ) ;
2010-03-19 14:48:33 +08:00
2011-10-18 06:25:08 +08:00
if ( mute )
2011-11-28 17:28:12 +01:00
regmap_update_bits ( ad193x - > regmap , AD193X_DAC_CTRL2 ,
2011-10-18 06:25:08 +08:00
AD193X_DAC_MASTER_MUTE ,
AD193X_DAC_MASTER_MUTE ) ;
else
2011-11-28 17:28:12 +01:00
regmap_update_bits ( ad193x - > regmap , AD193X_DAC_CTRL2 ,
2011-10-18 06:25:08 +08:00
AD193X_DAC_MASTER_MUTE , 0 ) ;
2010-03-19 14:48:33 +08:00
return 0 ;
}
static int ad193x_set_tdm_slot ( struct snd_soc_dai * dai , unsigned int tx_mask ,
unsigned int rx_mask , int slots , int width )
{
2011-11-28 17:28:12 +01:00
struct ad193x_priv * ad193x = snd_soc_codec_get_drvdata ( dai - > codec ) ;
2011-11-28 17:28:11 +01:00
unsigned int channels ;
2010-03-19 14:48:33 +08:00
switch ( slots ) {
case 2 :
2011-11-28 17:28:11 +01:00
channels = AD193X_2_CHANNELS ;
2010-03-19 14:48:33 +08:00
break ;
case 4 :
2011-11-28 17:28:11 +01:00
channels = AD193X_4_CHANNELS ;
2010-03-19 14:48:33 +08:00
break ;
case 8 :
2011-11-28 17:28:11 +01:00
channels = AD193X_8_CHANNELS ;
2010-03-19 14:48:33 +08:00
break ;
case 16 :
2011-11-28 17:28:11 +01:00
channels = AD193X_16_CHANNELS ;
2010-03-19 14:48:33 +08:00
break ;
default :
return - EINVAL ;
}
2011-11-28 17:28:12 +01:00
regmap_update_bits ( ad193x - > regmap , AD193X_DAC_CTRL1 ,
AD193X_DAC_CHAN_MASK , channels < < AD193X_DAC_CHAN_SHFT ) ;
regmap_update_bits ( ad193x - > regmap , AD193X_ADC_CTRL2 ,
AD193X_ADC_CHAN_MASK , channels < < AD193X_ADC_CHAN_SHFT ) ;
2010-03-19 14:48:33 +08:00
return 0 ;
}
static int ad193x_set_dai_fmt ( struct snd_soc_dai * codec_dai ,
unsigned int fmt )
{
2011-11-28 17:28:12 +01:00
struct ad193x_priv * ad193x = snd_soc_codec_get_drvdata ( codec_dai - > codec ) ;
2011-11-28 17:28:11 +01:00
unsigned int adc_serfmt = 0 ;
unsigned int adc_fmt = 0 ;
unsigned int dac_fmt = 0 ;
2010-03-19 14:48:33 +08:00
/* At present, the driver only support AUX ADC mode(SND_SOC_DAIFMT_I2S
* with TDM ) and ADC & DAC TDM mode ( SND_SOC_DAIFMT_DSP_A )
*/
switch ( fmt & SND_SOC_DAIFMT_FORMAT_MASK ) {
case SND_SOC_DAIFMT_I2S :
2011-11-28 17:28:11 +01:00
adc_serfmt | = AD193X_ADC_SERFMT_TDM ;
2010-03-19 14:48:33 +08:00
break ;
case SND_SOC_DAIFMT_DSP_A :
2011-11-28 17:28:11 +01:00
adc_serfmt | = AD193X_ADC_SERFMT_AUX ;
2010-03-19 14:48:33 +08:00
break ;
default :
return - EINVAL ;
}
switch ( fmt & SND_SOC_DAIFMT_INV_MASK ) {
case SND_SOC_DAIFMT_NB_NF : /* normal bit clock + frame */
break ;
case SND_SOC_DAIFMT_NB_IF : /* normal bclk + invert frm */
2011-11-28 17:28:11 +01:00
adc_fmt | = AD193X_ADC_LEFT_HIGH ;
dac_fmt | = AD193X_DAC_LEFT_HIGH ;
2010-03-19 14:48:33 +08:00
break ;
case SND_SOC_DAIFMT_IB_NF : /* invert bclk + normal frm */
2011-11-28 17:28:11 +01:00
adc_fmt | = AD193X_ADC_BCLK_INV ;
dac_fmt | = AD193X_DAC_BCLK_INV ;
2010-03-19 14:48:33 +08:00
break ;
case SND_SOC_DAIFMT_IB_IF : /* invert bclk + frm */
2011-11-28 17:28:11 +01:00
adc_fmt | = AD193X_ADC_LEFT_HIGH ;
adc_fmt | = AD193X_ADC_BCLK_INV ;
dac_fmt | = AD193X_DAC_LEFT_HIGH ;
dac_fmt | = AD193X_DAC_BCLK_INV ;
2010-03-19 14:48:33 +08:00
break ;
default :
return - EINVAL ;
}
switch ( fmt & SND_SOC_DAIFMT_MASTER_MASK ) {
case SND_SOC_DAIFMT_CBM_CFM : /* codec clk & frm master */
2011-11-28 17:28:11 +01:00
adc_fmt | = AD193X_ADC_LCR_MASTER ;
adc_fmt | = AD193X_ADC_BCLK_MASTER ;
dac_fmt | = AD193X_DAC_LCR_MASTER ;
dac_fmt | = AD193X_DAC_BCLK_MASTER ;
2010-03-19 14:48:33 +08:00
break ;
case SND_SOC_DAIFMT_CBS_CFM : /* codec clk slave & frm master */
2011-11-28 17:28:11 +01:00
adc_fmt | = AD193X_ADC_LCR_MASTER ;
dac_fmt | = AD193X_DAC_LCR_MASTER ;
2010-03-19 14:48:33 +08:00
break ;
case SND_SOC_DAIFMT_CBM_CFS : /* codec clk master & frame slave */
2011-11-28 17:28:11 +01:00
adc_fmt | = AD193X_ADC_BCLK_MASTER ;
dac_fmt | = AD193X_DAC_BCLK_MASTER ;
2010-03-19 14:48:33 +08:00
break ;
case SND_SOC_DAIFMT_CBS_CFS : /* codec clk & frm slave */
break ;
default :
return - EINVAL ;
}
2011-11-28 17:28:12 +01:00
regmap_update_bits ( ad193x - > regmap , AD193X_ADC_CTRL1 ,
AD193X_ADC_SERFMT_MASK , adc_serfmt ) ;
regmap_update_bits ( ad193x - > regmap , AD193X_ADC_CTRL2 ,
AD193X_ADC_FMT_MASK , adc_fmt ) ;
regmap_update_bits ( ad193x - > regmap , AD193X_DAC_CTRL1 ,
AD193X_DAC_FMT_MASK , dac_fmt ) ;
2010-03-19 14:48:33 +08:00
return 0 ;
}
2010-05-21 11:57:01 +08:00
static int ad193x_set_dai_sysclk ( struct snd_soc_dai * codec_dai ,
int clk_id , unsigned int freq , int dir )
{
struct snd_soc_codec * codec = codec_dai - > codec ;
struct ad193x_priv * ad193x = snd_soc_codec_get_drvdata ( codec ) ;
switch ( freq ) {
case 12288000 :
case 18432000 :
case 24576000 :
case 36864000 :
ad193x - > sysclk = freq ;
return 0 ;
}
return - EINVAL ;
}
2010-03-19 14:48:33 +08:00
static int ad193x_hw_params ( struct snd_pcm_substream * substream ,
struct snd_pcm_hw_params * params ,
struct snd_soc_dai * dai )
{
2011-10-18 06:25:08 +08:00
int word_len = 0 , master_rate = 0 ;
2010-03-19 14:48:33 +08:00
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
2010-03-17 20:15:21 +00:00
struct snd_soc_codec * codec = rtd - > codec ;
2010-05-21 11:57:01 +08:00
struct ad193x_priv * ad193x = snd_soc_codec_get_drvdata ( codec ) ;
2010-03-19 14:48:33 +08:00
/* bit size */
switch ( params_format ( params ) ) {
case SNDRV_PCM_FORMAT_S16_LE :
word_len = 3 ;
break ;
case SNDRV_PCM_FORMAT_S20_3LE :
word_len = 1 ;
break ;
case SNDRV_PCM_FORMAT_S24_LE :
case SNDRV_PCM_FORMAT_S32_LE :
word_len = 0 ;
break ;
}
2010-05-21 11:57:01 +08:00
switch ( ad193x - > sysclk ) {
case 12288000 :
master_rate = AD193X_PLL_INPUT_256 ;
break ;
case 18432000 :
master_rate = AD193X_PLL_INPUT_384 ;
break ;
case 24576000 :
master_rate = AD193X_PLL_INPUT_512 ;
break ;
case 36864000 :
master_rate = AD193X_PLL_INPUT_768 ;
break ;
}
2011-11-28 17:28:12 +01:00
regmap_update_bits ( ad193x - > regmap , AD193X_PLL_CLK_CTRL0 ,
2011-10-18 06:25:08 +08:00
AD193X_PLL_INPUT_MASK , master_rate ) ;
2010-05-21 11:57:01 +08:00
2011-11-28 17:28:12 +01:00
regmap_update_bits ( ad193x - > regmap , AD193X_DAC_CTRL2 ,
2011-10-18 06:25:08 +08:00
AD193X_DAC_WORD_LEN_MASK ,
word_len < < AD193X_DAC_WORD_LEN_SHFT ) ;
2010-03-19 14:48:33 +08:00
2011-11-28 17:28:12 +01:00
regmap_update_bits ( ad193x - > regmap , AD193X_ADC_CTRL1 ,
2011-10-18 06:25:08 +08:00
AD193X_ADC_WORD_LEN_MASK , word_len ) ;
2010-03-19 14:48:33 +08:00
return 0 ;
}
2011-11-23 11:40:40 +01:00
static const struct snd_soc_dai_ops ad193x_dai_ops = {
2010-03-29 11:16:00 +08:00
. hw_params = ad193x_hw_params ,
. digital_mute = ad193x_mute ,
. set_tdm_slot = ad193x_set_tdm_slot ,
2010-05-21 11:57:01 +08:00
. set_sysclk = ad193x_set_dai_sysclk ,
2010-03-29 11:16:00 +08:00
. set_fmt = ad193x_set_dai_fmt ,
} ;
/* codec DAI instance */
2010-03-17 20:15:21 +00:00
static struct snd_soc_dai_driver ad193x_dai = {
. name = " ad193x-hifi " ,
2010-03-29 11:16:00 +08:00
. playback = {
. stream_name = " Playback " ,
. channels_min = 2 ,
. channels_max = 8 ,
. rates = SNDRV_PCM_RATE_48000 ,
. formats = SNDRV_PCM_FMTBIT_S32_LE | SNDRV_PCM_FMTBIT_S16_LE |
SNDRV_PCM_FMTBIT_S20_3LE | SNDRV_PCM_FMTBIT_S24_LE ,
} ,
. capture = {
. stream_name = " Capture " ,
. channels_min = 2 ,
. channels_max = 4 ,
. rates = SNDRV_PCM_RATE_48000 ,
. formats = SNDRV_PCM_FMTBIT_S32_LE | SNDRV_PCM_FMTBIT_S16_LE |
SNDRV_PCM_FMTBIT_S20_3LE | SNDRV_PCM_FMTBIT_S24_LE ,
} ,
. ops = & ad193x_dai_ops ,
} ;
2010-03-17 20:15:21 +00:00
static int ad193x_probe ( struct snd_soc_codec * codec )
2010-03-19 14:48:33 +08:00
{
2010-03-17 20:15:21 +00:00
struct ad193x_priv * ad193x = snd_soc_codec_get_drvdata ( codec ) ;
int ret ;
2010-03-19 14:48:33 +08:00
2011-09-05 20:46:33 +02:00
codec - > control_data = ad193x - > regmap ;
ret = snd_soc_codec_set_cache_io ( codec , 0 , 0 , SND_SOC_REGMAP ) ;
2010-03-17 20:15:21 +00:00
if ( ret < 0 ) {
2011-03-26 03:52:20 -04:00
dev_err ( codec - > dev , " failed to set cache I/O: %d \n " , ret ) ;
2010-03-17 20:15:21 +00:00
return ret ;
2010-03-19 14:48:33 +08:00
}
2010-03-17 20:15:21 +00:00
/* default setting for ad193x */
2010-03-19 14:48:33 +08:00
2010-03-17 20:15:21 +00:00
/* unmute dac channels */
2011-11-28 17:28:12 +01:00
regmap_write ( ad193x - > regmap , AD193X_DAC_CHNL_MUTE , 0x0 ) ;
2010-03-17 20:15:21 +00:00
/* de-emphasis: 48kHz, powedown dac */
2011-11-28 17:28:12 +01:00
regmap_write ( ad193x - > regmap , AD193X_DAC_CTRL2 , 0x1A ) ;
2010-03-17 20:15:21 +00:00
/* powerdown dac, dac in tdm mode */
2011-11-28 17:28:12 +01:00
regmap_write ( ad193x - > regmap , AD193X_DAC_CTRL0 , 0x41 ) ;
2010-03-17 20:15:21 +00:00
/* high-pass filter enable */
2011-11-28 17:28:12 +01:00
regmap_write ( ad193x - > regmap , AD193X_ADC_CTRL0 , 0x3 ) ;
2010-03-17 20:15:21 +00:00
/* sata delay=1, adc aux mode */
2011-11-28 17:28:12 +01:00
regmap_write ( ad193x - > regmap , AD193X_ADC_CTRL1 , 0x43 ) ;
2010-03-17 20:15:21 +00:00
/* pll input: mclki/xi */
2011-11-28 17:28:12 +01:00
regmap_write ( ad193x - > regmap , AD193X_PLL_CLK_CTRL0 , 0x99 ) ; /* mclk=24.576Mhz: 0x9D; mclk=12.288Mhz: 0x99 */
regmap_write ( ad193x - > regmap , AD193X_PLL_CLK_CTRL1 , 0x04 ) ;
2010-03-19 14:48:33 +08:00
return ret ;
}
2010-03-17 20:15:21 +00:00
static struct snd_soc_codec_driver soc_codec_dev_ad193x = {
2010-03-19 14:48:33 +08:00
. probe = ad193x_probe ,
2011-11-28 17:28:06 +01:00
. controls = ad193x_snd_controls ,
. num_controls = ARRAY_SIZE ( ad193x_snd_controls ) ,
. dapm_widgets = ad193x_dapm_widgets ,
. num_dapm_widgets = ARRAY_SIZE ( ad193x_dapm_widgets ) ,
. dapm_routes = audio_paths ,
. num_dapm_routes = ARRAY_SIZE ( audio_paths ) ,
2010-03-19 14:48:33 +08:00
} ;
2011-11-28 17:28:12 +01:00
static bool adau193x_reg_volatile ( struct device * dev , unsigned int reg )
{
return false ;
}
2010-03-19 14:48:33 +08:00
# if defined(CONFIG_SPI_MASTER)
2011-09-05 20:46:33 +02:00
static const struct regmap_config ad193x_spi_regmap_config = {
. val_bits = 8 ,
. reg_bits = 16 ,
. read_flag_mask = 0x09 ,
. write_flag_mask = 0x08 ,
2011-11-28 17:28:12 +01:00
. max_register = AD193X_NUM_REGS - 1 ,
. volatile_reg = adau193x_reg_volatile ,
2011-09-05 20:46:33 +02:00
} ;
2010-03-19 14:48:33 +08:00
static int __devinit ad193x_spi_probe ( struct spi_device * spi )
{
2010-03-17 20:15:21 +00:00
struct ad193x_priv * ad193x ;
int ret ;
2011-12-26 20:51:16 +08:00
ad193x = devm_kzalloc ( & spi - > dev , sizeof ( struct ad193x_priv ) ,
GFP_KERNEL ) ;
2010-03-17 20:15:21 +00:00
if ( ad193x = = NULL )
return - ENOMEM ;
2011-09-05 20:46:33 +02:00
ad193x - > regmap = regmap_init_spi ( spi , & ad193x_spi_regmap_config ) ;
if ( IS_ERR ( ad193x - > regmap ) ) {
ret = PTR_ERR ( ad193x - > regmap ) ;
2011-12-26 20:51:16 +08:00
goto err_out ;
2011-09-05 20:46:33 +02:00
}
2010-03-17 20:15:21 +00:00
spi_set_drvdata ( spi , ad193x ) ;
ret = snd_soc_register_codec ( & spi - > dev ,
& soc_codec_dev_ad193x , & ad193x_dai , 1 ) ;
if ( ret < 0 )
2011-09-05 20:46:33 +02:00
goto err_regmap_exit ;
return 0 ;
err_regmap_exit :
regmap_exit ( ad193x - > regmap ) ;
2011-12-26 20:51:16 +08:00
err_out :
2010-03-17 20:15:21 +00:00
return ret ;
2010-03-19 14:48:33 +08:00
}
static int __devexit ad193x_spi_remove ( struct spi_device * spi )
{
2011-09-05 20:46:33 +02:00
struct ad193x_priv * ad193x = spi_get_drvdata ( spi ) ;
2010-03-17 20:15:21 +00:00
snd_soc_unregister_codec ( & spi - > dev ) ;
2011-09-05 20:46:33 +02:00
regmap_exit ( ad193x - > regmap ) ;
2010-03-17 20:15:21 +00:00
return 0 ;
2010-03-19 14:48:33 +08:00
}
static struct spi_driver ad193x_spi_driver = {
. driver = {
2011-03-27 00:44:12 -04:00
. name = " ad193x " ,
2010-03-19 14:48:33 +08:00
. owner = THIS_MODULE ,
} ,
. probe = ad193x_spi_probe ,
. remove = __devexit_p ( ad193x_spi_remove ) ,
} ;
# endif
# if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
2011-09-05 20:46:33 +02:00
static const struct regmap_config ad193x_i2c_regmap_config = {
. val_bits = 8 ,
. reg_bits = 8 ,
2011-11-28 17:28:12 +01:00
. max_register = AD193X_NUM_REGS - 1 ,
. volatile_reg = adau193x_reg_volatile ,
2011-09-05 20:46:33 +02:00
} ;
2010-03-19 14:48:33 +08:00
static const struct i2c_device_id ad193x_id [ ] = {
{ " ad1936 " , 0 } ,
{ " ad1937 " , 0 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , ad193x_id ) ;
static int __devinit ad193x_i2c_probe ( struct i2c_client * client ,
const struct i2c_device_id * id )
{
2010-03-17 20:15:21 +00:00
struct ad193x_priv * ad193x ;
int ret ;
2011-12-26 20:51:16 +08:00
ad193x = devm_kzalloc ( & client - > dev , sizeof ( struct ad193x_priv ) ,
GFP_KERNEL ) ;
2010-03-17 20:15:21 +00:00
if ( ad193x = = NULL )
return - ENOMEM ;
2011-09-05 20:46:33 +02:00
ad193x - > regmap = regmap_init_i2c ( client , & ad193x_i2c_regmap_config ) ;
if ( IS_ERR ( ad193x - > regmap ) ) {
ret = PTR_ERR ( ad193x - > regmap ) ;
2011-12-26 20:51:16 +08:00
goto err_out ;
2011-09-05 20:46:33 +02:00
}
2010-03-17 20:15:21 +00:00
i2c_set_clientdata ( client , ad193x ) ;
ret = snd_soc_register_codec ( & client - > dev ,
& soc_codec_dev_ad193x , & ad193x_dai , 1 ) ;
if ( ret < 0 )
2011-09-05 20:46:33 +02:00
goto err_regmap_exit ;
return 0 ;
err_regmap_exit :
regmap_exit ( ad193x - > regmap ) ;
2011-12-26 20:51:16 +08:00
err_out :
2010-03-17 20:15:21 +00:00
return ret ;
2010-03-19 14:48:33 +08:00
}
static int __devexit ad193x_i2c_remove ( struct i2c_client * client )
{
2011-09-05 20:46:33 +02:00
struct ad193x_priv * ad193x = i2c_get_clientdata ( client ) ;
2010-03-17 20:15:21 +00:00
snd_soc_unregister_codec ( & client - > dev ) ;
2011-09-05 20:46:33 +02:00
regmap_exit ( ad193x - > regmap ) ;
2010-03-17 20:15:21 +00:00
return 0 ;
2010-03-19 14:48:33 +08:00
}
static struct i2c_driver ad193x_i2c_driver = {
. driver = {
2011-03-27 00:44:12 -04:00
. name = " ad193x " ,
2010-03-19 14:48:33 +08:00
} ,
. probe = ad193x_i2c_probe ,
. remove = __devexit_p ( ad193x_i2c_remove ) ,
. id_table = ad193x_id ,
} ;
# endif
static int __init ad193x_modinit ( void )
{
int ret ;
# if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
ret = i2c_add_driver ( & ad193x_i2c_driver ) ;
if ( ret ! = 0 ) {
printk ( KERN_ERR " Failed to register AD193X I2C driver: %d \n " ,
ret ) ;
}
# endif
# if defined(CONFIG_SPI_MASTER)
ret = spi_register_driver ( & ad193x_spi_driver ) ;
if ( ret ! = 0 ) {
printk ( KERN_ERR " Failed to register AD193X SPI driver: %d \n " ,
ret ) ;
}
# endif
return ret ;
}
module_init ( ad193x_modinit ) ;
static void __exit ad193x_modexit ( void )
{
# if defined(CONFIG_SPI_MASTER)
spi_unregister_driver ( & ad193x_spi_driver ) ;
# endif
# if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE)
i2c_del_driver ( & ad193x_i2c_driver ) ;
# endif
}
module_exit ( ad193x_modexit ) ;
MODULE_DESCRIPTION ( " ASoC ad193x driver " ) ;
MODULE_AUTHOR ( " Barry Song <21cnbao@gmail.com> " ) ;
MODULE_LICENSE ( " GPL " ) ;