xref: /openbmc/linux/sound/soc/codecs/ak5558.c (revision 2612e3bbc0386368a850140a6c9b990cd496a5ec)
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