2018-02-13 16:29:33 +02:00
// SPDX-License-Identifier: GPL-2.0
2018-02-20 14:53:10 +02:00
//
// Audio driver for AK5558 ADC
//
// Copyright (C) 2015 Asahi Kasei Microdevices Corporation
// Copyright 2018 NXP
2018-02-13 16:29:33 +02:00
# include <linux/delay.h>
# include <linux/gpio/consumer.h>
# include <linux/i2c.h>
# include <linux/module.h>
# include <linux/pm_runtime.h>
# include <linux/regmap.h>
# include <linux/slab.h>
# include <sound/initval.h>
# include <sound/pcm.h>
# include <sound/pcm_params.h>
# include <sound/soc.h>
# include <sound/soc-dapm.h>
# include <sound/tlv.h>
# include "ak5558.h"
/* AK5558 Codec Private Data */
struct ak5558_priv {
2018-02-19 04:17:16 +00:00
struct snd_soc_component component ;
2018-02-13 16:29:33 +02:00
struct regmap * regmap ;
struct i2c_client * i2c ;
struct gpio_desc * reset_gpiod ; /* Reset & Power down GPIO */
int slots ;
int slot_width ;
} ;
/* ak5558 register cache & default register settings */
static const struct reg_default ak5558_reg [ ] = {
{ 0x0 , 0xFF } , /* 0x00 AK5558_00_POWER_MANAGEMENT1 */
{ 0x1 , 0x01 } , /* 0x01 AK5558_01_POWER_MANAGEMENT2 */
{ 0x2 , 0x01 } , /* 0x02 AK5558_02_CONTROL1 */
{ 0x3 , 0x00 } , /* 0x03 AK5558_03_CONTROL2 */
{ 0x4 , 0x00 } , /* 0x04 AK5558_04_CONTROL3 */
{ 0x5 , 0x00 } /* 0x05 AK5558_05_DSD */
} ;
static const char * const mono_texts [ ] = {
" 8 Slot " , " 2 Slot " , " 4 Slot " , " 1 Slot " ,
} ;
static const struct soc_enum ak5558_mono_enum [ ] = {
SOC_ENUM_SINGLE ( AK5558_01_POWER_MANAGEMENT2 , 1 ,
ARRAY_SIZE ( mono_texts ) , mono_texts ) ,
} ;
static const char * const digfil_texts [ ] = {
" Sharp Roll-Off " , " Show Roll-Off " ,
" Short Delay Sharp Roll-Off " , " Short Delay Show Roll-Off " ,
} ;
static const struct soc_enum ak5558_adcset_enum [ ] = {
SOC_ENUM_SINGLE ( AK5558_04_CONTROL3 , 0 ,
ARRAY_SIZE ( digfil_texts ) , digfil_texts ) ,
} ;
static const struct snd_kcontrol_new ak5558_snd_controls [ ] = {
SOC_ENUM ( " AK5558 Monaural Mode " , ak5558_mono_enum [ 0 ] ) ,
SOC_ENUM ( " AK5558 Digital Filter " , ak5558_adcset_enum [ 0 ] ) ,
} ;
static const struct snd_soc_dapm_widget ak5558_dapm_widgets [ ] = {
/* Analog Input */
SND_SOC_DAPM_INPUT ( " AIN1 " ) ,
SND_SOC_DAPM_INPUT ( " AIN2 " ) ,
SND_SOC_DAPM_INPUT ( " AIN3 " ) ,
SND_SOC_DAPM_INPUT ( " AIN4 " ) ,
SND_SOC_DAPM_INPUT ( " AIN5 " ) ,
SND_SOC_DAPM_INPUT ( " AIN6 " ) ,
SND_SOC_DAPM_INPUT ( " AIN7 " ) ,
SND_SOC_DAPM_INPUT ( " AIN8 " ) ,
SND_SOC_DAPM_ADC ( " ADC Ch1 " , NULL , AK5558_00_POWER_MANAGEMENT1 , 0 , 0 ) ,
SND_SOC_DAPM_ADC ( " ADC Ch2 " , NULL , AK5558_00_POWER_MANAGEMENT1 , 1 , 0 ) ,
SND_SOC_DAPM_ADC ( " ADC Ch3 " , NULL , AK5558_00_POWER_MANAGEMENT1 , 2 , 0 ) ,
SND_SOC_DAPM_ADC ( " ADC Ch4 " , NULL , AK5558_00_POWER_MANAGEMENT1 , 3 , 0 ) ,
SND_SOC_DAPM_ADC ( " ADC Ch5 " , NULL , AK5558_00_POWER_MANAGEMENT1 , 4 , 0 ) ,
SND_SOC_DAPM_ADC ( " ADC Ch6 " , NULL , AK5558_00_POWER_MANAGEMENT1 , 5 , 0 ) ,
SND_SOC_DAPM_ADC ( " ADC Ch7 " , NULL , AK5558_00_POWER_MANAGEMENT1 , 6 , 0 ) ,
SND_SOC_DAPM_ADC ( " ADC Ch8 " , NULL , AK5558_00_POWER_MANAGEMENT1 , 7 , 0 ) ,
SND_SOC_DAPM_AIF_OUT ( " SDTO " , " Capture " , 0 , SND_SOC_NOPM , 0 , 0 ) ,
} ;
static const struct snd_soc_dapm_route ak5558_intercon [ ] = {
{ " ADC Ch1 " , NULL , " AIN1 " } ,
{ " SDTO " , NULL , " ADC Ch1 " } ,
{ " ADC Ch2 " , NULL , " AIN2 " } ,
{ " SDTO " , NULL , " ADC Ch2 " } ,
{ " ADC Ch3 " , NULL , " AIN3 " } ,
{ " SDTO " , NULL , " ADC Ch3 " } ,
{ " ADC Ch4 " , NULL , " AIN4 " } ,
{ " SDTO " , NULL , " ADC Ch4 " } ,
{ " ADC Ch5 " , NULL , " AIN5 " } ,
{ " SDTO " , NULL , " ADC Ch5 " } ,
{ " ADC Ch6 " , NULL , " AIN6 " } ,
{ " SDTO " , NULL , " ADC Ch6 " } ,
{ " ADC Ch7 " , NULL , " AIN7 " } ,
{ " SDTO " , NULL , " ADC Ch7 " } ,
{ " ADC Ch8 " , NULL , " AIN8 " } ,
{ " SDTO " , NULL , " ADC Ch8 " } ,
} ;
2018-02-19 04:17:16 +00:00
static int ak5558_set_mcki ( struct snd_soc_component * component )
2018-02-13 16:29:33 +02:00
{
2018-02-19 04:17:16 +00:00
return snd_soc_component_update_bits ( component , AK5558_02_CONTROL1 , AK5558_CKS ,
2018-02-13 16:29:33 +02:00
AK5558_CKS_AUTO ) ;
}
static int ak5558_hw_params ( struct snd_pcm_substream * substream ,
struct snd_pcm_hw_params * params ,
struct snd_soc_dai * dai )
{
2018-02-19 04:17:16 +00:00
struct snd_soc_component * component = dai - > component ;
struct ak5558_priv * ak5558 = snd_soc_component_get_drvdata ( component ) ;
2018-02-13 16:29:33 +02:00
u8 bits ;
int pcm_width = max ( params_physical_width ( params ) , ak5558 - > slot_width ) ;
/* set master/slave audio interface */
2018-02-19 04:17:16 +00:00
bits = snd_soc_component_read32 ( component , AK5558_02_CONTROL1 ) ;
2018-02-13 16:29:33 +02:00
bits & = ~ AK5558_BITS ;
switch ( pcm_width ) {
case 16 :
bits | = AK5558_DIF_24BIT_MODE ;
break ;
case 32 :
bits | = AK5558_DIF_32BIT_MODE ;
break ;
default :
return - EINVAL ;
}
2018-02-19 04:17:16 +00:00
snd_soc_component_update_bits ( component , AK5558_02_CONTROL1 , AK5558_BITS , bits ) ;
2018-02-13 16:29:33 +02:00
return 0 ;
}
static int ak5558_set_dai_fmt ( struct snd_soc_dai * dai , unsigned int fmt )
{
2018-02-19 04:17:16 +00:00
struct snd_soc_component * component = dai - > component ;
2018-02-13 16:29:33 +02:00
u8 format ;
switch ( fmt & SND_SOC_DAIFMT_MASTER_MASK ) {
case SND_SOC_DAIFMT_CBS_CFS :
break ;
case SND_SOC_DAIFMT_CBM_CFM :
break ;
case SND_SOC_DAIFMT_CBS_CFM :
case SND_SOC_DAIFMT_CBM_CFS :
default :
dev_err ( dai - > dev , " Clock mode unsupported " ) ;
return - EINVAL ;
}
/* set master/slave audio interface */
2018-02-19 04:17:16 +00:00
format = snd_soc_component_read32 ( component , AK5558_02_CONTROL1 ) ;
2018-02-13 16:29:33 +02:00
format & = ~ AK5558_DIF ;
switch ( fmt & SND_SOC_DAIFMT_FORMAT_MASK ) {
case SND_SOC_DAIFMT_I2S :
format | = AK5558_DIF_I2S_MODE ;
break ;
case SND_SOC_DAIFMT_LEFT_J :
format | = AK5558_DIF_MSB_MODE ;
break ;
case SND_SOC_DAIFMT_DSP_B :
format | = AK5558_DIF_MSB_MODE ;
break ;
default :
return - EINVAL ;
}
2018-02-19 04:17:16 +00:00
snd_soc_component_update_bits ( component , AK5558_02_CONTROL1 , AK5558_DIF , format ) ;
2018-02-13 16:29:33 +02:00
return 0 ;
}
static int ak5558_set_tdm_slot ( struct snd_soc_dai * dai , unsigned int tx_mask ,
unsigned int rx_mask , int slots ,
int slot_width )
{
2018-02-19 04:17:16 +00:00
struct snd_soc_component * component = dai - > component ;
struct ak5558_priv * ak5558 = snd_soc_component_get_drvdata ( component ) ;
2018-02-13 16:29:33 +02:00
int tdm_mode ;
ak5558 - > slots = slots ;
ak5558 - > slot_width = slot_width ;
switch ( slots * slot_width ) {
case 128 :
tdm_mode = AK5558_MODE_TDM128 ;
break ;
case 256 :
tdm_mode = AK5558_MODE_TDM256 ;
break ;
case 512 :
tdm_mode = AK5558_MODE_TDM512 ;
break ;
default :
tdm_mode = AK5558_MODE_NORMAL ;
break ;
}
2018-02-19 04:17:16 +00:00
snd_soc_component_update_bits ( component , AK5558_03_CONTROL2 , AK5558_MODE_BITS ,
2018-02-13 16:29:33 +02:00
tdm_mode ) ;
return 0 ;
}
# define AK5558_FORMATS (SNDRV_PCM_FMTBIT_S16_LE |\
SNDRV_PCM_FMTBIT_S24_LE | \
SNDRV_PCM_FMTBIT_S32_LE )
static const unsigned int ak5558_rates [ ] = {
8000 , 11025 , 16000 , 22050 ,
32000 , 44100 , 48000 , 88200 ,
96000 , 176400 , 192000 , 352800 ,
384000 , 705600 , 768000 , 1411200 ,
2822400 ,
} ;
static const struct snd_pcm_hw_constraint_list ak5558_rate_constraints = {
. count = ARRAY_SIZE ( ak5558_rates ) ,
. list = ak5558_rates ,
} ;
static int ak5558_startup ( struct snd_pcm_substream * substream ,
struct snd_soc_dai * dai )
{
return snd_pcm_hw_constraint_list ( substream - > runtime , 0 ,
SNDRV_PCM_HW_PARAM_RATE ,
& ak5558_rate_constraints ) ;
}
static struct snd_soc_dai_ops ak5558_dai_ops = {
. startup = ak5558_startup ,
. hw_params = ak5558_hw_params ,
. set_fmt = ak5558_set_dai_fmt ,
. set_tdm_slot = ak5558_set_tdm_slot ,
} ;
static struct snd_soc_dai_driver ak5558_dai = {
. name = " ak5558-aif " ,
. capture = {
. stream_name = " Capture " ,
. channels_min = 1 ,
. channels_max = 8 ,
. rates = SNDRV_PCM_RATE_KNOT ,
. formats = AK5558_FORMATS ,
} ,
. ops = & ak5558_dai_ops ,
} ;
static void ak5558_power_off ( struct ak5558_priv * ak5558 )
{
if ( ! ak5558 - > reset_gpiod )
return ;
gpiod_set_value_cansleep ( ak5558 - > reset_gpiod , 0 ) ;
usleep_range ( 1000 , 2000 ) ;
}
static void ak5558_power_on ( struct ak5558_priv * ak5558 )
{
if ( ! ak5558 - > reset_gpiod )
return ;
gpiod_set_value_cansleep ( ak5558 - > reset_gpiod , 1 ) ;
usleep_range ( 1000 , 2000 ) ;
}
2018-02-19 04:17:16 +00:00
static int ak5558_probe ( struct snd_soc_component * component )
2018-02-13 16:29:33 +02:00
{
2018-02-19 04:17:16 +00:00
struct ak5558_priv * ak5558 = snd_soc_component_get_drvdata ( component ) ;
2018-02-13 16:29:33 +02:00
ak5558_power_on ( ak5558 ) ;
2018-02-19 04:17:16 +00:00
return ak5558_set_mcki ( component ) ;
2018-02-13 16:29:33 +02:00
}
2018-02-19 04:17:16 +00:00
static void ak5558_remove ( struct snd_soc_component * component )
2018-02-13 16:29:33 +02:00
{
2018-02-19 04:17:16 +00:00
struct ak5558_priv * ak5558 = snd_soc_component_get_drvdata ( component ) ;
2018-02-13 16:29:33 +02:00
ak5558_power_off ( ak5558 ) ;
}
static int __maybe_unused ak5558_runtime_suspend ( struct device * dev )
{
struct ak5558_priv * ak5558 = dev_get_drvdata ( dev ) ;
regcache_cache_only ( ak5558 - > regmap , true ) ;
ak5558_power_off ( ak5558 ) ;
return 0 ;
}
static int __maybe_unused ak5558_runtime_resume ( struct device * dev )
{
struct ak5558_priv * ak5558 = dev_get_drvdata ( dev ) ;
ak5558_power_off ( ak5558 ) ;
ak5558_power_on ( ak5558 ) ;
regcache_cache_only ( ak5558 - > regmap , false ) ;
regcache_mark_dirty ( ak5558 - > regmap ) ;
return regcache_sync ( ak5558 - > regmap ) ;
}
2018-06-06 10:57:19 +01:00
static const struct dev_pm_ops ak5558_pm = {
2018-02-13 16:29:33 +02:00
SET_RUNTIME_PM_OPS ( ak5558_runtime_suspend , ak5558_runtime_resume , NULL )
SET_SYSTEM_SLEEP_PM_OPS ( pm_runtime_force_suspend ,
pm_runtime_force_resume )
} ;
2018-06-06 10:57:19 +01:00
static const struct snd_soc_component_driver soc_codec_dev_ak5558 = {
2018-02-19 04:17:16 +00:00
. probe = ak5558_probe ,
. remove = ak5558_remove ,
. controls = ak5558_snd_controls ,
. num_controls = ARRAY_SIZE ( ak5558_snd_controls ) ,
. dapm_widgets = ak5558_dapm_widgets ,
. num_dapm_widgets = ARRAY_SIZE ( ak5558_dapm_widgets ) ,
. dapm_routes = ak5558_intercon ,
. num_dapm_routes = ARRAY_SIZE ( ak5558_intercon ) ,
. idle_bias_on = 1 ,
. use_pmdown_time = 1 ,
. endianness = 1 ,
. non_legacy_dai_naming = 1 ,
2018-02-13 16:29:33 +02:00
} ;
static const struct regmap_config ak5558_regmap = {
. reg_bits = 8 ,
. val_bits = 8 ,
. max_register = AK5558_05_DSD ,
. reg_defaults = ak5558_reg ,
. num_reg_defaults = ARRAY_SIZE ( ak5558_reg ) ,
. cache_type = REGCACHE_RBTREE ,
} ;
static int ak5558_i2c_probe ( struct i2c_client * i2c )
{
struct ak5558_priv * ak5558 ;
int ret = 0 ;
ak5558 = devm_kzalloc ( & i2c - > dev , sizeof ( * ak5558 ) , GFP_KERNEL ) ;
if ( ! ak5558 )
return - ENOMEM ;
ak5558 - > regmap = devm_regmap_init_i2c ( i2c , & ak5558_regmap ) ;
if ( IS_ERR ( ak5558 - > regmap ) )
return PTR_ERR ( ak5558 - > regmap ) ;
i2c_set_clientdata ( i2c , ak5558 ) ;
ak5558 - > i2c = i2c ;
ak5558 - > reset_gpiod = devm_gpiod_get_optional ( & i2c - > dev , " reset " ,
GPIOD_OUT_LOW ) ;
if ( IS_ERR ( ak5558 - > reset_gpiod ) )
return PTR_ERR ( ak5558 - > reset_gpiod ) ;
2018-02-19 04:17:16 +00:00
ret = devm_snd_soc_register_component ( & i2c - > dev ,
& soc_codec_dev_ak5558 ,
2018-02-13 16:29:33 +02:00
& ak5558_dai , 1 ) ;
if ( ret )
return ret ;
pm_runtime_enable ( & i2c - > dev ) ;
return 0 ;
}
static int ak5558_i2c_remove ( struct i2c_client * i2c )
{
pm_runtime_disable ( & i2c - > dev ) ;
return 0 ;
}
static const struct of_device_id ak5558_i2c_dt_ids [ ] = {
{ . compatible = " asahi-kasei,ak5558 " } ,
{ }
} ;
static struct i2c_driver ak5558_i2c_driver = {
. driver = {
. name = " ak5558 " ,
. of_match_table = of_match_ptr ( ak5558_i2c_dt_ids ) ,
. pm = & ak5558_pm ,
} ,
. probe_new = ak5558_i2c_probe ,
. remove = ak5558_i2c_remove ,
} ;
module_i2c_driver ( ak5558_i2c_driver ) ;
MODULE_AUTHOR ( " Junichi Wakasugi <wakasugi.jb@om.asahi-kasei.co.jp> " ) ;
MODULE_AUTHOR ( " Mihai Serban <mihai.serban@nxp.com> " ) ;
MODULE_DESCRIPTION ( " ASoC AK5558 ADC driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;