162306a36Sopenharmony_ci/*
262306a36Sopenharmony_ci * ASoC codec driver for spear platform
362306a36Sopenharmony_ci *
462306a36Sopenharmony_ci * sound/soc/codecs/sta529.c -- spear ALSA Soc codec driver
562306a36Sopenharmony_ci *
662306a36Sopenharmony_ci * Copyright (C) 2012 ST Microelectronics
762306a36Sopenharmony_ci * Rajeev Kumar <rajeevkumar.linux@gmail.com>
862306a36Sopenharmony_ci *
962306a36Sopenharmony_ci * This file is licensed under the terms of the GNU General Public
1062306a36Sopenharmony_ci * License version 2. This program is licensed "as is" without any
1162306a36Sopenharmony_ci * warranty of any kind, whether express or implied.
1262306a36Sopenharmony_ci */
1362306a36Sopenharmony_ci
1462306a36Sopenharmony_ci#include <linux/clk.h>
1562306a36Sopenharmony_ci#include <linux/init.h>
1662306a36Sopenharmony_ci#include <linux/i2c.h>
1762306a36Sopenharmony_ci#include <linux/io.h>
1862306a36Sopenharmony_ci#include <linux/module.h>
1962306a36Sopenharmony_ci#include <linux/moduleparam.h>
2062306a36Sopenharmony_ci#include <linux/pm.h>
2162306a36Sopenharmony_ci#include <linux/regmap.h>
2262306a36Sopenharmony_ci#include <linux/slab.h>
2362306a36Sopenharmony_ci
2462306a36Sopenharmony_ci#include <sound/core.h>
2562306a36Sopenharmony_ci#include <sound/initval.h>
2662306a36Sopenharmony_ci#include <sound/pcm.h>
2762306a36Sopenharmony_ci#include <sound/pcm_params.h>
2862306a36Sopenharmony_ci#include <sound/soc.h>
2962306a36Sopenharmony_ci#include <sound/soc-dapm.h>
3062306a36Sopenharmony_ci#include <sound/tlv.h>
3162306a36Sopenharmony_ci
3262306a36Sopenharmony_ci/* STA529 Register offsets */
3362306a36Sopenharmony_ci#define	 STA529_FFXCFG0		0x00
3462306a36Sopenharmony_ci#define	 STA529_FFXCFG1		0x01
3562306a36Sopenharmony_ci#define	 STA529_MVOL		0x02
3662306a36Sopenharmony_ci#define	 STA529_LVOL		0x03
3762306a36Sopenharmony_ci#define	 STA529_RVOL		0x04
3862306a36Sopenharmony_ci#define	 STA529_TTF0		0x05
3962306a36Sopenharmony_ci#define	 STA529_TTF1		0x06
4062306a36Sopenharmony_ci#define	 STA529_TTP0		0x07
4162306a36Sopenharmony_ci#define	 STA529_TTP1		0x08
4262306a36Sopenharmony_ci#define	 STA529_S2PCFG0		0x0A
4362306a36Sopenharmony_ci#define	 STA529_S2PCFG1		0x0B
4462306a36Sopenharmony_ci#define	 STA529_P2SCFG0		0x0C
4562306a36Sopenharmony_ci#define	 STA529_P2SCFG1		0x0D
4662306a36Sopenharmony_ci#define	 STA529_PLLCFG0		0x14
4762306a36Sopenharmony_ci#define	 STA529_PLLCFG1		0x15
4862306a36Sopenharmony_ci#define	 STA529_PLLCFG2		0x16
4962306a36Sopenharmony_ci#define	 STA529_PLLCFG3		0x17
5062306a36Sopenharmony_ci#define	 STA529_PLLPFE		0x18
5162306a36Sopenharmony_ci#define	 STA529_PLLST		0x19
5262306a36Sopenharmony_ci#define	 STA529_ADCCFG		0x1E /*mic_select*/
5362306a36Sopenharmony_ci#define	 STA529_CKOCFG		0x1F
5462306a36Sopenharmony_ci#define	 STA529_MISC		0x20
5562306a36Sopenharmony_ci#define	 STA529_PADST0		0x21
5662306a36Sopenharmony_ci#define	 STA529_PADST1		0x22
5762306a36Sopenharmony_ci#define	 STA529_FFXST		0x23
5862306a36Sopenharmony_ci#define	 STA529_PWMIN1		0x2D
5962306a36Sopenharmony_ci#define	 STA529_PWMIN2		0x2E
6062306a36Sopenharmony_ci#define	 STA529_POWST		0x32
6162306a36Sopenharmony_ci
6262306a36Sopenharmony_ci#define STA529_MAX_REGISTER	0x32
6362306a36Sopenharmony_ci
6462306a36Sopenharmony_ci#define STA529_RATES		(SNDRV_PCM_RATE_8000 | \
6562306a36Sopenharmony_ci				SNDRV_PCM_RATE_11025 | \
6662306a36Sopenharmony_ci				SNDRV_PCM_RATE_16000 | \
6762306a36Sopenharmony_ci				SNDRV_PCM_RATE_22050 | \
6862306a36Sopenharmony_ci				SNDRV_PCM_RATE_32000 | \
6962306a36Sopenharmony_ci				SNDRV_PCM_RATE_44100 | \
7062306a36Sopenharmony_ci				SNDRV_PCM_RATE_48000)
7162306a36Sopenharmony_ci
7262306a36Sopenharmony_ci#define STA529_FORMAT		(SNDRV_PCM_FMTBIT_S16_LE | \
7362306a36Sopenharmony_ci				SNDRV_PCM_FMTBIT_S24_LE | \
7462306a36Sopenharmony_ci				SNDRV_PCM_FMTBIT_S32_LE)
7562306a36Sopenharmony_ci#define	S2PC_VALUE		0x98
7662306a36Sopenharmony_ci#define CLOCK_OUT		0x60
7762306a36Sopenharmony_ci#define DATA_FORMAT_MSK		0x0E
7862306a36Sopenharmony_ci#define LEFT_J_DATA_FORMAT	0x00
7962306a36Sopenharmony_ci#define I2S_DATA_FORMAT		0x02
8062306a36Sopenharmony_ci#define RIGHT_J_DATA_FORMAT	0x04
8162306a36Sopenharmony_ci#define CODEC_MUTE_VAL		0x80
8262306a36Sopenharmony_ci
8362306a36Sopenharmony_ci#define POWER_CNTLMSAK		0x40
8462306a36Sopenharmony_ci#define POWER_STDBY		0x40
8562306a36Sopenharmony_ci#define FFX_MASK		0x80
8662306a36Sopenharmony_ci#define FFX_OFF			0x80
8762306a36Sopenharmony_ci#define POWER_UP		0x00
8862306a36Sopenharmony_ci#define FFX_CLK_ENB		0x01
8962306a36Sopenharmony_ci#define FFX_CLK_DIS		0x00
9062306a36Sopenharmony_ci#define FFX_CLK_MSK		0x01
9162306a36Sopenharmony_ci#define PLAY_FREQ_RANGE_MSK	0x70
9262306a36Sopenharmony_ci#define CAP_FREQ_RANGE_MSK	0x0C
9362306a36Sopenharmony_ci#define PDATA_LEN_MSK		0xC0
9462306a36Sopenharmony_ci#define BCLK_TO_FS_MSK		0x30
9562306a36Sopenharmony_ci#define AUDIO_MUTE_MSK		0x80
9662306a36Sopenharmony_ci
9762306a36Sopenharmony_cistatic const struct reg_default sta529_reg_defaults[] = {
9862306a36Sopenharmony_ci	{ 0,  0x35 },     /* R0   - FFX Configuration reg 0 */
9962306a36Sopenharmony_ci	{ 1,  0xc8 },     /* R1   - FFX Configuration reg 1 */
10062306a36Sopenharmony_ci	{ 2,  0x50 },     /* R2   - Master Volume */
10162306a36Sopenharmony_ci	{ 3,  0x00 },     /* R3   - Left Volume */
10262306a36Sopenharmony_ci	{ 4,  0x00 },     /* R4  -  Right Volume */
10362306a36Sopenharmony_ci	{ 10, 0xb2 },     /* R10  - S2P Config Reg 0 */
10462306a36Sopenharmony_ci	{ 11, 0x41 },     /* R11  - S2P Config Reg 1 */
10562306a36Sopenharmony_ci	{ 12, 0x92 },     /* R12  - P2S Config Reg 0 */
10662306a36Sopenharmony_ci	{ 13, 0x41 },     /* R13  - P2S Config Reg 1 */
10762306a36Sopenharmony_ci	{ 30, 0xd2 },     /* R30  - ADC Config Reg */
10862306a36Sopenharmony_ci	{ 31, 0x40 },     /* R31  - clock Out Reg */
10962306a36Sopenharmony_ci	{ 32, 0x21 },     /* R32  - Misc Register */
11062306a36Sopenharmony_ci};
11162306a36Sopenharmony_ci
11262306a36Sopenharmony_cistruct sta529 {
11362306a36Sopenharmony_ci	struct regmap *regmap;
11462306a36Sopenharmony_ci};
11562306a36Sopenharmony_ci
11662306a36Sopenharmony_cistatic bool sta529_readable(struct device *dev, unsigned int reg)
11762306a36Sopenharmony_ci{
11862306a36Sopenharmony_ci	switch (reg) {
11962306a36Sopenharmony_ci
12062306a36Sopenharmony_ci	case STA529_FFXCFG0:
12162306a36Sopenharmony_ci	case STA529_FFXCFG1:
12262306a36Sopenharmony_ci	case STA529_MVOL:
12362306a36Sopenharmony_ci	case STA529_LVOL:
12462306a36Sopenharmony_ci	case STA529_RVOL:
12562306a36Sopenharmony_ci	case STA529_S2PCFG0:
12662306a36Sopenharmony_ci	case STA529_S2PCFG1:
12762306a36Sopenharmony_ci	case STA529_P2SCFG0:
12862306a36Sopenharmony_ci	case STA529_P2SCFG1:
12962306a36Sopenharmony_ci	case STA529_ADCCFG:
13062306a36Sopenharmony_ci	case STA529_CKOCFG:
13162306a36Sopenharmony_ci	case STA529_MISC:
13262306a36Sopenharmony_ci		return true;
13362306a36Sopenharmony_ci	default:
13462306a36Sopenharmony_ci		return false;
13562306a36Sopenharmony_ci	}
13662306a36Sopenharmony_ci}
13762306a36Sopenharmony_ci
13862306a36Sopenharmony_ci
13962306a36Sopenharmony_cistatic const char *pwm_mode_text[] = { "Binary", "Headphone", "Ternary",
14062306a36Sopenharmony_ci	"Phase-shift"};
14162306a36Sopenharmony_ci
14262306a36Sopenharmony_cistatic const DECLARE_TLV_DB_SCALE(out_gain_tlv, -9150, 50, 0);
14362306a36Sopenharmony_cistatic const DECLARE_TLV_DB_SCALE(master_vol_tlv, -12750, 50, 0);
14462306a36Sopenharmony_cistatic SOC_ENUM_SINGLE_DECL(pwm_src, STA529_FFXCFG1, 4, pwm_mode_text);
14562306a36Sopenharmony_ci
14662306a36Sopenharmony_cistatic const struct snd_kcontrol_new sta529_snd_controls[] = {
14762306a36Sopenharmony_ci	SOC_DOUBLE_R_TLV("Digital Playback Volume", STA529_LVOL, STA529_RVOL, 0,
14862306a36Sopenharmony_ci			127, 0, out_gain_tlv),
14962306a36Sopenharmony_ci	SOC_SINGLE_TLV("Master Playback Volume", STA529_MVOL, 0, 127, 1,
15062306a36Sopenharmony_ci			master_vol_tlv),
15162306a36Sopenharmony_ci	SOC_ENUM("PWM Select", pwm_src),
15262306a36Sopenharmony_ci};
15362306a36Sopenharmony_ci
15462306a36Sopenharmony_cistatic int sta529_set_bias_level(struct snd_soc_component *component, enum
15562306a36Sopenharmony_ci		snd_soc_bias_level level)
15662306a36Sopenharmony_ci{
15762306a36Sopenharmony_ci	struct sta529 *sta529 = snd_soc_component_get_drvdata(component);
15862306a36Sopenharmony_ci
15962306a36Sopenharmony_ci	switch (level) {
16062306a36Sopenharmony_ci	case SND_SOC_BIAS_ON:
16162306a36Sopenharmony_ci	case SND_SOC_BIAS_PREPARE:
16262306a36Sopenharmony_ci		snd_soc_component_update_bits(component, STA529_FFXCFG0, POWER_CNTLMSAK,
16362306a36Sopenharmony_ci				POWER_UP);
16462306a36Sopenharmony_ci		snd_soc_component_update_bits(component, STA529_MISC,	FFX_CLK_MSK,
16562306a36Sopenharmony_ci				FFX_CLK_ENB);
16662306a36Sopenharmony_ci		break;
16762306a36Sopenharmony_ci	case SND_SOC_BIAS_STANDBY:
16862306a36Sopenharmony_ci		if (snd_soc_component_get_bias_level(component) == SND_SOC_BIAS_OFF)
16962306a36Sopenharmony_ci			regcache_sync(sta529->regmap);
17062306a36Sopenharmony_ci		snd_soc_component_update_bits(component, STA529_FFXCFG0,
17162306a36Sopenharmony_ci					POWER_CNTLMSAK, POWER_STDBY);
17262306a36Sopenharmony_ci		/* Making FFX output to zero */
17362306a36Sopenharmony_ci		snd_soc_component_update_bits(component, STA529_FFXCFG0, FFX_MASK,
17462306a36Sopenharmony_ci				FFX_OFF);
17562306a36Sopenharmony_ci		snd_soc_component_update_bits(component, STA529_MISC, FFX_CLK_MSK,
17662306a36Sopenharmony_ci				FFX_CLK_DIS);
17762306a36Sopenharmony_ci		break;
17862306a36Sopenharmony_ci	case SND_SOC_BIAS_OFF:
17962306a36Sopenharmony_ci		break;
18062306a36Sopenharmony_ci	}
18162306a36Sopenharmony_ci
18262306a36Sopenharmony_ci	return 0;
18362306a36Sopenharmony_ci
18462306a36Sopenharmony_ci}
18562306a36Sopenharmony_ci
18662306a36Sopenharmony_cistatic int sta529_hw_params(struct snd_pcm_substream *substream,
18762306a36Sopenharmony_ci		struct snd_pcm_hw_params *params,
18862306a36Sopenharmony_ci		struct snd_soc_dai *dai)
18962306a36Sopenharmony_ci{
19062306a36Sopenharmony_ci	struct snd_soc_component *component = dai->component;
19162306a36Sopenharmony_ci	int pdata, play_freq_val, record_freq_val;
19262306a36Sopenharmony_ci	int bclk_to_fs_ratio;
19362306a36Sopenharmony_ci
19462306a36Sopenharmony_ci	switch (params_width(params)) {
19562306a36Sopenharmony_ci	case 16:
19662306a36Sopenharmony_ci		pdata = 1;
19762306a36Sopenharmony_ci		bclk_to_fs_ratio = 0;
19862306a36Sopenharmony_ci		break;
19962306a36Sopenharmony_ci	case 24:
20062306a36Sopenharmony_ci		pdata = 2;
20162306a36Sopenharmony_ci		bclk_to_fs_ratio = 1;
20262306a36Sopenharmony_ci		break;
20362306a36Sopenharmony_ci	case 32:
20462306a36Sopenharmony_ci		pdata = 3;
20562306a36Sopenharmony_ci		bclk_to_fs_ratio = 2;
20662306a36Sopenharmony_ci		break;
20762306a36Sopenharmony_ci	default:
20862306a36Sopenharmony_ci		dev_err(component->dev, "Unsupported format\n");
20962306a36Sopenharmony_ci		return -EINVAL;
21062306a36Sopenharmony_ci	}
21162306a36Sopenharmony_ci
21262306a36Sopenharmony_ci	switch (params_rate(params)) {
21362306a36Sopenharmony_ci	case 8000:
21462306a36Sopenharmony_ci	case 11025:
21562306a36Sopenharmony_ci		play_freq_val = 0;
21662306a36Sopenharmony_ci		record_freq_val = 2;
21762306a36Sopenharmony_ci		break;
21862306a36Sopenharmony_ci	case 16000:
21962306a36Sopenharmony_ci	case 22050:
22062306a36Sopenharmony_ci		play_freq_val = 1;
22162306a36Sopenharmony_ci		record_freq_val = 0;
22262306a36Sopenharmony_ci		break;
22362306a36Sopenharmony_ci
22462306a36Sopenharmony_ci	case 32000:
22562306a36Sopenharmony_ci	case 44100:
22662306a36Sopenharmony_ci	case 48000:
22762306a36Sopenharmony_ci		play_freq_val = 2;
22862306a36Sopenharmony_ci		record_freq_val = 0;
22962306a36Sopenharmony_ci		break;
23062306a36Sopenharmony_ci	default:
23162306a36Sopenharmony_ci		dev_err(component->dev, "Unsupported rate\n");
23262306a36Sopenharmony_ci		return -EINVAL;
23362306a36Sopenharmony_ci	}
23462306a36Sopenharmony_ci
23562306a36Sopenharmony_ci	if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
23662306a36Sopenharmony_ci		snd_soc_component_update_bits(component, STA529_S2PCFG1, PDATA_LEN_MSK,
23762306a36Sopenharmony_ci				pdata << 6);
23862306a36Sopenharmony_ci		snd_soc_component_update_bits(component, STA529_S2PCFG1, BCLK_TO_FS_MSK,
23962306a36Sopenharmony_ci				bclk_to_fs_ratio << 4);
24062306a36Sopenharmony_ci		snd_soc_component_update_bits(component, STA529_MISC, PLAY_FREQ_RANGE_MSK,
24162306a36Sopenharmony_ci				play_freq_val << 4);
24262306a36Sopenharmony_ci	} else {
24362306a36Sopenharmony_ci		snd_soc_component_update_bits(component, STA529_P2SCFG1, PDATA_LEN_MSK,
24462306a36Sopenharmony_ci				pdata << 6);
24562306a36Sopenharmony_ci		snd_soc_component_update_bits(component, STA529_P2SCFG1, BCLK_TO_FS_MSK,
24662306a36Sopenharmony_ci				bclk_to_fs_ratio << 4);
24762306a36Sopenharmony_ci		snd_soc_component_update_bits(component, STA529_MISC, CAP_FREQ_RANGE_MSK,
24862306a36Sopenharmony_ci				record_freq_val << 2);
24962306a36Sopenharmony_ci	}
25062306a36Sopenharmony_ci
25162306a36Sopenharmony_ci	return 0;
25262306a36Sopenharmony_ci}
25362306a36Sopenharmony_ci
25462306a36Sopenharmony_cistatic int sta529_mute(struct snd_soc_dai *dai, int mute, int direction)
25562306a36Sopenharmony_ci{
25662306a36Sopenharmony_ci	u8 val = 0;
25762306a36Sopenharmony_ci
25862306a36Sopenharmony_ci	if (mute)
25962306a36Sopenharmony_ci		val |= CODEC_MUTE_VAL;
26062306a36Sopenharmony_ci
26162306a36Sopenharmony_ci	snd_soc_component_update_bits(dai->component, STA529_FFXCFG0, AUDIO_MUTE_MSK, val);
26262306a36Sopenharmony_ci
26362306a36Sopenharmony_ci	return 0;
26462306a36Sopenharmony_ci}
26562306a36Sopenharmony_ci
26662306a36Sopenharmony_cistatic int sta529_set_dai_fmt(struct snd_soc_dai *codec_dai, u32 fmt)
26762306a36Sopenharmony_ci{
26862306a36Sopenharmony_ci	struct snd_soc_component *component = codec_dai->component;
26962306a36Sopenharmony_ci	u8 mode = 0;
27062306a36Sopenharmony_ci
27162306a36Sopenharmony_ci	/* interface format */
27262306a36Sopenharmony_ci	switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
27362306a36Sopenharmony_ci	case SND_SOC_DAIFMT_LEFT_J:
27462306a36Sopenharmony_ci		mode = LEFT_J_DATA_FORMAT;
27562306a36Sopenharmony_ci		break;
27662306a36Sopenharmony_ci	case SND_SOC_DAIFMT_I2S:
27762306a36Sopenharmony_ci		mode = I2S_DATA_FORMAT;
27862306a36Sopenharmony_ci		break;
27962306a36Sopenharmony_ci	case SND_SOC_DAIFMT_RIGHT_J:
28062306a36Sopenharmony_ci		mode = RIGHT_J_DATA_FORMAT;
28162306a36Sopenharmony_ci		break;
28262306a36Sopenharmony_ci	default:
28362306a36Sopenharmony_ci		return -EINVAL;
28462306a36Sopenharmony_ci	}
28562306a36Sopenharmony_ci
28662306a36Sopenharmony_ci	snd_soc_component_update_bits(component, STA529_S2PCFG0, DATA_FORMAT_MSK, mode);
28762306a36Sopenharmony_ci
28862306a36Sopenharmony_ci	return 0;
28962306a36Sopenharmony_ci}
29062306a36Sopenharmony_ci
29162306a36Sopenharmony_cistatic const struct snd_soc_dai_ops sta529_dai_ops = {
29262306a36Sopenharmony_ci	.hw_params	=	sta529_hw_params,
29362306a36Sopenharmony_ci	.set_fmt	=	sta529_set_dai_fmt,
29462306a36Sopenharmony_ci	.mute_stream	=	sta529_mute,
29562306a36Sopenharmony_ci	.no_capture_mute = 1,
29662306a36Sopenharmony_ci};
29762306a36Sopenharmony_ci
29862306a36Sopenharmony_cistatic struct snd_soc_dai_driver sta529_dai = {
29962306a36Sopenharmony_ci	.name = "sta529-audio",
30062306a36Sopenharmony_ci	.playback = {
30162306a36Sopenharmony_ci		.stream_name = "Playback",
30262306a36Sopenharmony_ci		.channels_min = 2,
30362306a36Sopenharmony_ci		.channels_max = 2,
30462306a36Sopenharmony_ci		.rates = STA529_RATES,
30562306a36Sopenharmony_ci		.formats = STA529_FORMAT,
30662306a36Sopenharmony_ci	},
30762306a36Sopenharmony_ci	.capture = {
30862306a36Sopenharmony_ci		.stream_name = "Capture",
30962306a36Sopenharmony_ci		.channels_min = 2,
31062306a36Sopenharmony_ci		.channels_max = 2,
31162306a36Sopenharmony_ci		.rates = STA529_RATES,
31262306a36Sopenharmony_ci		.formats = STA529_FORMAT,
31362306a36Sopenharmony_ci	},
31462306a36Sopenharmony_ci	.ops	= &sta529_dai_ops,
31562306a36Sopenharmony_ci};
31662306a36Sopenharmony_ci
31762306a36Sopenharmony_cistatic const struct snd_soc_component_driver sta529_component_driver = {
31862306a36Sopenharmony_ci	.set_bias_level		= sta529_set_bias_level,
31962306a36Sopenharmony_ci	.controls		= sta529_snd_controls,
32062306a36Sopenharmony_ci	.num_controls		= ARRAY_SIZE(sta529_snd_controls),
32162306a36Sopenharmony_ci	.suspend_bias_off	= 1,
32262306a36Sopenharmony_ci	.idle_bias_on		= 1,
32362306a36Sopenharmony_ci	.use_pmdown_time	= 1,
32462306a36Sopenharmony_ci	.endianness		= 1,
32562306a36Sopenharmony_ci};
32662306a36Sopenharmony_ci
32762306a36Sopenharmony_cistatic const struct regmap_config sta529_regmap = {
32862306a36Sopenharmony_ci	.reg_bits = 8,
32962306a36Sopenharmony_ci	.val_bits = 8,
33062306a36Sopenharmony_ci
33162306a36Sopenharmony_ci	.max_register = STA529_MAX_REGISTER,
33262306a36Sopenharmony_ci	.readable_reg = sta529_readable,
33362306a36Sopenharmony_ci
33462306a36Sopenharmony_ci	.cache_type = REGCACHE_MAPLE,
33562306a36Sopenharmony_ci	.reg_defaults = sta529_reg_defaults,
33662306a36Sopenharmony_ci	.num_reg_defaults = ARRAY_SIZE(sta529_reg_defaults),
33762306a36Sopenharmony_ci};
33862306a36Sopenharmony_ci
33962306a36Sopenharmony_cistatic int sta529_i2c_probe(struct i2c_client *i2c)
34062306a36Sopenharmony_ci{
34162306a36Sopenharmony_ci	struct sta529 *sta529;
34262306a36Sopenharmony_ci	int ret;
34362306a36Sopenharmony_ci
34462306a36Sopenharmony_ci	sta529 = devm_kzalloc(&i2c->dev, sizeof(struct sta529), GFP_KERNEL);
34562306a36Sopenharmony_ci	if (!sta529)
34662306a36Sopenharmony_ci		return -ENOMEM;
34762306a36Sopenharmony_ci
34862306a36Sopenharmony_ci	sta529->regmap = devm_regmap_init_i2c(i2c, &sta529_regmap);
34962306a36Sopenharmony_ci	if (IS_ERR(sta529->regmap)) {
35062306a36Sopenharmony_ci		ret = PTR_ERR(sta529->regmap);
35162306a36Sopenharmony_ci		dev_err(&i2c->dev, "Failed to allocate regmap: %d\n", ret);
35262306a36Sopenharmony_ci		return ret;
35362306a36Sopenharmony_ci	}
35462306a36Sopenharmony_ci
35562306a36Sopenharmony_ci	i2c_set_clientdata(i2c, sta529);
35662306a36Sopenharmony_ci
35762306a36Sopenharmony_ci	ret = devm_snd_soc_register_component(&i2c->dev,
35862306a36Sopenharmony_ci			&sta529_component_driver, &sta529_dai, 1);
35962306a36Sopenharmony_ci	if (ret != 0)
36062306a36Sopenharmony_ci		dev_err(&i2c->dev, "Failed to register CODEC: %d\n", ret);
36162306a36Sopenharmony_ci
36262306a36Sopenharmony_ci	return ret;
36362306a36Sopenharmony_ci}
36462306a36Sopenharmony_ci
36562306a36Sopenharmony_cistatic const struct i2c_device_id sta529_i2c_id[] = {
36662306a36Sopenharmony_ci	{ "sta529", 0 },
36762306a36Sopenharmony_ci	{ }
36862306a36Sopenharmony_ci};
36962306a36Sopenharmony_ciMODULE_DEVICE_TABLE(i2c, sta529_i2c_id);
37062306a36Sopenharmony_ci
37162306a36Sopenharmony_cistatic const struct of_device_id sta529_of_match[] = {
37262306a36Sopenharmony_ci	{ .compatible = "st,sta529", },
37362306a36Sopenharmony_ci	{ }
37462306a36Sopenharmony_ci};
37562306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, sta529_of_match);
37662306a36Sopenharmony_ci
37762306a36Sopenharmony_cistatic struct i2c_driver sta529_i2c_driver = {
37862306a36Sopenharmony_ci	.driver = {
37962306a36Sopenharmony_ci		.name = "sta529",
38062306a36Sopenharmony_ci		.of_match_table = sta529_of_match,
38162306a36Sopenharmony_ci	},
38262306a36Sopenharmony_ci	.probe		= sta529_i2c_probe,
38362306a36Sopenharmony_ci	.id_table	= sta529_i2c_id,
38462306a36Sopenharmony_ci};
38562306a36Sopenharmony_ci
38662306a36Sopenharmony_cimodule_i2c_driver(sta529_i2c_driver);
38762306a36Sopenharmony_ci
38862306a36Sopenharmony_ciMODULE_DESCRIPTION("ASoC STA529 codec driver");
38962306a36Sopenharmony_ciMODULE_AUTHOR("Rajeev Kumar <rajeevkumar.linux@gmail.com>");
39062306a36Sopenharmony_ciMODULE_LICENSE("GPL");
391