192088477SDaniel Baluta // SPDX-License-Identifier: GPL-2.0
2af3acca3SDaniel Baluta //
3af3acca3SDaniel Baluta // Audio driver for AK5558 ADC
4af3acca3SDaniel Baluta //
5af3acca3SDaniel Baluta // Copyright (C) 2015 Asahi Kasei Microdevices Corporation
6af3acca3SDaniel Baluta // Copyright 2018 NXP
792088477SDaniel Baluta
892088477SDaniel Baluta #include <linux/delay.h>
992088477SDaniel Baluta #include <linux/gpio/consumer.h>
1092088477SDaniel Baluta #include <linux/i2c.h>
1192088477SDaniel Baluta #include <linux/module.h>
12d8c5c82eSViorel Suman #include <linux/of_device.h>
1392088477SDaniel Baluta #include <linux/pm_runtime.h>
1492088477SDaniel Baluta #include <linux/regmap.h>
152ff6d5a1SShengjiu Wang #include <linux/regulator/consumer.h>
1692088477SDaniel Baluta #include <linux/slab.h>
1792088477SDaniel Baluta
1892088477SDaniel Baluta #include <sound/initval.h>
1992088477SDaniel Baluta #include <sound/pcm.h>
2092088477SDaniel Baluta #include <sound/pcm_params.h>
2192088477SDaniel Baluta #include <sound/soc.h>
2292088477SDaniel Baluta #include <sound/soc-dapm.h>
2392088477SDaniel Baluta #include <sound/tlv.h>
2492088477SDaniel Baluta
2592088477SDaniel Baluta #include "ak5558.h"
2692088477SDaniel Baluta
27d8c5c82eSViorel Suman enum ak555x_type {
28d8c5c82eSViorel Suman AK5558,
29d8c5c82eSViorel Suman AK5552,
30d8c5c82eSViorel Suman };
31d8c5c82eSViorel Suman
322ff6d5a1SShengjiu Wang #define AK5558_NUM_SUPPLIES 2
332ff6d5a1SShengjiu Wang static const char *ak5558_supply_names[AK5558_NUM_SUPPLIES] = {
342ff6d5a1SShengjiu Wang "DVDD",
352ff6d5a1SShengjiu Wang "AVDD",
362ff6d5a1SShengjiu Wang };
372ff6d5a1SShengjiu Wang
3892088477SDaniel Baluta /* AK5558 Codec Private Data */
3992088477SDaniel Baluta struct ak5558_priv {
402ff6d5a1SShengjiu Wang struct regulator_bulk_data supplies[AK5558_NUM_SUPPLIES];
413f15aa19SKuninori Morimoto struct snd_soc_component component;
4292088477SDaniel Baluta struct regmap *regmap;
4392088477SDaniel Baluta struct i2c_client *i2c;
4492088477SDaniel Baluta struct gpio_desc *reset_gpiod; /* Reset & Power down GPIO */
4592088477SDaniel Baluta int slots;
4692088477SDaniel Baluta int slot_width;
4792088477SDaniel Baluta };
4892088477SDaniel Baluta
4992088477SDaniel Baluta /* ak5558 register cache & default register settings */
5092088477SDaniel Baluta static const struct reg_default ak5558_reg[] = {
5192088477SDaniel Baluta { 0x0, 0xFF }, /* 0x00 AK5558_00_POWER_MANAGEMENT1 */
5292088477SDaniel Baluta { 0x1, 0x01 }, /* 0x01 AK5558_01_POWER_MANAGEMENT2 */
5392088477SDaniel Baluta { 0x2, 0x01 }, /* 0x02 AK5558_02_CONTROL1 */
5492088477SDaniel Baluta { 0x3, 0x00 }, /* 0x03 AK5558_03_CONTROL2 */
5592088477SDaniel Baluta { 0x4, 0x00 }, /* 0x04 AK5558_04_CONTROL3 */
5692088477SDaniel Baluta { 0x5, 0x00 } /* 0x05 AK5558_05_DSD */
5792088477SDaniel Baluta };
5892088477SDaniel Baluta
5992088477SDaniel Baluta static const char * const mono_texts[] = {
6092088477SDaniel Baluta "8 Slot", "2 Slot", "4 Slot", "1 Slot",
6192088477SDaniel Baluta };
6292088477SDaniel Baluta
6392088477SDaniel Baluta static const struct soc_enum ak5558_mono_enum[] = {
6492088477SDaniel Baluta SOC_ENUM_SINGLE(AK5558_01_POWER_MANAGEMENT2, 1,
6592088477SDaniel Baluta ARRAY_SIZE(mono_texts), mono_texts),
6692088477SDaniel Baluta };
6792088477SDaniel Baluta
68d8c5c82eSViorel Suman static const char * const mono_5552_texts[] = {
69d8c5c82eSViorel Suman "2 Slot", "1 Slot (Fixed)", "2 Slot", "1 Slot (Optimal)",
70d8c5c82eSViorel Suman };
71d8c5c82eSViorel Suman
72d8c5c82eSViorel Suman static const struct soc_enum ak5552_mono_enum[] = {
73d8c5c82eSViorel Suman SOC_ENUM_SINGLE(AK5558_01_POWER_MANAGEMENT2, 1,
74d8c5c82eSViorel Suman ARRAY_SIZE(mono_5552_texts), mono_5552_texts),
75d8c5c82eSViorel Suman };
76d8c5c82eSViorel Suman
7792088477SDaniel Baluta static const char * const digfil_texts[] = {
78a4350899SShengjiu Wang "Sharp Roll-Off", "Slow Roll-Off",
79a4350899SShengjiu Wang "Short Delay Sharp Roll-Off", "Short Delay Slow Roll-Off",
8092088477SDaniel Baluta };
8192088477SDaniel Baluta
8292088477SDaniel Baluta static const struct soc_enum ak5558_adcset_enum[] = {
8392088477SDaniel Baluta SOC_ENUM_SINGLE(AK5558_04_CONTROL3, 0,
8492088477SDaniel Baluta ARRAY_SIZE(digfil_texts), digfil_texts),
8592088477SDaniel Baluta };
8692088477SDaniel Baluta
8792088477SDaniel Baluta static const struct snd_kcontrol_new ak5558_snd_controls[] = {
88d8c5c82eSViorel Suman SOC_ENUM("Monaural Mode", ak5558_mono_enum[0]),
89d8c5c82eSViorel Suman SOC_ENUM("Digital Filter", ak5558_adcset_enum[0]),
90d8c5c82eSViorel Suman };
91d8c5c82eSViorel Suman
92d8c5c82eSViorel Suman static const struct snd_kcontrol_new ak5552_snd_controls[] = {
93d8c5c82eSViorel Suman SOC_ENUM("Monaural Mode", ak5552_mono_enum[0]),
94d8c5c82eSViorel Suman SOC_ENUM("Digital Filter", ak5558_adcset_enum[0]),
9592088477SDaniel Baluta };
9692088477SDaniel Baluta
9792088477SDaniel Baluta static const struct snd_soc_dapm_widget ak5558_dapm_widgets[] = {
9892088477SDaniel Baluta /* Analog Input */
9992088477SDaniel Baluta SND_SOC_DAPM_INPUT("AIN1"),
10092088477SDaniel Baluta SND_SOC_DAPM_INPUT("AIN2"),
10192088477SDaniel Baluta SND_SOC_DAPM_INPUT("AIN3"),
10292088477SDaniel Baluta SND_SOC_DAPM_INPUT("AIN4"),
10392088477SDaniel Baluta SND_SOC_DAPM_INPUT("AIN5"),
10492088477SDaniel Baluta SND_SOC_DAPM_INPUT("AIN6"),
10592088477SDaniel Baluta SND_SOC_DAPM_INPUT("AIN7"),
10692088477SDaniel Baluta SND_SOC_DAPM_INPUT("AIN8"),
10792088477SDaniel Baluta
10892088477SDaniel Baluta SND_SOC_DAPM_ADC("ADC Ch1", NULL, AK5558_00_POWER_MANAGEMENT1, 0, 0),
10992088477SDaniel Baluta SND_SOC_DAPM_ADC("ADC Ch2", NULL, AK5558_00_POWER_MANAGEMENT1, 1, 0),
11092088477SDaniel Baluta SND_SOC_DAPM_ADC("ADC Ch3", NULL, AK5558_00_POWER_MANAGEMENT1, 2, 0),
11192088477SDaniel Baluta SND_SOC_DAPM_ADC("ADC Ch4", NULL, AK5558_00_POWER_MANAGEMENT1, 3, 0),
11292088477SDaniel Baluta SND_SOC_DAPM_ADC("ADC Ch5", NULL, AK5558_00_POWER_MANAGEMENT1, 4, 0),
11392088477SDaniel Baluta SND_SOC_DAPM_ADC("ADC Ch6", NULL, AK5558_00_POWER_MANAGEMENT1, 5, 0),
11492088477SDaniel Baluta SND_SOC_DAPM_ADC("ADC Ch7", NULL, AK5558_00_POWER_MANAGEMENT1, 6, 0),
11592088477SDaniel Baluta SND_SOC_DAPM_ADC("ADC Ch8", NULL, AK5558_00_POWER_MANAGEMENT1, 7, 0),
11692088477SDaniel Baluta
11792088477SDaniel Baluta SND_SOC_DAPM_AIF_OUT("SDTO", "Capture", 0, SND_SOC_NOPM, 0, 0),
11892088477SDaniel Baluta };
11992088477SDaniel Baluta
120d8c5c82eSViorel Suman static const struct snd_soc_dapm_widget ak5552_dapm_widgets[] = {
121d8c5c82eSViorel Suman /* Analog Input */
122d8c5c82eSViorel Suman SND_SOC_DAPM_INPUT("AIN1"),
123d8c5c82eSViorel Suman SND_SOC_DAPM_INPUT("AIN2"),
124d8c5c82eSViorel Suman
125d8c5c82eSViorel Suman SND_SOC_DAPM_ADC("ADC Ch1", NULL, AK5558_00_POWER_MANAGEMENT1, 0, 0),
126d8c5c82eSViorel Suman SND_SOC_DAPM_ADC("ADC Ch2", NULL, AK5558_00_POWER_MANAGEMENT1, 1, 0),
127d8c5c82eSViorel Suman
128d8c5c82eSViorel Suman SND_SOC_DAPM_AIF_OUT("SDTO", "Capture", 0, SND_SOC_NOPM, 0, 0),
129d8c5c82eSViorel Suman };
130d8c5c82eSViorel Suman
13192088477SDaniel Baluta static const struct snd_soc_dapm_route ak5558_intercon[] = {
13292088477SDaniel Baluta {"ADC Ch1", NULL, "AIN1"},
13392088477SDaniel Baluta {"SDTO", NULL, "ADC Ch1"},
13492088477SDaniel Baluta
13592088477SDaniel Baluta {"ADC Ch2", NULL, "AIN2"},
13692088477SDaniel Baluta {"SDTO", NULL, "ADC Ch2"},
13792088477SDaniel Baluta
13892088477SDaniel Baluta {"ADC Ch3", NULL, "AIN3"},
13992088477SDaniel Baluta {"SDTO", NULL, "ADC Ch3"},
14092088477SDaniel Baluta
14192088477SDaniel Baluta {"ADC Ch4", NULL, "AIN4"},
14292088477SDaniel Baluta {"SDTO", NULL, "ADC Ch4"},
14392088477SDaniel Baluta
14492088477SDaniel Baluta {"ADC Ch5", NULL, "AIN5"},
14592088477SDaniel Baluta {"SDTO", NULL, "ADC Ch5"},
14692088477SDaniel Baluta
14792088477SDaniel Baluta {"ADC Ch6", NULL, "AIN6"},
14892088477SDaniel Baluta {"SDTO", NULL, "ADC Ch6"},
14992088477SDaniel Baluta
15092088477SDaniel Baluta {"ADC Ch7", NULL, "AIN7"},
15192088477SDaniel Baluta {"SDTO", NULL, "ADC Ch7"},
15292088477SDaniel Baluta
15392088477SDaniel Baluta {"ADC Ch8", NULL, "AIN8"},
15492088477SDaniel Baluta {"SDTO", NULL, "ADC Ch8"},
15592088477SDaniel Baluta };
15692088477SDaniel Baluta
157d8c5c82eSViorel Suman static const struct snd_soc_dapm_route ak5552_intercon[] = {
158d8c5c82eSViorel Suman {"ADC Ch1", NULL, "AIN1"},
159d8c5c82eSViorel Suman {"SDTO", NULL, "ADC Ch1"},
160d8c5c82eSViorel Suman
161d8c5c82eSViorel Suman {"ADC Ch2", NULL, "AIN2"},
162d8c5c82eSViorel Suman {"SDTO", NULL, "ADC Ch2"},
163d8c5c82eSViorel Suman };
164d8c5c82eSViorel Suman
ak5558_set_mcki(struct snd_soc_component * component)1653f15aa19SKuninori Morimoto static int ak5558_set_mcki(struct snd_soc_component *component)
16692088477SDaniel Baluta {
1673f15aa19SKuninori Morimoto return snd_soc_component_update_bits(component, AK5558_02_CONTROL1, AK5558_CKS,
16892088477SDaniel Baluta AK5558_CKS_AUTO);
16992088477SDaniel Baluta }
17092088477SDaniel Baluta
ak5558_hw_params(struct snd_pcm_substream * substream,struct snd_pcm_hw_params * params,struct snd_soc_dai * dai)17192088477SDaniel Baluta static int ak5558_hw_params(struct snd_pcm_substream *substream,
17292088477SDaniel Baluta struct snd_pcm_hw_params *params,
17392088477SDaniel Baluta struct snd_soc_dai *dai)
17492088477SDaniel Baluta {
1753f15aa19SKuninori Morimoto struct snd_soc_component *component = dai->component;
1763f15aa19SKuninori Morimoto struct ak5558_priv *ak5558 = snd_soc_component_get_drvdata(component);
17792088477SDaniel Baluta u8 bits;
17892088477SDaniel Baluta int pcm_width = max(params_physical_width(params), ak5558->slot_width);
17992088477SDaniel Baluta
18092088477SDaniel Baluta switch (pcm_width) {
18192088477SDaniel Baluta case 16:
18239dfdf00SAxel Lin bits = AK5558_DIF_24BIT_MODE;
18392088477SDaniel Baluta break;
18492088477SDaniel Baluta case 32:
18539dfdf00SAxel Lin bits = AK5558_DIF_32BIT_MODE;
18692088477SDaniel Baluta break;
18792088477SDaniel Baluta default:
18892088477SDaniel Baluta return -EINVAL;
18992088477SDaniel Baluta }
19092088477SDaniel Baluta
1913f15aa19SKuninori Morimoto snd_soc_component_update_bits(component, AK5558_02_CONTROL1, AK5558_BITS, bits);
19292088477SDaniel Baluta
19392088477SDaniel Baluta return 0;
19492088477SDaniel Baluta }
19592088477SDaniel Baluta
ak5558_set_dai_fmt(struct snd_soc_dai * dai,unsigned int fmt)19692088477SDaniel Baluta static int ak5558_set_dai_fmt(struct snd_soc_dai *dai, unsigned int fmt)
19792088477SDaniel Baluta {
1983f15aa19SKuninori Morimoto struct snd_soc_component *component = dai->component;
19992088477SDaniel Baluta u8 format;
20092088477SDaniel Baluta
201b55f0343SMark Brown switch (fmt & SND_SOC_DAIFMT_CLOCK_PROVIDER_MASK) {
202b55f0343SMark Brown case SND_SOC_DAIFMT_CBC_CFC:
20392088477SDaniel Baluta break;
204b55f0343SMark Brown case SND_SOC_DAIFMT_CBP_CFP:
20592088477SDaniel Baluta break;
206b55f0343SMark Brown case SND_SOC_DAIFMT_CBC_CFP:
207b55f0343SMark Brown case SND_SOC_DAIFMT_CBP_CFC:
20892088477SDaniel Baluta default:
20992088477SDaniel Baluta dev_err(dai->dev, "Clock mode unsupported");
21092088477SDaniel Baluta return -EINVAL;
21192088477SDaniel Baluta }
21292088477SDaniel Baluta
21392088477SDaniel Baluta /* set master/slave audio interface */
21492088477SDaniel Baluta switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
21592088477SDaniel Baluta case SND_SOC_DAIFMT_I2S:
21639dfdf00SAxel Lin format = AK5558_DIF_I2S_MODE;
21792088477SDaniel Baluta break;
21892088477SDaniel Baluta case SND_SOC_DAIFMT_LEFT_J:
21939dfdf00SAxel Lin format = AK5558_DIF_MSB_MODE;
22092088477SDaniel Baluta break;
22192088477SDaniel Baluta case SND_SOC_DAIFMT_DSP_B:
22239dfdf00SAxel Lin format = AK5558_DIF_MSB_MODE;
22392088477SDaniel Baluta break;
22492088477SDaniel Baluta default:
22592088477SDaniel Baluta return -EINVAL;
22692088477SDaniel Baluta }
22792088477SDaniel Baluta
2283f15aa19SKuninori Morimoto snd_soc_component_update_bits(component, AK5558_02_CONTROL1, AK5558_DIF, format);
22992088477SDaniel Baluta
23092088477SDaniel Baluta return 0;
23192088477SDaniel Baluta }
23292088477SDaniel Baluta
ak5558_set_tdm_slot(struct snd_soc_dai * dai,unsigned int tx_mask,unsigned int rx_mask,int slots,int slot_width)23392088477SDaniel Baluta static int ak5558_set_tdm_slot(struct snd_soc_dai *dai, unsigned int tx_mask,
23492088477SDaniel Baluta unsigned int rx_mask, int slots,
23592088477SDaniel Baluta int slot_width)
23692088477SDaniel Baluta {
2373f15aa19SKuninori Morimoto struct snd_soc_component *component = dai->component;
2383f15aa19SKuninori Morimoto struct ak5558_priv *ak5558 = snd_soc_component_get_drvdata(component);
23992088477SDaniel Baluta int tdm_mode;
24092088477SDaniel Baluta
24192088477SDaniel Baluta ak5558->slots = slots;
24292088477SDaniel Baluta ak5558->slot_width = slot_width;
24392088477SDaniel Baluta
24492088477SDaniel Baluta switch (slots * slot_width) {
24592088477SDaniel Baluta case 128:
24692088477SDaniel Baluta tdm_mode = AK5558_MODE_TDM128;
24792088477SDaniel Baluta break;
24892088477SDaniel Baluta case 256:
24992088477SDaniel Baluta tdm_mode = AK5558_MODE_TDM256;
25092088477SDaniel Baluta break;
25192088477SDaniel Baluta case 512:
25292088477SDaniel Baluta tdm_mode = AK5558_MODE_TDM512;
25392088477SDaniel Baluta break;
25492088477SDaniel Baluta default:
25592088477SDaniel Baluta tdm_mode = AK5558_MODE_NORMAL;
25692088477SDaniel Baluta break;
25792088477SDaniel Baluta }
25892088477SDaniel Baluta
2593f15aa19SKuninori Morimoto snd_soc_component_update_bits(component, AK5558_03_CONTROL2, AK5558_MODE_BITS,
26092088477SDaniel Baluta tdm_mode);
26192088477SDaniel Baluta return 0;
26292088477SDaniel Baluta }
26392088477SDaniel Baluta
26492088477SDaniel Baluta #define AK5558_FORMATS (SNDRV_PCM_FMTBIT_S16_LE |\
26592088477SDaniel Baluta SNDRV_PCM_FMTBIT_S24_LE |\
26692088477SDaniel Baluta SNDRV_PCM_FMTBIT_S32_LE)
26792088477SDaniel Baluta
26892088477SDaniel Baluta static const unsigned int ak5558_rates[] = {
26992088477SDaniel Baluta 8000, 11025, 16000, 22050,
27092088477SDaniel Baluta 32000, 44100, 48000, 88200,
27192088477SDaniel Baluta 96000, 176400, 192000, 352800,
27292088477SDaniel Baluta 384000, 705600, 768000, 1411200,
27392088477SDaniel Baluta 2822400,
27492088477SDaniel Baluta };
27592088477SDaniel Baluta
27692088477SDaniel Baluta static const struct snd_pcm_hw_constraint_list ak5558_rate_constraints = {
27792088477SDaniel Baluta .count = ARRAY_SIZE(ak5558_rates),
27892088477SDaniel Baluta .list = ak5558_rates,
27992088477SDaniel Baluta };
28092088477SDaniel Baluta
ak5558_startup(struct snd_pcm_substream * substream,struct snd_soc_dai * dai)28192088477SDaniel Baluta static int ak5558_startup(struct snd_pcm_substream *substream,
28292088477SDaniel Baluta struct snd_soc_dai *dai)
28392088477SDaniel Baluta {
28492088477SDaniel Baluta return snd_pcm_hw_constraint_list(substream->runtime, 0,
28592088477SDaniel Baluta SNDRV_PCM_HW_PARAM_RATE,
28692088477SDaniel Baluta &ak5558_rate_constraints);
28792088477SDaniel Baluta }
28892088477SDaniel Baluta
289704a9fc2SJulia Lawall static const struct snd_soc_dai_ops ak5558_dai_ops = {
29092088477SDaniel Baluta .startup = ak5558_startup,
29192088477SDaniel Baluta .hw_params = ak5558_hw_params,
29292088477SDaniel Baluta
29392088477SDaniel Baluta .set_fmt = ak5558_set_dai_fmt,
29492088477SDaniel Baluta .set_tdm_slot = ak5558_set_tdm_slot,
29592088477SDaniel Baluta };
29692088477SDaniel Baluta
29792088477SDaniel Baluta static struct snd_soc_dai_driver ak5558_dai = {
29892088477SDaniel Baluta .name = "ak5558-aif",
29992088477SDaniel Baluta .capture = {
30092088477SDaniel Baluta .stream_name = "Capture",
30192088477SDaniel Baluta .channels_min = 1,
30292088477SDaniel Baluta .channels_max = 8,
30392088477SDaniel Baluta .rates = SNDRV_PCM_RATE_KNOT,
30492088477SDaniel Baluta .formats = AK5558_FORMATS,
30592088477SDaniel Baluta },
30692088477SDaniel Baluta .ops = &ak5558_dai_ops,
30792088477SDaniel Baluta };
30892088477SDaniel Baluta
309d8c5c82eSViorel Suman static struct snd_soc_dai_driver ak5552_dai = {
310b23584d6SShengjiu Wang .name = "ak5552-aif",
311d8c5c82eSViorel Suman .capture = {
312d8c5c82eSViorel Suman .stream_name = "Capture",
313d8c5c82eSViorel Suman .channels_min = 1,
314d8c5c82eSViorel Suman .channels_max = 2,
315d8c5c82eSViorel Suman .rates = SNDRV_PCM_RATE_KNOT,
316d8c5c82eSViorel Suman .formats = AK5558_FORMATS,
317d8c5c82eSViorel Suman },
318d8c5c82eSViorel Suman .ops = &ak5558_dai_ops,
319d8c5c82eSViorel Suman };
320d8c5c82eSViorel Suman
ak5558_reset(struct ak5558_priv * ak5558,bool active)3214d5d75ceSShengjiu Wang static void ak5558_reset(struct ak5558_priv *ak5558, bool active)
32292088477SDaniel Baluta {
32392088477SDaniel Baluta if (!ak5558->reset_gpiod)
32492088477SDaniel Baluta return;
32592088477SDaniel Baluta
3264d5d75ceSShengjiu Wang gpiod_set_value_cansleep(ak5558->reset_gpiod, active);
32792088477SDaniel Baluta usleep_range(1000, 2000);
32892088477SDaniel Baluta }
32992088477SDaniel Baluta
ak5558_probe(struct snd_soc_component * component)3303f15aa19SKuninori Morimoto static int ak5558_probe(struct snd_soc_component *component)
33192088477SDaniel Baluta {
3323f15aa19SKuninori Morimoto struct ak5558_priv *ak5558 = snd_soc_component_get_drvdata(component);
33392088477SDaniel Baluta
3344d5d75ceSShengjiu Wang ak5558_reset(ak5558, false);
3353f15aa19SKuninori Morimoto return ak5558_set_mcki(component);
33692088477SDaniel Baluta }
33792088477SDaniel Baluta
ak5558_remove(struct snd_soc_component * component)3383f15aa19SKuninori Morimoto static void ak5558_remove(struct snd_soc_component *component)
33992088477SDaniel Baluta {
3403f15aa19SKuninori Morimoto struct ak5558_priv *ak5558 = snd_soc_component_get_drvdata(component);
34192088477SDaniel Baluta
3424d5d75ceSShengjiu Wang ak5558_reset(ak5558, true);
34392088477SDaniel Baluta }
34492088477SDaniel Baluta
ak5558_runtime_suspend(struct device * dev)34592088477SDaniel Baluta static int __maybe_unused ak5558_runtime_suspend(struct device *dev)
34692088477SDaniel Baluta {
34792088477SDaniel Baluta struct ak5558_priv *ak5558 = dev_get_drvdata(dev);
34892088477SDaniel Baluta
34992088477SDaniel Baluta regcache_cache_only(ak5558->regmap, true);
3504d5d75ceSShengjiu Wang ak5558_reset(ak5558, true);
35192088477SDaniel Baluta
3522ff6d5a1SShengjiu Wang regulator_bulk_disable(ARRAY_SIZE(ak5558->supplies),
3532ff6d5a1SShengjiu Wang ak5558->supplies);
35492088477SDaniel Baluta return 0;
35592088477SDaniel Baluta }
35692088477SDaniel Baluta
ak5558_runtime_resume(struct device * dev)35792088477SDaniel Baluta static int __maybe_unused ak5558_runtime_resume(struct device *dev)
35892088477SDaniel Baluta {
35992088477SDaniel Baluta struct ak5558_priv *ak5558 = dev_get_drvdata(dev);
3602ff6d5a1SShengjiu Wang int ret;
3612ff6d5a1SShengjiu Wang
3622ff6d5a1SShengjiu Wang ret = regulator_bulk_enable(ARRAY_SIZE(ak5558->supplies),
3632ff6d5a1SShengjiu Wang ak5558->supplies);
3642ff6d5a1SShengjiu Wang if (ret != 0) {
3652ff6d5a1SShengjiu Wang dev_err(dev, "Failed to enable supplies: %d\n", ret);
3662ff6d5a1SShengjiu Wang return ret;
3672ff6d5a1SShengjiu Wang }
36892088477SDaniel Baluta
3694d5d75ceSShengjiu Wang ak5558_reset(ak5558, true);
3704d5d75ceSShengjiu Wang ak5558_reset(ak5558, false);
37192088477SDaniel Baluta
37292088477SDaniel Baluta regcache_cache_only(ak5558->regmap, false);
37392088477SDaniel Baluta regcache_mark_dirty(ak5558->regmap);
37492088477SDaniel Baluta
37592088477SDaniel Baluta return regcache_sync(ak5558->regmap);
37692088477SDaniel Baluta }
37792088477SDaniel Baluta
378e380be7cSColin Ian King static const struct dev_pm_ops ak5558_pm = {
37992088477SDaniel Baluta SET_RUNTIME_PM_OPS(ak5558_runtime_suspend, ak5558_runtime_resume, NULL)
38092088477SDaniel Baluta SET_SYSTEM_SLEEP_PM_OPS(pm_runtime_force_suspend,
38192088477SDaniel Baluta pm_runtime_force_resume)
38292088477SDaniel Baluta };
38392088477SDaniel Baluta
384e380be7cSColin Ian King static const struct snd_soc_component_driver soc_codec_dev_ak5558 = {
38592088477SDaniel Baluta .probe = ak5558_probe,
38692088477SDaniel Baluta .remove = ak5558_remove,
38792088477SDaniel Baluta .controls = ak5558_snd_controls,
38892088477SDaniel Baluta .num_controls = ARRAY_SIZE(ak5558_snd_controls),
38992088477SDaniel Baluta .dapm_widgets = ak5558_dapm_widgets,
39092088477SDaniel Baluta .num_dapm_widgets = ARRAY_SIZE(ak5558_dapm_widgets),
39192088477SDaniel Baluta .dapm_routes = ak5558_intercon,
39292088477SDaniel Baluta .num_dapm_routes = ARRAY_SIZE(ak5558_intercon),
3933f15aa19SKuninori Morimoto .idle_bias_on = 1,
3943f15aa19SKuninori Morimoto .use_pmdown_time = 1,
3953f15aa19SKuninori Morimoto .endianness = 1,
39692088477SDaniel Baluta };
39792088477SDaniel Baluta
398d8c5c82eSViorel Suman static const struct snd_soc_component_driver soc_codec_dev_ak5552 = {
399d8c5c82eSViorel Suman .probe = ak5558_probe,
400d8c5c82eSViorel Suman .remove = ak5558_remove,
401d8c5c82eSViorel Suman .controls = ak5552_snd_controls,
402d8c5c82eSViorel Suman .num_controls = ARRAY_SIZE(ak5552_snd_controls),
403d8c5c82eSViorel Suman .dapm_widgets = ak5552_dapm_widgets,
404d8c5c82eSViorel Suman .num_dapm_widgets = ARRAY_SIZE(ak5552_dapm_widgets),
405d8c5c82eSViorel Suman .dapm_routes = ak5552_intercon,
406d8c5c82eSViorel Suman .num_dapm_routes = ARRAY_SIZE(ak5552_intercon),
407d8c5c82eSViorel Suman .idle_bias_on = 1,
408d8c5c82eSViorel Suman .use_pmdown_time = 1,
409d8c5c82eSViorel Suman .endianness = 1,
410d8c5c82eSViorel Suman };
411d8c5c82eSViorel Suman
41292088477SDaniel Baluta static const struct regmap_config ak5558_regmap = {
41392088477SDaniel Baluta .reg_bits = 8,
41492088477SDaniel Baluta .val_bits = 8,
41592088477SDaniel Baluta
41692088477SDaniel Baluta .max_register = AK5558_05_DSD,
41792088477SDaniel Baluta .reg_defaults = ak5558_reg,
41892088477SDaniel Baluta .num_reg_defaults = ARRAY_SIZE(ak5558_reg),
41992088477SDaniel Baluta .cache_type = REGCACHE_RBTREE,
42092088477SDaniel Baluta };
42192088477SDaniel Baluta
ak5558_i2c_probe(struct i2c_client * i2c)42292088477SDaniel Baluta static int ak5558_i2c_probe(struct i2c_client *i2c)
42392088477SDaniel Baluta {
42492088477SDaniel Baluta struct ak5558_priv *ak5558;
42592088477SDaniel Baluta int ret = 0;
426d8c5c82eSViorel Suman int dev_id;
4272ff6d5a1SShengjiu Wang int i;
42892088477SDaniel Baluta
42992088477SDaniel Baluta ak5558 = devm_kzalloc(&i2c->dev, sizeof(*ak5558), GFP_KERNEL);
43092088477SDaniel Baluta if (!ak5558)
43192088477SDaniel Baluta return -ENOMEM;
43292088477SDaniel Baluta
43392088477SDaniel Baluta ak5558->regmap = devm_regmap_init_i2c(i2c, &ak5558_regmap);
43492088477SDaniel Baluta if (IS_ERR(ak5558->regmap))
43592088477SDaniel Baluta return PTR_ERR(ak5558->regmap);
43692088477SDaniel Baluta
43792088477SDaniel Baluta i2c_set_clientdata(i2c, ak5558);
43892088477SDaniel Baluta ak5558->i2c = i2c;
43992088477SDaniel Baluta
44092088477SDaniel Baluta ak5558->reset_gpiod = devm_gpiod_get_optional(&i2c->dev, "reset",
44192088477SDaniel Baluta GPIOD_OUT_LOW);
44292088477SDaniel Baluta if (IS_ERR(ak5558->reset_gpiod))
44392088477SDaniel Baluta return PTR_ERR(ak5558->reset_gpiod);
44492088477SDaniel Baluta
4452ff6d5a1SShengjiu Wang for (i = 0; i < ARRAY_SIZE(ak5558->supplies); i++)
4462ff6d5a1SShengjiu Wang ak5558->supplies[i].supply = ak5558_supply_names[i];
4472ff6d5a1SShengjiu Wang
4482ff6d5a1SShengjiu Wang ret = devm_regulator_bulk_get(&i2c->dev, ARRAY_SIZE(ak5558->supplies),
4492ff6d5a1SShengjiu Wang ak5558->supplies);
4502ff6d5a1SShengjiu Wang if (ret != 0) {
4512ff6d5a1SShengjiu Wang dev_err(&i2c->dev, "Failed to request supplies: %d\n", ret);
4522ff6d5a1SShengjiu Wang return ret;
4532ff6d5a1SShengjiu Wang }
4542ff6d5a1SShengjiu Wang
455d8c5c82eSViorel Suman dev_id = (uintptr_t)of_device_get_match_data(&i2c->dev);
456d8c5c82eSViorel Suman switch (dev_id) {
457d8c5c82eSViorel Suman case AK5552:
458d8c5c82eSViorel Suman ret = devm_snd_soc_register_component(&i2c->dev,
459d8c5c82eSViorel Suman &soc_codec_dev_ak5552,
460d8c5c82eSViorel Suman &ak5552_dai, 1);
461d8c5c82eSViorel Suman break;
462d8c5c82eSViorel Suman case AK5558:
4633f15aa19SKuninori Morimoto ret = devm_snd_soc_register_component(&i2c->dev,
4643f15aa19SKuninori Morimoto &soc_codec_dev_ak5558,
46592088477SDaniel Baluta &ak5558_dai, 1);
466d8c5c82eSViorel Suman break;
467d8c5c82eSViorel Suman default:
468d8c5c82eSViorel Suman dev_err(&i2c->dev, "unexpected device type\n");
469d8c5c82eSViorel Suman return -EINVAL;
470d8c5c82eSViorel Suman }
471d8c5c82eSViorel Suman if (ret < 0) {
472d8c5c82eSViorel Suman dev_err(&i2c->dev, "failed to register component: %d\n", ret);
47392088477SDaniel Baluta return ret;
474d8c5c82eSViorel Suman }
47592088477SDaniel Baluta
47692088477SDaniel Baluta pm_runtime_enable(&i2c->dev);
4772ff6d5a1SShengjiu Wang regcache_cache_only(ak5558->regmap, true);
47892088477SDaniel Baluta
47992088477SDaniel Baluta return 0;
48092088477SDaniel Baluta }
48192088477SDaniel Baluta
ak5558_i2c_remove(struct i2c_client * i2c)482ed5c2f5fSUwe Kleine-König static void ak5558_i2c_remove(struct i2c_client *i2c)
48392088477SDaniel Baluta {
48492088477SDaniel Baluta pm_runtime_disable(&i2c->dev);
48592088477SDaniel Baluta }
48692088477SDaniel Baluta
4879f34c040SKrzysztof Kozlowski static const struct of_device_id ak5558_i2c_dt_ids[] __maybe_unused = {
488d8c5c82eSViorel Suman { .compatible = "asahi-kasei,ak5558", .data = (void *) AK5558 },
489d8c5c82eSViorel Suman { .compatible = "asahi-kasei,ak5552", .data = (void *) AK5552 },
49092088477SDaniel Baluta { }
49192088477SDaniel Baluta };
49280cffd24SShengjiu Wang MODULE_DEVICE_TABLE(of, ak5558_i2c_dt_ids);
49392088477SDaniel Baluta
49492088477SDaniel Baluta static struct i2c_driver ak5558_i2c_driver = {
49592088477SDaniel Baluta .driver = {
49692088477SDaniel Baluta .name = "ak5558",
49792088477SDaniel Baluta .of_match_table = of_match_ptr(ak5558_i2c_dt_ids),
49892088477SDaniel Baluta .pm = &ak5558_pm,
49992088477SDaniel Baluta },
500*9abcd240SUwe Kleine-König .probe = ak5558_i2c_probe,
50192088477SDaniel Baluta .remove = ak5558_i2c_remove,
50292088477SDaniel Baluta };
50392088477SDaniel Baluta
50492088477SDaniel Baluta module_i2c_driver(ak5558_i2c_driver);
50592088477SDaniel Baluta
50692088477SDaniel Baluta MODULE_AUTHOR("Junichi Wakasugi <wakasugi.jb@om.asahi-kasei.co.jp>");
50792088477SDaniel Baluta MODULE_AUTHOR("Mihai Serban <mihai.serban@nxp.com>");
50892088477SDaniel Baluta MODULE_DESCRIPTION("ASoC AK5558 ADC driver");
50992088477SDaniel Baluta MODULE_LICENSE("GPL v2");
510