xref: /openbmc/linux/sound/soc/codecs/wm8988.c (revision fb7d79e56f6b0b7ce2d0ae3366d6a0e59145e37d)
15409fb4eSMark Brown /*
25409fb4eSMark Brown  * wm8988.c -- WM8988 ALSA SoC audio driver
35409fb4eSMark Brown  *
45409fb4eSMark Brown  * Copyright 2009 Wolfson Microelectronics plc
55409fb4eSMark Brown  * Copyright 2005 Openedhand Ltd.
65409fb4eSMark Brown  *
75409fb4eSMark Brown  * Author: Mark Brown <broonie@opensource.wolfsonmicro.com>
85409fb4eSMark Brown  *
95409fb4eSMark Brown  * This program is free software; you can redistribute it and/or modify
105409fb4eSMark Brown  * it under the terms of the GNU General Public License version 2 as
115409fb4eSMark Brown  * published by the Free Software Foundation.
125409fb4eSMark Brown  */
135409fb4eSMark Brown 
145409fb4eSMark Brown #include <linux/module.h>
155409fb4eSMark Brown #include <linux/moduleparam.h>
165409fb4eSMark Brown #include <linux/init.h>
175409fb4eSMark Brown #include <linux/delay.h>
185409fb4eSMark Brown #include <linux/pm.h>
195409fb4eSMark Brown #include <linux/i2c.h>
205409fb4eSMark Brown #include <linux/spi/spi.h>
215a0e3ad6STejun Heo #include <linux/slab.h>
225409fb4eSMark Brown #include <sound/core.h>
235409fb4eSMark Brown #include <sound/pcm.h>
245409fb4eSMark Brown #include <sound/pcm_params.h>
255409fb4eSMark Brown #include <sound/tlv.h>
265409fb4eSMark Brown #include <sound/soc.h>
275409fb4eSMark Brown #include <sound/initval.h>
285409fb4eSMark Brown 
295409fb4eSMark Brown #include "wm8988.h"
305409fb4eSMark Brown 
315409fb4eSMark Brown /*
325409fb4eSMark Brown  * wm8988 register cache
335409fb4eSMark Brown  * We can't read the WM8988 register space when we
345409fb4eSMark Brown  * are using 2 wire for device control, so we cache them instead.
355409fb4eSMark Brown  */
36d2dc0a77SMark Brown static const struct reg_default wm8988_reg_defaults[] = {
37d2dc0a77SMark Brown 	{ 0, 0x0097 },
38d2dc0a77SMark Brown 	{ 1, 0x0097 },
39d2dc0a77SMark Brown 	{ 2, 0x0079 },
40d2dc0a77SMark Brown 	{ 3, 0x0079 },
41d2dc0a77SMark Brown 	{ 5, 0x0008 },
42d2dc0a77SMark Brown 	{ 7, 0x000a },
43d2dc0a77SMark Brown 	{ 8, 0x0000 },
44d2dc0a77SMark Brown 	{ 10, 0x00ff },
45d2dc0a77SMark Brown 	{ 11, 0x00ff },
46d2dc0a77SMark Brown 	{ 12, 0x000f },
47d2dc0a77SMark Brown 	{ 13, 0x000f },
48d2dc0a77SMark Brown 	{ 16, 0x0000 },
49d2dc0a77SMark Brown 	{ 17, 0x007b },
50d2dc0a77SMark Brown 	{ 18, 0x0000 },
51d2dc0a77SMark Brown 	{ 19, 0x0032 },
52d2dc0a77SMark Brown 	{ 20, 0x0000 },
53d2dc0a77SMark Brown 	{ 21, 0x00c3 },
54d2dc0a77SMark Brown 	{ 22, 0x00c3 },
55d2dc0a77SMark Brown 	{ 23, 0x00c0 },
56d2dc0a77SMark Brown 	{ 24, 0x0000 },
57d2dc0a77SMark Brown 	{ 25, 0x0000 },
58d2dc0a77SMark Brown 	{ 26, 0x0000 },
59d2dc0a77SMark Brown 	{ 27, 0x0000 },
60d2dc0a77SMark Brown 	{ 31, 0x0000 },
61d2dc0a77SMark Brown 	{ 32, 0x0000 },
62d2dc0a77SMark Brown 	{ 33, 0x0000 },
63d2dc0a77SMark Brown 	{ 34, 0x0050 },
64d2dc0a77SMark Brown 	{ 35, 0x0050 },
65d2dc0a77SMark Brown 	{ 36, 0x0050 },
66d2dc0a77SMark Brown 	{ 37, 0x0050 },
67d2dc0a77SMark Brown 	{ 40, 0x0079 },
68d2dc0a77SMark Brown 	{ 41, 0x0079 },
69d2dc0a77SMark Brown 	{ 42, 0x0079 },
705409fb4eSMark Brown };
715409fb4eSMark Brown 
72d2dc0a77SMark Brown static bool wm8988_writeable(struct device *dev, unsigned int reg)
73d2dc0a77SMark Brown {
74d2dc0a77SMark Brown 	switch (reg) {
75d2dc0a77SMark Brown 	case WM8988_LINVOL:
76d2dc0a77SMark Brown 	case WM8988_RINVOL:
77d2dc0a77SMark Brown 	case WM8988_LOUT1V:
78d2dc0a77SMark Brown 	case WM8988_ROUT1V:
79d2dc0a77SMark Brown 	case WM8988_ADCDAC:
80d2dc0a77SMark Brown 	case WM8988_IFACE:
81d2dc0a77SMark Brown 	case WM8988_SRATE:
82d2dc0a77SMark Brown 	case WM8988_LDAC:
83d2dc0a77SMark Brown 	case WM8988_RDAC:
84d2dc0a77SMark Brown 	case WM8988_BASS:
85d2dc0a77SMark Brown 	case WM8988_TREBLE:
86d2dc0a77SMark Brown 	case WM8988_RESET:
87d2dc0a77SMark Brown 	case WM8988_3D:
88d2dc0a77SMark Brown 	case WM8988_ALC1:
89d2dc0a77SMark Brown 	case WM8988_ALC2:
90d2dc0a77SMark Brown 	case WM8988_ALC3:
91d2dc0a77SMark Brown 	case WM8988_NGATE:
92d2dc0a77SMark Brown 	case WM8988_LADC:
93d2dc0a77SMark Brown 	case WM8988_RADC:
94d2dc0a77SMark Brown 	case WM8988_ADCTL1:
95d2dc0a77SMark Brown 	case WM8988_ADCTL2:
96d2dc0a77SMark Brown 	case WM8988_PWR1:
97d2dc0a77SMark Brown 	case WM8988_PWR2:
98d2dc0a77SMark Brown 	case WM8988_ADCTL3:
99d2dc0a77SMark Brown 	case WM8988_ADCIN:
100d2dc0a77SMark Brown 	case WM8988_LADCIN:
101d2dc0a77SMark Brown 	case WM8988_RADCIN:
102d2dc0a77SMark Brown 	case WM8988_LOUTM1:
103d2dc0a77SMark Brown 	case WM8988_LOUTM2:
104d2dc0a77SMark Brown 	case WM8988_ROUTM1:
105d2dc0a77SMark Brown 	case WM8988_ROUTM2:
106d2dc0a77SMark Brown 	case WM8988_LOUT2V:
107d2dc0a77SMark Brown 	case WM8988_ROUT2V:
108d2dc0a77SMark Brown 	case WM8988_LPPB:
109d2dc0a77SMark Brown 		return true;
110d2dc0a77SMark Brown 	default:
111d2dc0a77SMark Brown 		return false;
112d2dc0a77SMark Brown 	}
113d2dc0a77SMark Brown }
114d2dc0a77SMark Brown 
1155409fb4eSMark Brown /* codec private data */
1165409fb4eSMark Brown struct wm8988_priv {
117d2dc0a77SMark Brown 	struct regmap *regmap;
1185409fb4eSMark Brown 	unsigned int sysclk;
119b57efda1SLars-Peter Clausen 	const struct snd_pcm_hw_constraint_list *sysclk_constraints;
1205409fb4eSMark Brown };
1215409fb4eSMark Brown 
12217a52fd6SMark Brown #define wm8988_reset(c)	snd_soc_write(c, WM8988_RESET, 0)
1235409fb4eSMark Brown 
1245409fb4eSMark Brown /*
1255409fb4eSMark Brown  * WM8988 Controls
1265409fb4eSMark Brown  */
1275409fb4eSMark Brown 
1285409fb4eSMark Brown static const char *bass_boost_txt[] = {"Linear Control", "Adaptive Boost"};
129b13a054aSTakashi Iwai static SOC_ENUM_SINGLE_DECL(bass_boost,
130b13a054aSTakashi Iwai 			    WM8988_BASS, 7, bass_boost_txt);
1315409fb4eSMark Brown 
1325409fb4eSMark Brown static const char *bass_filter_txt[] = { "130Hz @ 48kHz", "200Hz @ 48kHz" };
133b13a054aSTakashi Iwai static SOC_ENUM_SINGLE_DECL(bass_filter,
134b13a054aSTakashi Iwai 			    WM8988_BASS, 6, bass_filter_txt);
1355409fb4eSMark Brown 
1365409fb4eSMark Brown static const char *treble_txt[] = {"8kHz", "4kHz"};
137b13a054aSTakashi Iwai static SOC_ENUM_SINGLE_DECL(treble,
138b13a054aSTakashi Iwai 			    WM8988_TREBLE, 6, treble_txt);
1395409fb4eSMark Brown 
1405409fb4eSMark Brown static const char *stereo_3d_lc_txt[] = {"200Hz", "500Hz"};
141b13a054aSTakashi Iwai static SOC_ENUM_SINGLE_DECL(stereo_3d_lc,
142b13a054aSTakashi Iwai 			    WM8988_3D, 5, stereo_3d_lc_txt);
1435409fb4eSMark Brown 
1445409fb4eSMark Brown static const char *stereo_3d_uc_txt[] = {"2.2kHz", "1.5kHz"};
145b13a054aSTakashi Iwai static SOC_ENUM_SINGLE_DECL(stereo_3d_uc,
146b13a054aSTakashi Iwai 			    WM8988_3D, 6, stereo_3d_uc_txt);
1475409fb4eSMark Brown 
1485409fb4eSMark Brown static const char *stereo_3d_func_txt[] = {"Capture", "Playback"};
149b13a054aSTakashi Iwai static SOC_ENUM_SINGLE_DECL(stereo_3d_func,
150b13a054aSTakashi Iwai 			    WM8988_3D, 7, stereo_3d_func_txt);
1515409fb4eSMark Brown 
1525409fb4eSMark Brown static const char *alc_func_txt[] = {"Off", "Right", "Left", "Stereo"};
153b13a054aSTakashi Iwai static SOC_ENUM_SINGLE_DECL(alc_func,
154b13a054aSTakashi Iwai 			    WM8988_ALC1, 7, alc_func_txt);
1555409fb4eSMark Brown 
1565409fb4eSMark Brown static const char *ng_type_txt[] = {"Constant PGA Gain",
1575409fb4eSMark Brown 				    "Mute ADC Output"};
158b13a054aSTakashi Iwai static SOC_ENUM_SINGLE_DECL(ng_type,
159b13a054aSTakashi Iwai 			    WM8988_NGATE, 1, ng_type_txt);
1605409fb4eSMark Brown 
1615409fb4eSMark Brown static const char *deemph_txt[] = {"None", "32Khz", "44.1Khz", "48Khz"};
162b13a054aSTakashi Iwai static SOC_ENUM_SINGLE_DECL(deemph,
163b13a054aSTakashi Iwai 			    WM8988_ADCDAC, 1, deemph_txt);
1645409fb4eSMark Brown 
1655409fb4eSMark Brown static const char *adcpol_txt[] = {"Normal", "L Invert", "R Invert",
1665409fb4eSMark Brown 				   "L + R Invert"};
167b13a054aSTakashi Iwai static SOC_ENUM_SINGLE_DECL(adcpol,
168b13a054aSTakashi Iwai 			    WM8988_ADCDAC, 5, adcpol_txt);
1695409fb4eSMark Brown 
1705409fb4eSMark Brown static const DECLARE_TLV_DB_SCALE(pga_tlv, -1725, 75, 0);
1715409fb4eSMark Brown static const DECLARE_TLV_DB_SCALE(adc_tlv, -9750, 50, 1);
1725409fb4eSMark Brown static const DECLARE_TLV_DB_SCALE(dac_tlv, -12750, 50, 1);
1735409fb4eSMark Brown static const DECLARE_TLV_DB_SCALE(out_tlv, -12100, 100, 1);
1745409fb4eSMark Brown static const DECLARE_TLV_DB_SCALE(bypass_tlv, -1500, 300, 0);
1755409fb4eSMark Brown 
1765409fb4eSMark Brown static const struct snd_kcontrol_new wm8988_snd_controls[] = {
1775409fb4eSMark Brown 
1785409fb4eSMark Brown SOC_ENUM("Bass Boost", bass_boost),
1795409fb4eSMark Brown SOC_ENUM("Bass Filter", bass_filter),
1805409fb4eSMark Brown SOC_SINGLE("Bass Volume", WM8988_BASS, 0, 15, 1),
1815409fb4eSMark Brown 
1825409fb4eSMark Brown SOC_SINGLE("Treble Volume", WM8988_TREBLE, 0, 15, 0),
1835409fb4eSMark Brown SOC_ENUM("Treble Cut-off", treble),
1845409fb4eSMark Brown 
1855409fb4eSMark Brown SOC_SINGLE("3D Switch", WM8988_3D, 0, 1, 0),
1865409fb4eSMark Brown SOC_SINGLE("3D Volume", WM8988_3D, 1, 15, 0),
1875409fb4eSMark Brown SOC_ENUM("3D Lower Cut-off", stereo_3d_lc),
1885409fb4eSMark Brown SOC_ENUM("3D Upper Cut-off", stereo_3d_uc),
1895409fb4eSMark Brown SOC_ENUM("3D Mode", stereo_3d_func),
1905409fb4eSMark Brown 
1915409fb4eSMark Brown SOC_SINGLE("ALC Capture Target Volume", WM8988_ALC1, 0, 7, 0),
1925409fb4eSMark Brown SOC_SINGLE("ALC Capture Max Volume", WM8988_ALC1, 4, 7, 0),
1935409fb4eSMark Brown SOC_ENUM("ALC Capture Function", alc_func),
1945409fb4eSMark Brown SOC_SINGLE("ALC Capture ZC Switch", WM8988_ALC2, 7, 1, 0),
1955409fb4eSMark Brown SOC_SINGLE("ALC Capture Hold Time", WM8988_ALC2, 0, 15, 0),
1965409fb4eSMark Brown SOC_SINGLE("ALC Capture Decay Time", WM8988_ALC3, 4, 15, 0),
1975409fb4eSMark Brown SOC_SINGLE("ALC Capture Attack Time", WM8988_ALC3, 0, 15, 0),
1985409fb4eSMark Brown SOC_SINGLE("ALC Capture NG Threshold", WM8988_NGATE, 3, 31, 0),
1995409fb4eSMark Brown SOC_ENUM("ALC Capture NG Type", ng_type),
2005409fb4eSMark Brown SOC_SINGLE("ALC Capture NG Switch", WM8988_NGATE, 0, 1, 0),
2015409fb4eSMark Brown 
2025409fb4eSMark Brown SOC_SINGLE("ZC Timeout Switch", WM8988_ADCTL1, 0, 1, 0),
2035409fb4eSMark Brown 
2045409fb4eSMark Brown SOC_DOUBLE_R_TLV("Capture Digital Volume", WM8988_LADC, WM8988_RADC,
2055409fb4eSMark Brown 		 0, 255, 0, adc_tlv),
2065409fb4eSMark Brown SOC_DOUBLE_R_TLV("Capture Volume", WM8988_LINVOL, WM8988_RINVOL,
2075409fb4eSMark Brown 		 0, 63, 0, pga_tlv),
2085409fb4eSMark Brown SOC_DOUBLE_R("Capture ZC Switch", WM8988_LINVOL, WM8988_RINVOL, 6, 1, 0),
2095409fb4eSMark Brown SOC_DOUBLE_R("Capture Switch", WM8988_LINVOL, WM8988_RINVOL, 7, 1, 1),
2105409fb4eSMark Brown 
2115409fb4eSMark Brown SOC_ENUM("Playback De-emphasis", deemph),
2125409fb4eSMark Brown 
2135409fb4eSMark Brown SOC_ENUM("Capture Polarity", adcpol),
2145409fb4eSMark Brown SOC_SINGLE("Playback 6dB Attenuate", WM8988_ADCDAC, 7, 1, 0),
2155409fb4eSMark Brown SOC_SINGLE("Capture 6dB Attenuate", WM8988_ADCDAC, 8, 1, 0),
2165409fb4eSMark Brown 
2175409fb4eSMark Brown SOC_DOUBLE_R_TLV("PCM Volume", WM8988_LDAC, WM8988_RDAC, 0, 255, 0, dac_tlv),
2185409fb4eSMark Brown 
2195409fb4eSMark Brown SOC_SINGLE_TLV("Left Mixer Left Bypass Volume", WM8988_LOUTM1, 4, 7, 1,
2205409fb4eSMark Brown 	       bypass_tlv),
2215409fb4eSMark Brown SOC_SINGLE_TLV("Left Mixer Right Bypass Volume", WM8988_LOUTM2, 4, 7, 1,
2225409fb4eSMark Brown 	       bypass_tlv),
2235409fb4eSMark Brown SOC_SINGLE_TLV("Right Mixer Left Bypass Volume", WM8988_ROUTM1, 4, 7, 1,
2245409fb4eSMark Brown 	       bypass_tlv),
2255409fb4eSMark Brown SOC_SINGLE_TLV("Right Mixer Right Bypass Volume", WM8988_ROUTM2, 4, 7, 1,
2265409fb4eSMark Brown 	       bypass_tlv),
2275409fb4eSMark Brown 
2285409fb4eSMark Brown SOC_DOUBLE_R("Output 1 Playback ZC Switch", WM8988_LOUT1V,
2295409fb4eSMark Brown 	     WM8988_ROUT1V, 7, 1, 0),
2305409fb4eSMark Brown SOC_DOUBLE_R_TLV("Output 1 Playback Volume", WM8988_LOUT1V, WM8988_ROUT1V,
2315409fb4eSMark Brown 		 0, 127, 0, out_tlv),
2325409fb4eSMark Brown 
2335409fb4eSMark Brown SOC_DOUBLE_R("Output 2 Playback ZC Switch", WM8988_LOUT2V,
2345409fb4eSMark Brown 	     WM8988_ROUT2V, 7, 1, 0),
2355409fb4eSMark Brown SOC_DOUBLE_R_TLV("Output 2 Playback Volume", WM8988_LOUT2V, WM8988_ROUT2V,
2365409fb4eSMark Brown 		 0, 127, 0, out_tlv),
2375409fb4eSMark Brown 
2385409fb4eSMark Brown };
2395409fb4eSMark Brown 
2405409fb4eSMark Brown /*
2415409fb4eSMark Brown  * DAPM Controls
2425409fb4eSMark Brown  */
2435409fb4eSMark Brown 
2445409fb4eSMark Brown static int wm8988_lrc_control(struct snd_soc_dapm_widget *w,
2455409fb4eSMark Brown 			      struct snd_kcontrol *kcontrol, int event)
2465409fb4eSMark Brown {
2475409fb4eSMark Brown 	struct snd_soc_codec *codec = w->codec;
24817a52fd6SMark Brown 	u16 adctl2 = snd_soc_read(codec, WM8988_ADCTL2);
2495409fb4eSMark Brown 
2505409fb4eSMark Brown 	/* Use the DAC to gate LRC if active, otherwise use ADC */
25117a52fd6SMark Brown 	if (snd_soc_read(codec, WM8988_PWR2) & 0x180)
2525409fb4eSMark Brown 		adctl2 &= ~0x4;
2535409fb4eSMark Brown 	else
2545409fb4eSMark Brown 		adctl2 |= 0x4;
2555409fb4eSMark Brown 
25617a52fd6SMark Brown 	return snd_soc_write(codec, WM8988_ADCTL2, adctl2);
2575409fb4eSMark Brown }
2585409fb4eSMark Brown 
2595409fb4eSMark Brown static const char *wm8988_line_texts[] = {
2605409fb4eSMark Brown 	"Line 1", "Line 2", "PGA", "Differential"};
2615409fb4eSMark Brown 
2625409fb4eSMark Brown static const unsigned int wm8988_line_values[] = {
2635409fb4eSMark Brown 	0, 1, 3, 4};
2645409fb4eSMark Brown 
2655409fb4eSMark Brown static const struct soc_enum wm8988_lline_enum =
2665409fb4eSMark Brown 	SOC_VALUE_ENUM_SINGLE(WM8988_LOUTM1, 0, 7,
2675409fb4eSMark Brown 			      ARRAY_SIZE(wm8988_line_texts),
2685409fb4eSMark Brown 			      wm8988_line_texts,
2695409fb4eSMark Brown 			      wm8988_line_values);
2705409fb4eSMark Brown static const struct snd_kcontrol_new wm8988_left_line_controls =
271*fb7d79e5SLars-Peter Clausen 	SOC_DAPM_ENUM("Route", wm8988_lline_enum);
2725409fb4eSMark Brown 
2735409fb4eSMark Brown static const struct soc_enum wm8988_rline_enum =
2745409fb4eSMark Brown 	SOC_VALUE_ENUM_SINGLE(WM8988_ROUTM1, 0, 7,
2755409fb4eSMark Brown 			      ARRAY_SIZE(wm8988_line_texts),
2765409fb4eSMark Brown 			      wm8988_line_texts,
2775409fb4eSMark Brown 			      wm8988_line_values);
2785409fb4eSMark Brown static const struct snd_kcontrol_new wm8988_right_line_controls =
279*fb7d79e5SLars-Peter Clausen 	SOC_DAPM_ENUM("Route", wm8988_lline_enum);
2805409fb4eSMark Brown 
2815409fb4eSMark Brown /* Left Mixer */
2825409fb4eSMark Brown static const struct snd_kcontrol_new wm8988_left_mixer_controls[] = {
2835409fb4eSMark Brown 	SOC_DAPM_SINGLE("Playback Switch", WM8988_LOUTM1, 8, 1, 0),
2845409fb4eSMark Brown 	SOC_DAPM_SINGLE("Left Bypass Switch", WM8988_LOUTM1, 7, 1, 0),
2855409fb4eSMark Brown 	SOC_DAPM_SINGLE("Right Playback Switch", WM8988_LOUTM2, 8, 1, 0),
2865409fb4eSMark Brown 	SOC_DAPM_SINGLE("Right Bypass Switch", WM8988_LOUTM2, 7, 1, 0),
2875409fb4eSMark Brown };
2885409fb4eSMark Brown 
2895409fb4eSMark Brown /* Right Mixer */
2905409fb4eSMark Brown static const struct snd_kcontrol_new wm8988_right_mixer_controls[] = {
2915409fb4eSMark Brown 	SOC_DAPM_SINGLE("Left Playback Switch", WM8988_ROUTM1, 8, 1, 0),
2925409fb4eSMark Brown 	SOC_DAPM_SINGLE("Left Bypass Switch", WM8988_ROUTM1, 7, 1, 0),
2935409fb4eSMark Brown 	SOC_DAPM_SINGLE("Playback Switch", WM8988_ROUTM2, 8, 1, 0),
2945409fb4eSMark Brown 	SOC_DAPM_SINGLE("Right Bypass Switch", WM8988_ROUTM2, 7, 1, 0),
2955409fb4eSMark Brown };
2965409fb4eSMark Brown 
2975409fb4eSMark Brown static const char *wm8988_pga_sel[] = {"Line 1", "Line 2", "Differential"};
2985409fb4eSMark Brown static const unsigned int wm8988_pga_val[] = { 0, 1, 3 };
2995409fb4eSMark Brown 
3005409fb4eSMark Brown /* Left PGA Mux */
3015409fb4eSMark Brown static const struct soc_enum wm8988_lpga_enum =
3025409fb4eSMark Brown 	SOC_VALUE_ENUM_SINGLE(WM8988_LADCIN, 6, 3,
3035409fb4eSMark Brown 			      ARRAY_SIZE(wm8988_pga_sel),
3045409fb4eSMark Brown 			      wm8988_pga_sel,
3055409fb4eSMark Brown 			      wm8988_pga_val);
3065409fb4eSMark Brown static const struct snd_kcontrol_new wm8988_left_pga_controls =
307*fb7d79e5SLars-Peter Clausen 	SOC_DAPM_ENUM("Route", wm8988_lpga_enum);
3085409fb4eSMark Brown 
3095409fb4eSMark Brown /* Right PGA Mux */
3105409fb4eSMark Brown static const struct soc_enum wm8988_rpga_enum =
3115409fb4eSMark Brown 	SOC_VALUE_ENUM_SINGLE(WM8988_RADCIN, 6, 3,
3125409fb4eSMark Brown 			      ARRAY_SIZE(wm8988_pga_sel),
3135409fb4eSMark Brown 			      wm8988_pga_sel,
3145409fb4eSMark Brown 			      wm8988_pga_val);
3155409fb4eSMark Brown static const struct snd_kcontrol_new wm8988_right_pga_controls =
316*fb7d79e5SLars-Peter Clausen 	SOC_DAPM_ENUM("Route", wm8988_rpga_enum);
3175409fb4eSMark Brown 
3185409fb4eSMark Brown /* Differential Mux */
3195409fb4eSMark Brown static const char *wm8988_diff_sel[] = {"Line 1", "Line 2"};
320b13a054aSTakashi Iwai static SOC_ENUM_SINGLE_DECL(diffmux,
321b13a054aSTakashi Iwai 			    WM8988_ADCIN, 8, wm8988_diff_sel);
3225409fb4eSMark Brown static const struct snd_kcontrol_new wm8988_diffmux_controls =
3235409fb4eSMark Brown 	SOC_DAPM_ENUM("Route", diffmux);
3245409fb4eSMark Brown 
3255409fb4eSMark Brown /* Mono ADC Mux */
3265409fb4eSMark Brown static const char *wm8988_mono_mux[] = {"Stereo", "Mono (Left)",
3275409fb4eSMark Brown 	"Mono (Right)", "Digital Mono"};
328b13a054aSTakashi Iwai static SOC_ENUM_SINGLE_DECL(monomux,
329b13a054aSTakashi Iwai 			    WM8988_ADCIN, 6, wm8988_mono_mux);
3305409fb4eSMark Brown static const struct snd_kcontrol_new wm8988_monomux_controls =
3315409fb4eSMark Brown 	SOC_DAPM_ENUM("Route", monomux);
3325409fb4eSMark Brown 
3335409fb4eSMark Brown static const struct snd_soc_dapm_widget wm8988_dapm_widgets[] = {
334be48f20dSMark Brown 	SND_SOC_DAPM_SUPPLY("Mic Bias", WM8988_PWR1, 1, 0, NULL, 0),
3355409fb4eSMark Brown 
3365409fb4eSMark Brown 	SND_SOC_DAPM_MUX("Differential Mux", SND_SOC_NOPM, 0, 0,
3375409fb4eSMark Brown 		&wm8988_diffmux_controls),
3385409fb4eSMark Brown 	SND_SOC_DAPM_MUX("Left ADC Mux", SND_SOC_NOPM, 0, 0,
3395409fb4eSMark Brown 		&wm8988_monomux_controls),
3405409fb4eSMark Brown 	SND_SOC_DAPM_MUX("Right ADC Mux", SND_SOC_NOPM, 0, 0,
3415409fb4eSMark Brown 		&wm8988_monomux_controls),
3425409fb4eSMark Brown 
3435409fb4eSMark Brown 	SND_SOC_DAPM_MUX("Left PGA Mux", WM8988_PWR1, 5, 0,
3445409fb4eSMark Brown 		&wm8988_left_pga_controls),
3455409fb4eSMark Brown 	SND_SOC_DAPM_MUX("Right PGA Mux", WM8988_PWR1, 4, 0,
3465409fb4eSMark Brown 		&wm8988_right_pga_controls),
3475409fb4eSMark Brown 
3485409fb4eSMark Brown 	SND_SOC_DAPM_MUX("Left Line Mux", SND_SOC_NOPM, 0, 0,
3495409fb4eSMark Brown 		&wm8988_left_line_controls),
3505409fb4eSMark Brown 	SND_SOC_DAPM_MUX("Right Line Mux", SND_SOC_NOPM, 0, 0,
3515409fb4eSMark Brown 		&wm8988_right_line_controls),
3525409fb4eSMark Brown 
3535409fb4eSMark Brown 	SND_SOC_DAPM_ADC("Right ADC", "Right Capture", WM8988_PWR1, 2, 0),
3545409fb4eSMark Brown 	SND_SOC_DAPM_ADC("Left ADC", "Left Capture", WM8988_PWR1, 3, 0),
3555409fb4eSMark Brown 
3565409fb4eSMark Brown 	SND_SOC_DAPM_DAC("Right DAC", "Right Playback", WM8988_PWR2, 7, 0),
3575409fb4eSMark Brown 	SND_SOC_DAPM_DAC("Left DAC", "Left Playback", WM8988_PWR2, 8, 0),
3585409fb4eSMark Brown 
3595409fb4eSMark Brown 	SND_SOC_DAPM_MIXER("Left Mixer", SND_SOC_NOPM, 0, 0,
3605409fb4eSMark Brown 		&wm8988_left_mixer_controls[0],
3615409fb4eSMark Brown 		ARRAY_SIZE(wm8988_left_mixer_controls)),
3625409fb4eSMark Brown 	SND_SOC_DAPM_MIXER("Right Mixer", SND_SOC_NOPM, 0, 0,
3635409fb4eSMark Brown 		&wm8988_right_mixer_controls[0],
3645409fb4eSMark Brown 		ARRAY_SIZE(wm8988_right_mixer_controls)),
3655409fb4eSMark Brown 
3665409fb4eSMark Brown 	SND_SOC_DAPM_PGA("Right Out 2", WM8988_PWR2, 3, 0, NULL, 0),
3675409fb4eSMark Brown 	SND_SOC_DAPM_PGA("Left Out 2", WM8988_PWR2, 4, 0, NULL, 0),
3685409fb4eSMark Brown 	SND_SOC_DAPM_PGA("Right Out 1", WM8988_PWR2, 5, 0, NULL, 0),
3695409fb4eSMark Brown 	SND_SOC_DAPM_PGA("Left Out 1", WM8988_PWR2, 6, 0, NULL, 0),
3705409fb4eSMark Brown 
3715409fb4eSMark Brown 	SND_SOC_DAPM_POST("LRC control", wm8988_lrc_control),
3725409fb4eSMark Brown 
3735409fb4eSMark Brown 	SND_SOC_DAPM_OUTPUT("LOUT1"),
3745409fb4eSMark Brown 	SND_SOC_DAPM_OUTPUT("ROUT1"),
3755409fb4eSMark Brown 	SND_SOC_DAPM_OUTPUT("LOUT2"),
3765409fb4eSMark Brown 	SND_SOC_DAPM_OUTPUT("ROUT2"),
3775409fb4eSMark Brown 	SND_SOC_DAPM_OUTPUT("VREF"),
3785409fb4eSMark Brown 
3795409fb4eSMark Brown 	SND_SOC_DAPM_INPUT("LINPUT1"),
3805409fb4eSMark Brown 	SND_SOC_DAPM_INPUT("LINPUT2"),
3815409fb4eSMark Brown 	SND_SOC_DAPM_INPUT("RINPUT1"),
3825409fb4eSMark Brown 	SND_SOC_DAPM_INPUT("RINPUT2"),
3835409fb4eSMark Brown };
3845409fb4eSMark Brown 
385dd21353fSMark Brown static const struct snd_soc_dapm_route wm8988_dapm_routes[] = {
3865409fb4eSMark Brown 
3875409fb4eSMark Brown 	{ "Left Line Mux", "Line 1", "LINPUT1" },
3885409fb4eSMark Brown 	{ "Left Line Mux", "Line 2", "LINPUT2" },
3895409fb4eSMark Brown 	{ "Left Line Mux", "PGA", "Left PGA Mux" },
3905409fb4eSMark Brown 	{ "Left Line Mux", "Differential", "Differential Mux" },
3915409fb4eSMark Brown 
3925409fb4eSMark Brown 	{ "Right Line Mux", "Line 1", "RINPUT1" },
3935409fb4eSMark Brown 	{ "Right Line Mux", "Line 2", "RINPUT2" },
3945409fb4eSMark Brown 	{ "Right Line Mux", "PGA", "Right PGA Mux" },
3955409fb4eSMark Brown 	{ "Right Line Mux", "Differential", "Differential Mux" },
3965409fb4eSMark Brown 
3975409fb4eSMark Brown 	{ "Left PGA Mux", "Line 1", "LINPUT1" },
3985409fb4eSMark Brown 	{ "Left PGA Mux", "Line 2", "LINPUT2" },
3995409fb4eSMark Brown 	{ "Left PGA Mux", "Differential", "Differential Mux" },
4005409fb4eSMark Brown 
4015409fb4eSMark Brown 	{ "Right PGA Mux", "Line 1", "RINPUT1" },
4025409fb4eSMark Brown 	{ "Right PGA Mux", "Line 2", "RINPUT2" },
4035409fb4eSMark Brown 	{ "Right PGA Mux", "Differential", "Differential Mux" },
4045409fb4eSMark Brown 
4055409fb4eSMark Brown 	{ "Differential Mux", "Line 1", "LINPUT1" },
4065409fb4eSMark Brown 	{ "Differential Mux", "Line 1", "RINPUT1" },
4075409fb4eSMark Brown 	{ "Differential Mux", "Line 2", "LINPUT2" },
4085409fb4eSMark Brown 	{ "Differential Mux", "Line 2", "RINPUT2" },
4095409fb4eSMark Brown 
4105409fb4eSMark Brown 	{ "Left ADC Mux", "Stereo", "Left PGA Mux" },
4115409fb4eSMark Brown 	{ "Left ADC Mux", "Mono (Left)", "Left PGA Mux" },
4125409fb4eSMark Brown 	{ "Left ADC Mux", "Digital Mono", "Left PGA Mux" },
4135409fb4eSMark Brown 
4145409fb4eSMark Brown 	{ "Right ADC Mux", "Stereo", "Right PGA Mux" },
4155409fb4eSMark Brown 	{ "Right ADC Mux", "Mono (Right)", "Right PGA Mux" },
4165409fb4eSMark Brown 	{ "Right ADC Mux", "Digital Mono", "Right PGA Mux" },
4175409fb4eSMark Brown 
4185409fb4eSMark Brown 	{ "Left ADC", NULL, "Left ADC Mux" },
4195409fb4eSMark Brown 	{ "Right ADC", NULL, "Right ADC Mux" },
4205409fb4eSMark Brown 
4215409fb4eSMark Brown 	{ "Left Line Mux", "Line 1", "LINPUT1" },
4225409fb4eSMark Brown 	{ "Left Line Mux", "Line 2", "LINPUT2" },
4235409fb4eSMark Brown 	{ "Left Line Mux", "PGA", "Left PGA Mux" },
4245409fb4eSMark Brown 	{ "Left Line Mux", "Differential", "Differential Mux" },
4255409fb4eSMark Brown 
4265409fb4eSMark Brown 	{ "Right Line Mux", "Line 1", "RINPUT1" },
4275409fb4eSMark Brown 	{ "Right Line Mux", "Line 2", "RINPUT2" },
4285409fb4eSMark Brown 	{ "Right Line Mux", "PGA", "Right PGA Mux" },
4295409fb4eSMark Brown 	{ "Right Line Mux", "Differential", "Differential Mux" },
4305409fb4eSMark Brown 
4315409fb4eSMark Brown 	{ "Left Mixer", "Playback Switch", "Left DAC" },
4325409fb4eSMark Brown 	{ "Left Mixer", "Left Bypass Switch", "Left Line Mux" },
4335409fb4eSMark Brown 	{ "Left Mixer", "Right Playback Switch", "Right DAC" },
4345409fb4eSMark Brown 	{ "Left Mixer", "Right Bypass Switch", "Right Line Mux" },
4355409fb4eSMark Brown 
4365409fb4eSMark Brown 	{ "Right Mixer", "Left Playback Switch", "Left DAC" },
4375409fb4eSMark Brown 	{ "Right Mixer", "Left Bypass Switch", "Left Line Mux" },
4385409fb4eSMark Brown 	{ "Right Mixer", "Playback Switch", "Right DAC" },
4395409fb4eSMark Brown 	{ "Right Mixer", "Right Bypass Switch", "Right Line Mux" },
4405409fb4eSMark Brown 
4415409fb4eSMark Brown 	{ "Left Out 1", NULL, "Left Mixer" },
4425409fb4eSMark Brown 	{ "LOUT1", NULL, "Left Out 1" },
4435409fb4eSMark Brown 	{ "Right Out 1", NULL, "Right Mixer" },
4445409fb4eSMark Brown 	{ "ROUT1", NULL, "Right Out 1" },
4455409fb4eSMark Brown 
4465409fb4eSMark Brown 	{ "Left Out 2", NULL, "Left Mixer" },
4475409fb4eSMark Brown 	{ "LOUT2", NULL, "Left Out 2" },
4485409fb4eSMark Brown 	{ "Right Out 2", NULL, "Right Mixer" },
4495409fb4eSMark Brown 	{ "ROUT2", NULL, "Right Out 2" },
4505409fb4eSMark Brown };
4515409fb4eSMark Brown 
4525409fb4eSMark Brown struct _coeff_div {
4535409fb4eSMark Brown 	u32 mclk;
4545409fb4eSMark Brown 	u32 rate;
4555409fb4eSMark Brown 	u16 fs;
4565409fb4eSMark Brown 	u8 sr:5;
4575409fb4eSMark Brown 	u8 usb:1;
4585409fb4eSMark Brown };
4595409fb4eSMark Brown 
4605409fb4eSMark Brown /* codec hifi mclk clock divider coefficients */
4615409fb4eSMark Brown static const struct _coeff_div coeff_div[] = {
4625409fb4eSMark Brown 	/* 8k */
4635409fb4eSMark Brown 	{12288000, 8000, 1536, 0x6, 0x0},
4645409fb4eSMark Brown 	{11289600, 8000, 1408, 0x16, 0x0},
4655409fb4eSMark Brown 	{18432000, 8000, 2304, 0x7, 0x0},
4665409fb4eSMark Brown 	{16934400, 8000, 2112, 0x17, 0x0},
4675409fb4eSMark Brown 	{12000000, 8000, 1500, 0x6, 0x1},
4685409fb4eSMark Brown 
4695409fb4eSMark Brown 	/* 11.025k */
4705409fb4eSMark Brown 	{11289600, 11025, 1024, 0x18, 0x0},
4715409fb4eSMark Brown 	{16934400, 11025, 1536, 0x19, 0x0},
4725409fb4eSMark Brown 	{12000000, 11025, 1088, 0x19, 0x1},
4735409fb4eSMark Brown 
4745409fb4eSMark Brown 	/* 16k */
4755409fb4eSMark Brown 	{12288000, 16000, 768, 0xa, 0x0},
4765409fb4eSMark Brown 	{18432000, 16000, 1152, 0xb, 0x0},
4775409fb4eSMark Brown 	{12000000, 16000, 750, 0xa, 0x1},
4785409fb4eSMark Brown 
4795409fb4eSMark Brown 	/* 22.05k */
4805409fb4eSMark Brown 	{11289600, 22050, 512, 0x1a, 0x0},
4815409fb4eSMark Brown 	{16934400, 22050, 768, 0x1b, 0x0},
4825409fb4eSMark Brown 	{12000000, 22050, 544, 0x1b, 0x1},
4835409fb4eSMark Brown 
4845409fb4eSMark Brown 	/* 32k */
4855409fb4eSMark Brown 	{12288000, 32000, 384, 0xc, 0x0},
4865409fb4eSMark Brown 	{18432000, 32000, 576, 0xd, 0x0},
4875409fb4eSMark Brown 	{12000000, 32000, 375, 0xa, 0x1},
4885409fb4eSMark Brown 
4895409fb4eSMark Brown 	/* 44.1k */
4905409fb4eSMark Brown 	{11289600, 44100, 256, 0x10, 0x0},
4915409fb4eSMark Brown 	{16934400, 44100, 384, 0x11, 0x0},
4925409fb4eSMark Brown 	{12000000, 44100, 272, 0x11, 0x1},
4935409fb4eSMark Brown 
4945409fb4eSMark Brown 	/* 48k */
4955409fb4eSMark Brown 	{12288000, 48000, 256, 0x0, 0x0},
4965409fb4eSMark Brown 	{18432000, 48000, 384, 0x1, 0x0},
4975409fb4eSMark Brown 	{12000000, 48000, 250, 0x0, 0x1},
4985409fb4eSMark Brown 
4995409fb4eSMark Brown 	/* 88.2k */
5005409fb4eSMark Brown 	{11289600, 88200, 128, 0x1e, 0x0},
5015409fb4eSMark Brown 	{16934400, 88200, 192, 0x1f, 0x0},
5025409fb4eSMark Brown 	{12000000, 88200, 136, 0x1f, 0x1},
5035409fb4eSMark Brown 
5045409fb4eSMark Brown 	/* 96k */
5055409fb4eSMark Brown 	{12288000, 96000, 128, 0xe, 0x0},
5065409fb4eSMark Brown 	{18432000, 96000, 192, 0xf, 0x0},
5075409fb4eSMark Brown 	{12000000, 96000, 125, 0xe, 0x1},
5085409fb4eSMark Brown };
5095409fb4eSMark Brown 
5105409fb4eSMark Brown static inline int get_coeff(int mclk, int rate)
5115409fb4eSMark Brown {
5125409fb4eSMark Brown 	int i;
5135409fb4eSMark Brown 
5145409fb4eSMark Brown 	for (i = 0; i < ARRAY_SIZE(coeff_div); i++) {
5155409fb4eSMark Brown 		if (coeff_div[i].rate == rate && coeff_div[i].mclk == mclk)
5165409fb4eSMark Brown 			return i;
5175409fb4eSMark Brown 	}
5185409fb4eSMark Brown 
5195409fb4eSMark Brown 	return -EINVAL;
5205409fb4eSMark Brown }
5215409fb4eSMark Brown 
5225409fb4eSMark Brown /* The set of rates we can generate from the above for each SYSCLK */
5235409fb4eSMark Brown 
524b57efda1SLars-Peter Clausen static const unsigned int rates_12288[] = {
5255409fb4eSMark Brown 	8000, 12000, 16000, 24000, 24000, 32000, 48000, 96000,
5265409fb4eSMark Brown };
5275409fb4eSMark Brown 
528b57efda1SLars-Peter Clausen static const struct snd_pcm_hw_constraint_list constraints_12288 = {
5295409fb4eSMark Brown 	.count	= ARRAY_SIZE(rates_12288),
5305409fb4eSMark Brown 	.list	= rates_12288,
5315409fb4eSMark Brown };
5325409fb4eSMark Brown 
533b57efda1SLars-Peter Clausen static const unsigned int rates_112896[] = {
5345409fb4eSMark Brown 	8000, 11025, 22050, 44100,
5355409fb4eSMark Brown };
5365409fb4eSMark Brown 
537b57efda1SLars-Peter Clausen static const struct snd_pcm_hw_constraint_list constraints_112896 = {
5385409fb4eSMark Brown 	.count	= ARRAY_SIZE(rates_112896),
5395409fb4eSMark Brown 	.list	= rates_112896,
5405409fb4eSMark Brown };
5415409fb4eSMark Brown 
542b57efda1SLars-Peter Clausen static const unsigned int rates_12[] = {
5435409fb4eSMark Brown 	8000, 11025, 12000, 16000, 22050, 2400, 32000, 41100, 48000,
5445409fb4eSMark Brown 	48000, 88235, 96000,
5455409fb4eSMark Brown };
5465409fb4eSMark Brown 
547b57efda1SLars-Peter Clausen static const struct snd_pcm_hw_constraint_list constraints_12 = {
5485409fb4eSMark Brown 	.count	= ARRAY_SIZE(rates_12),
5495409fb4eSMark Brown 	.list	= rates_12,
5505409fb4eSMark Brown };
5515409fb4eSMark Brown 
5525409fb4eSMark Brown /*
5535409fb4eSMark Brown  * Note that this should be called from init rather than from hw_params.
5545409fb4eSMark Brown  */
5555409fb4eSMark Brown static int wm8988_set_dai_sysclk(struct snd_soc_dai *codec_dai,
5565409fb4eSMark Brown 		int clk_id, unsigned int freq, int dir)
5575409fb4eSMark Brown {
5585409fb4eSMark Brown 	struct snd_soc_codec *codec = codec_dai->codec;
559b2c812e2SMark Brown 	struct wm8988_priv *wm8988 = snd_soc_codec_get_drvdata(codec);
5605409fb4eSMark Brown 
5615409fb4eSMark Brown 	switch (freq) {
5625409fb4eSMark Brown 	case 11289600:
5635409fb4eSMark Brown 	case 18432000:
5645409fb4eSMark Brown 	case 22579200:
5655409fb4eSMark Brown 	case 36864000:
5665409fb4eSMark Brown 		wm8988->sysclk_constraints = &constraints_112896;
5675409fb4eSMark Brown 		wm8988->sysclk = freq;
5685409fb4eSMark Brown 		return 0;
5695409fb4eSMark Brown 
5705409fb4eSMark Brown 	case 12288000:
5715409fb4eSMark Brown 	case 16934400:
5725409fb4eSMark Brown 	case 24576000:
5735409fb4eSMark Brown 	case 33868800:
5745409fb4eSMark Brown 		wm8988->sysclk_constraints = &constraints_12288;
5755409fb4eSMark Brown 		wm8988->sysclk = freq;
5765409fb4eSMark Brown 		return 0;
5775409fb4eSMark Brown 
5785409fb4eSMark Brown 	case 12000000:
5795409fb4eSMark Brown 	case 24000000:
5805409fb4eSMark Brown 		wm8988->sysclk_constraints = &constraints_12;
5815409fb4eSMark Brown 		wm8988->sysclk = freq;
5825409fb4eSMark Brown 		return 0;
5835409fb4eSMark Brown 	}
5845409fb4eSMark Brown 	return -EINVAL;
5855409fb4eSMark Brown }
5865409fb4eSMark Brown 
5875409fb4eSMark Brown static int wm8988_set_dai_fmt(struct snd_soc_dai *codec_dai,
5885409fb4eSMark Brown 		unsigned int fmt)
5895409fb4eSMark Brown {
5905409fb4eSMark Brown 	struct snd_soc_codec *codec = codec_dai->codec;
5915409fb4eSMark Brown 	u16 iface = 0;
5925409fb4eSMark Brown 
5935409fb4eSMark Brown 	/* set master/slave audio interface */
5945409fb4eSMark Brown 	switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) {
5955409fb4eSMark Brown 	case SND_SOC_DAIFMT_CBM_CFM:
5965409fb4eSMark Brown 		iface = 0x0040;
5975409fb4eSMark Brown 		break;
5985409fb4eSMark Brown 	case SND_SOC_DAIFMT_CBS_CFS:
5995409fb4eSMark Brown 		break;
6005409fb4eSMark Brown 	default:
6015409fb4eSMark Brown 		return -EINVAL;
6025409fb4eSMark Brown 	}
6035409fb4eSMark Brown 
6045409fb4eSMark Brown 	/* interface format */
6055409fb4eSMark Brown 	switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
6065409fb4eSMark Brown 	case SND_SOC_DAIFMT_I2S:
6075409fb4eSMark Brown 		iface |= 0x0002;
6085409fb4eSMark Brown 		break;
6095409fb4eSMark Brown 	case SND_SOC_DAIFMT_RIGHT_J:
6105409fb4eSMark Brown 		break;
6115409fb4eSMark Brown 	case SND_SOC_DAIFMT_LEFT_J:
6125409fb4eSMark Brown 		iface |= 0x0001;
6135409fb4eSMark Brown 		break;
6145409fb4eSMark Brown 	case SND_SOC_DAIFMT_DSP_A:
6155409fb4eSMark Brown 		iface |= 0x0003;
6165409fb4eSMark Brown 		break;
6175409fb4eSMark Brown 	case SND_SOC_DAIFMT_DSP_B:
6185409fb4eSMark Brown 		iface |= 0x0013;
6195409fb4eSMark Brown 		break;
6205409fb4eSMark Brown 	default:
6215409fb4eSMark Brown 		return -EINVAL;
6225409fb4eSMark Brown 	}
6235409fb4eSMark Brown 
6245409fb4eSMark Brown 	/* clock inversion */
6255409fb4eSMark Brown 	switch (fmt & SND_SOC_DAIFMT_INV_MASK) {
6265409fb4eSMark Brown 	case SND_SOC_DAIFMT_NB_NF:
6275409fb4eSMark Brown 		break;
6285409fb4eSMark Brown 	case SND_SOC_DAIFMT_IB_IF:
6295409fb4eSMark Brown 		iface |= 0x0090;
6305409fb4eSMark Brown 		break;
6315409fb4eSMark Brown 	case SND_SOC_DAIFMT_IB_NF:
6325409fb4eSMark Brown 		iface |= 0x0080;
6335409fb4eSMark Brown 		break;
6345409fb4eSMark Brown 	case SND_SOC_DAIFMT_NB_IF:
6355409fb4eSMark Brown 		iface |= 0x0010;
6365409fb4eSMark Brown 		break;
6375409fb4eSMark Brown 	default:
6385409fb4eSMark Brown 		return -EINVAL;
6395409fb4eSMark Brown 	}
6405409fb4eSMark Brown 
64117a52fd6SMark Brown 	snd_soc_write(codec, WM8988_IFACE, iface);
6425409fb4eSMark Brown 	return 0;
6435409fb4eSMark Brown }
6445409fb4eSMark Brown 
6455409fb4eSMark Brown static int wm8988_pcm_startup(struct snd_pcm_substream *substream,
6465409fb4eSMark Brown 			      struct snd_soc_dai *dai)
6475409fb4eSMark Brown {
6485409fb4eSMark Brown 	struct snd_soc_codec *codec = dai->codec;
649b2c812e2SMark Brown 	struct wm8988_priv *wm8988 = snd_soc_codec_get_drvdata(codec);
6505409fb4eSMark Brown 
6515409fb4eSMark Brown 	/* The set of sample rates that can be supported depends on the
6525409fb4eSMark Brown 	 * MCLK supplied to the CODEC - enforce this.
6535409fb4eSMark Brown 	 */
6545409fb4eSMark Brown 	if (!wm8988->sysclk) {
6555409fb4eSMark Brown 		dev_err(codec->dev,
6565409fb4eSMark Brown 			"No MCLK configured, call set_sysclk() on init\n");
6575409fb4eSMark Brown 		return -EINVAL;
6585409fb4eSMark Brown 	}
6595409fb4eSMark Brown 
6605409fb4eSMark Brown 	snd_pcm_hw_constraint_list(substream->runtime, 0,
6615409fb4eSMark Brown 				   SNDRV_PCM_HW_PARAM_RATE,
6625409fb4eSMark Brown 				   wm8988->sysclk_constraints);
6635409fb4eSMark Brown 
6645409fb4eSMark Brown 	return 0;
6655409fb4eSMark Brown }
6665409fb4eSMark Brown 
6675409fb4eSMark Brown static int wm8988_pcm_hw_params(struct snd_pcm_substream *substream,
6685409fb4eSMark Brown 				struct snd_pcm_hw_params *params,
6695409fb4eSMark Brown 				struct snd_soc_dai *dai)
6705409fb4eSMark Brown {
671e6968a17SMark Brown 	struct snd_soc_codec *codec = dai->codec;
672b2c812e2SMark Brown 	struct wm8988_priv *wm8988 = snd_soc_codec_get_drvdata(codec);
67317a52fd6SMark Brown 	u16 iface = snd_soc_read(codec, WM8988_IFACE) & 0x1f3;
67417a52fd6SMark Brown 	u16 srate = snd_soc_read(codec, WM8988_SRATE) & 0x180;
6755409fb4eSMark Brown 	int coeff;
6765409fb4eSMark Brown 
6775409fb4eSMark Brown 	coeff = get_coeff(wm8988->sysclk, params_rate(params));
6785409fb4eSMark Brown 	if (coeff < 0) {
6795409fb4eSMark Brown 		coeff = get_coeff(wm8988->sysclk / 2, params_rate(params));
6805409fb4eSMark Brown 		srate |= 0x40;
6815409fb4eSMark Brown 	}
6825409fb4eSMark Brown 	if (coeff < 0) {
6835409fb4eSMark Brown 		dev_err(codec->dev,
6845409fb4eSMark Brown 			"Unable to configure sample rate %dHz with %dHz MCLK\n",
6855409fb4eSMark Brown 			params_rate(params), wm8988->sysclk);
6865409fb4eSMark Brown 		return coeff;
6875409fb4eSMark Brown 	}
6885409fb4eSMark Brown 
6895409fb4eSMark Brown 	/* bit size */
6905409fb4eSMark Brown 	switch (params_format(params)) {
6915409fb4eSMark Brown 	case SNDRV_PCM_FORMAT_S16_LE:
6925409fb4eSMark Brown 		break;
6935409fb4eSMark Brown 	case SNDRV_PCM_FORMAT_S20_3LE:
6945409fb4eSMark Brown 		iface |= 0x0004;
6955409fb4eSMark Brown 		break;
6965409fb4eSMark Brown 	case SNDRV_PCM_FORMAT_S24_LE:
6975409fb4eSMark Brown 		iface |= 0x0008;
6985409fb4eSMark Brown 		break;
6995409fb4eSMark Brown 	case SNDRV_PCM_FORMAT_S32_LE:
7005409fb4eSMark Brown 		iface |= 0x000c;
7015409fb4eSMark Brown 		break;
7025409fb4eSMark Brown 	}
7035409fb4eSMark Brown 
7045409fb4eSMark Brown 	/* set iface & srate */
70517a52fd6SMark Brown 	snd_soc_write(codec, WM8988_IFACE, iface);
7065409fb4eSMark Brown 	if (coeff >= 0)
70717a52fd6SMark Brown 		snd_soc_write(codec, WM8988_SRATE, srate |
7085409fb4eSMark Brown 			(coeff_div[coeff].sr << 1) | coeff_div[coeff].usb);
7095409fb4eSMark Brown 
7105409fb4eSMark Brown 	return 0;
7115409fb4eSMark Brown }
7125409fb4eSMark Brown 
7135409fb4eSMark Brown static int wm8988_mute(struct snd_soc_dai *dai, int mute)
7145409fb4eSMark Brown {
7155409fb4eSMark Brown 	struct snd_soc_codec *codec = dai->codec;
71617a52fd6SMark Brown 	u16 mute_reg = snd_soc_read(codec, WM8988_ADCDAC) & 0xfff7;
7175409fb4eSMark Brown 
7185409fb4eSMark Brown 	if (mute)
71917a52fd6SMark Brown 		snd_soc_write(codec, WM8988_ADCDAC, mute_reg | 0x8);
7205409fb4eSMark Brown 	else
72117a52fd6SMark Brown 		snd_soc_write(codec, WM8988_ADCDAC, mute_reg);
7225409fb4eSMark Brown 	return 0;
7235409fb4eSMark Brown }
7245409fb4eSMark Brown 
7255409fb4eSMark Brown static int wm8988_set_bias_level(struct snd_soc_codec *codec,
7265409fb4eSMark Brown 				 enum snd_soc_bias_level level)
7275409fb4eSMark Brown {
728d2dc0a77SMark Brown 	struct wm8988_priv *wm8988 = snd_soc_codec_get_drvdata(codec);
72917a52fd6SMark Brown 	u16 pwr_reg = snd_soc_read(codec, WM8988_PWR1) & ~0x1c1;
7305409fb4eSMark Brown 
7315409fb4eSMark Brown 	switch (level) {
7325409fb4eSMark Brown 	case SND_SOC_BIAS_ON:
7335409fb4eSMark Brown 		break;
7345409fb4eSMark Brown 
7355409fb4eSMark Brown 	case SND_SOC_BIAS_PREPARE:
7365409fb4eSMark Brown 		/* VREF, VMID=2x50k, digital enabled */
73717a52fd6SMark Brown 		snd_soc_write(codec, WM8988_PWR1, pwr_reg | 0x00c0);
7385409fb4eSMark Brown 		break;
7395409fb4eSMark Brown 
7405409fb4eSMark Brown 	case SND_SOC_BIAS_STANDBY:
741ce6120ccSLiam Girdwood 		if (codec->dapm.bias_level == SND_SOC_BIAS_OFF) {
742d2dc0a77SMark Brown 			regcache_sync(wm8988->regmap);
743fa5fdb47SAxel Lin 
7445409fb4eSMark Brown 			/* VREF, VMID=2x5k */
74517a52fd6SMark Brown 			snd_soc_write(codec, WM8988_PWR1, pwr_reg | 0x1c1);
7465409fb4eSMark Brown 
7475409fb4eSMark Brown 			/* Charge caps */
7485409fb4eSMark Brown 			msleep(100);
7495409fb4eSMark Brown 		}
7505409fb4eSMark Brown 
7515409fb4eSMark Brown 		/* VREF, VMID=2*500k, digital stopped */
75217a52fd6SMark Brown 		snd_soc_write(codec, WM8988_PWR1, pwr_reg | 0x0141);
7535409fb4eSMark Brown 		break;
7545409fb4eSMark Brown 
7555409fb4eSMark Brown 	case SND_SOC_BIAS_OFF:
75617a52fd6SMark Brown 		snd_soc_write(codec, WM8988_PWR1, 0x0000);
7575409fb4eSMark Brown 		break;
7585409fb4eSMark Brown 	}
759ce6120ccSLiam Girdwood 	codec->dapm.bias_level = level;
7605409fb4eSMark Brown 	return 0;
7615409fb4eSMark Brown }
7625409fb4eSMark Brown 
7635409fb4eSMark Brown #define WM8988_RATES SNDRV_PCM_RATE_8000_96000
7645409fb4eSMark Brown 
7655409fb4eSMark Brown #define WM8988_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE |\
7665409fb4eSMark Brown 	SNDRV_PCM_FMTBIT_S24_LE)
7675409fb4eSMark Brown 
76885e7652dSLars-Peter Clausen static const struct snd_soc_dai_ops wm8988_ops = {
7695409fb4eSMark Brown 	.startup = wm8988_pcm_startup,
7705409fb4eSMark Brown 	.hw_params = wm8988_pcm_hw_params,
7715409fb4eSMark Brown 	.set_fmt = wm8988_set_dai_fmt,
7725409fb4eSMark Brown 	.set_sysclk = wm8988_set_dai_sysclk,
7735409fb4eSMark Brown 	.digital_mute = wm8988_mute,
7745409fb4eSMark Brown };
7755409fb4eSMark Brown 
776f0fba2adSLiam Girdwood static struct snd_soc_dai_driver wm8988_dai = {
777f0fba2adSLiam Girdwood 	.name = "wm8988-hifi",
7785409fb4eSMark Brown 	.playback = {
7795409fb4eSMark Brown 		.stream_name = "Playback",
7805409fb4eSMark Brown 		.channels_min = 1,
7815409fb4eSMark Brown 		.channels_max = 2,
7825409fb4eSMark Brown 		.rates = WM8988_RATES,
7835409fb4eSMark Brown 		.formats = WM8988_FORMATS,
7845409fb4eSMark Brown 	},
7855409fb4eSMark Brown 	.capture = {
7865409fb4eSMark Brown 		.stream_name = "Capture",
7875409fb4eSMark Brown 		.channels_min = 1,
7885409fb4eSMark Brown 		.channels_max = 2,
7895409fb4eSMark Brown 		.rates = WM8988_RATES,
7905409fb4eSMark Brown 		.formats = WM8988_FORMATS,
7915409fb4eSMark Brown 	 },
7925409fb4eSMark Brown 	.ops = &wm8988_ops,
7935409fb4eSMark Brown 	.symmetric_rates = 1,
7945409fb4eSMark Brown };
7955409fb4eSMark Brown 
79684b315eeSLars-Peter Clausen static int wm8988_suspend(struct snd_soc_codec *codec)
7975409fb4eSMark Brown {
798d2dc0a77SMark Brown 	struct wm8988_priv *wm8988 = snd_soc_codec_get_drvdata(codec);
799d2dc0a77SMark Brown 
8005409fb4eSMark Brown 	wm8988_set_bias_level(codec, SND_SOC_BIAS_OFF);
801d2dc0a77SMark Brown 	regcache_mark_dirty(wm8988->regmap);
8025409fb4eSMark Brown 	return 0;
8035409fb4eSMark Brown }
8045409fb4eSMark Brown 
805f0fba2adSLiam Girdwood static int wm8988_resume(struct snd_soc_codec *codec)
8065409fb4eSMark Brown {
8075409fb4eSMark Brown 	wm8988_set_bias_level(codec, SND_SOC_BIAS_STANDBY);
8085409fb4eSMark Brown 	return 0;
8095409fb4eSMark Brown }
8105409fb4eSMark Brown 
811f0fba2adSLiam Girdwood static int wm8988_probe(struct snd_soc_codec *codec)
8125409fb4eSMark Brown {
8135409fb4eSMark Brown 	int ret = 0;
8145409fb4eSMark Brown 
8155409fb4eSMark Brown 	ret = wm8988_reset(codec);
8165409fb4eSMark Brown 	if (ret < 0) {
8175409fb4eSMark Brown 		dev_err(codec->dev, "Failed to issue reset\n");
818f0fba2adSLiam Girdwood 		return ret;
8195409fb4eSMark Brown 	}
8205409fb4eSMark Brown 
8215409fb4eSMark Brown 	/* set the update bits (we always update left then right) */
8229cd11326SAxel Lin 	snd_soc_update_bits(codec, WM8988_RADC, 0x0100, 0x0100);
8239cd11326SAxel Lin 	snd_soc_update_bits(codec, WM8988_RDAC, 0x0100, 0x0100);
8249cd11326SAxel Lin 	snd_soc_update_bits(codec, WM8988_ROUT1V, 0x0100, 0x0100);
8259cd11326SAxel Lin 	snd_soc_update_bits(codec, WM8988_ROUT2V, 0x0100, 0x0100);
8269cd11326SAxel Lin 	snd_soc_update_bits(codec, WM8988_RINVOL, 0x0100, 0x0100);
8275409fb4eSMark Brown 
828f0fba2adSLiam Girdwood 	wm8988_set_bias_level(codec, SND_SOC_BIAS_STANDBY);
8295409fb4eSMark Brown 
8305409fb4eSMark Brown 	return 0;
8315409fb4eSMark Brown }
8325409fb4eSMark Brown 
833f0fba2adSLiam Girdwood static int wm8988_remove(struct snd_soc_codec *codec)
8345409fb4eSMark Brown {
835f0fba2adSLiam Girdwood 	wm8988_set_bias_level(codec, SND_SOC_BIAS_OFF);
836f0fba2adSLiam Girdwood 	return 0;
8375409fb4eSMark Brown }
8385409fb4eSMark Brown 
839f0fba2adSLiam Girdwood static struct snd_soc_codec_driver soc_codec_dev_wm8988 = {
840f0fba2adSLiam Girdwood 	.probe =	wm8988_probe,
841f0fba2adSLiam Girdwood 	.remove =	wm8988_remove,
842f0fba2adSLiam Girdwood 	.suspend =	wm8988_suspend,
843f0fba2adSLiam Girdwood 	.resume =	wm8988_resume,
844f0fba2adSLiam Girdwood 	.set_bias_level = wm8988_set_bias_level,
845dd21353fSMark Brown 
846dd21353fSMark Brown 	.controls = wm8988_snd_controls,
847dd21353fSMark Brown 	.num_controls = ARRAY_SIZE(wm8988_snd_controls),
848dd21353fSMark Brown 	.dapm_widgets = wm8988_dapm_widgets,
849dd21353fSMark Brown 	.num_dapm_widgets = ARRAY_SIZE(wm8988_dapm_widgets),
850dd21353fSMark Brown 	.dapm_routes = wm8988_dapm_routes,
851dd21353fSMark Brown 	.num_dapm_routes = ARRAY_SIZE(wm8988_dapm_routes),
852f0fba2adSLiam Girdwood };
853f0fba2adSLiam Girdwood 
854d2dc0a77SMark Brown static struct regmap_config wm8988_regmap = {
855d2dc0a77SMark Brown 	.reg_bits = 7,
856d2dc0a77SMark Brown 	.val_bits = 9,
857d2dc0a77SMark Brown 
858d2dc0a77SMark Brown 	.max_register = WM8988_LPPB,
859d2dc0a77SMark Brown 	.writeable_reg = wm8988_writeable,
860d2dc0a77SMark Brown 
861d2dc0a77SMark Brown 	.cache_type = REGCACHE_RBTREE,
862d2dc0a77SMark Brown 	.reg_defaults = wm8988_reg_defaults,
863d2dc0a77SMark Brown 	.num_reg_defaults = ARRAY_SIZE(wm8988_reg_defaults),
864d2dc0a77SMark Brown };
865d2dc0a77SMark Brown 
866f0fba2adSLiam Girdwood #if defined(CONFIG_SPI_MASTER)
8677a79e94eSBill Pemberton static int wm8988_spi_probe(struct spi_device *spi)
8685409fb4eSMark Brown {
8695409fb4eSMark Brown 	struct wm8988_priv *wm8988;
870f0fba2adSLiam Girdwood 	int ret;
8715409fb4eSMark Brown 
87282fa3670SMark Brown 	wm8988 = devm_kzalloc(&spi->dev, sizeof(struct wm8988_priv),
87382fa3670SMark Brown 			      GFP_KERNEL);
8745409fb4eSMark Brown 	if (wm8988 == NULL)
8755409fb4eSMark Brown 		return -ENOMEM;
8765409fb4eSMark Brown 
87728517f5cSTushar Behera 	wm8988->regmap = devm_regmap_init_spi(spi, &wm8988_regmap);
878d2dc0a77SMark Brown 	if (IS_ERR(wm8988->regmap)) {
879d2dc0a77SMark Brown 		ret = PTR_ERR(wm8988->regmap);
880d2dc0a77SMark Brown 		dev_err(&spi->dev, "Failed to init regmap: %d\n", ret);
881d2dc0a77SMark Brown 		return ret;
882d2dc0a77SMark Brown 	}
883d2dc0a77SMark Brown 
884f0fba2adSLiam Girdwood 	spi_set_drvdata(spi, wm8988);
8855409fb4eSMark Brown 
886f0fba2adSLiam Girdwood 	ret = snd_soc_register_codec(&spi->dev,
887f0fba2adSLiam Girdwood 			&soc_codec_dev_wm8988, &wm8988_dai, 1);
888f0fba2adSLiam Girdwood 	return ret;
8895409fb4eSMark Brown }
8905409fb4eSMark Brown 
8917a79e94eSBill Pemberton static int wm8988_spi_remove(struct spi_device *spi)
8925409fb4eSMark Brown {
893f0fba2adSLiam Girdwood 	snd_soc_unregister_codec(&spi->dev);
894f0fba2adSLiam Girdwood 	return 0;
895f0fba2adSLiam Girdwood }
896f0fba2adSLiam Girdwood 
897f0fba2adSLiam Girdwood static struct spi_driver wm8988_spi_driver = {
898f0fba2adSLiam Girdwood 	.driver = {
899091edccfSMark Brown 		.name	= "wm8988",
900f0fba2adSLiam Girdwood 		.owner	= THIS_MODULE,
901f0fba2adSLiam Girdwood 	},
902f0fba2adSLiam Girdwood 	.probe		= wm8988_spi_probe,
9037a79e94eSBill Pemberton 	.remove		= wm8988_spi_remove,
904f0fba2adSLiam Girdwood };
905f0fba2adSLiam Girdwood #endif /* CONFIG_SPI_MASTER */
906f0fba2adSLiam Girdwood 
90763587116SFabio Estevam #if IS_ENABLED(CONFIG_I2C)
9087a79e94eSBill Pemberton static int wm8988_i2c_probe(struct i2c_client *i2c,
909f0fba2adSLiam Girdwood 			    const struct i2c_device_id *id)
910f0fba2adSLiam Girdwood {
911f0fba2adSLiam Girdwood 	struct wm8988_priv *wm8988;
912f0fba2adSLiam Girdwood 	int ret;
913f0fba2adSLiam Girdwood 
91482fa3670SMark Brown 	wm8988 = devm_kzalloc(&i2c->dev, sizeof(struct wm8988_priv),
91582fa3670SMark Brown 			      GFP_KERNEL);
916f0fba2adSLiam Girdwood 	if (wm8988 == NULL)
917f0fba2adSLiam Girdwood 		return -ENOMEM;
918f0fba2adSLiam Girdwood 
919f0fba2adSLiam Girdwood 	i2c_set_clientdata(i2c, wm8988);
920d2dc0a77SMark Brown 
92128517f5cSTushar Behera 	wm8988->regmap = devm_regmap_init_i2c(i2c, &wm8988_regmap);
922d2dc0a77SMark Brown 	if (IS_ERR(wm8988->regmap)) {
923d2dc0a77SMark Brown 		ret = PTR_ERR(wm8988->regmap);
924d2dc0a77SMark Brown 		dev_err(&i2c->dev, "Failed to init regmap: %d\n", ret);
925d2dc0a77SMark Brown 		return ret;
926d2dc0a77SMark Brown 	}
927f0fba2adSLiam Girdwood 
928f0fba2adSLiam Girdwood 	ret =  snd_soc_register_codec(&i2c->dev,
929f0fba2adSLiam Girdwood 			&soc_codec_dev_wm8988, &wm8988_dai, 1);
930f0fba2adSLiam Girdwood 	return ret;
931f0fba2adSLiam Girdwood }
932f0fba2adSLiam Girdwood 
9337a79e94eSBill Pemberton static int wm8988_i2c_remove(struct i2c_client *client)
934f0fba2adSLiam Girdwood {
935f0fba2adSLiam Girdwood 	snd_soc_unregister_codec(&client->dev);
9365409fb4eSMark Brown 	return 0;
9375409fb4eSMark Brown }
9385409fb4eSMark Brown 
9395409fb4eSMark Brown static const struct i2c_device_id wm8988_i2c_id[] = {
9405409fb4eSMark Brown 	{ "wm8988", 0 },
9415409fb4eSMark Brown 	{ }
9425409fb4eSMark Brown };
9435409fb4eSMark Brown MODULE_DEVICE_TABLE(i2c, wm8988_i2c_id);
9445409fb4eSMark Brown 
9455409fb4eSMark Brown static struct i2c_driver wm8988_i2c_driver = {
9465409fb4eSMark Brown 	.driver = {
9471f5cafb2SMark Brown 		.name = "wm8988",
9485409fb4eSMark Brown 		.owner = THIS_MODULE,
9495409fb4eSMark Brown 	},
9505409fb4eSMark Brown 	.probe =    wm8988_i2c_probe,
9517a79e94eSBill Pemberton 	.remove =   wm8988_i2c_remove,
9525409fb4eSMark Brown 	.id_table = wm8988_i2c_id,
9535409fb4eSMark Brown };
9545409fb4eSMark Brown #endif
9555409fb4eSMark Brown 
9565409fb4eSMark Brown static int __init wm8988_modinit(void)
9575409fb4eSMark Brown {
958f0fba2adSLiam Girdwood 	int ret = 0;
95963587116SFabio Estevam #if IS_ENABLED(CONFIG_I2C)
9605409fb4eSMark Brown 	ret = i2c_add_driver(&wm8988_i2c_driver);
961f0fba2adSLiam Girdwood 	if (ret != 0) {
962f0fba2adSLiam Girdwood 		printk(KERN_ERR "Failed to register WM8988 I2C driver: %d\n",
963f0fba2adSLiam Girdwood 		       ret);
964f0fba2adSLiam Girdwood 	}
9655409fb4eSMark Brown #endif
9665409fb4eSMark Brown #if defined(CONFIG_SPI_MASTER)
9675409fb4eSMark Brown 	ret = spi_register_driver(&wm8988_spi_driver);
968f0fba2adSLiam Girdwood 	if (ret != 0) {
969f0fba2adSLiam Girdwood 		printk(KERN_ERR "Failed to register WM8988 SPI driver: %d\n",
970f0fba2adSLiam Girdwood 		       ret);
971f0fba2adSLiam Girdwood 	}
9725409fb4eSMark Brown #endif
9735409fb4eSMark Brown 	return ret;
9745409fb4eSMark Brown }
9755409fb4eSMark Brown module_init(wm8988_modinit);
9765409fb4eSMark Brown 
9775409fb4eSMark Brown static void __exit wm8988_exit(void)
9785409fb4eSMark Brown {
97963587116SFabio Estevam #if IS_ENABLED(CONFIG_I2C)
9805409fb4eSMark Brown 	i2c_del_driver(&wm8988_i2c_driver);
9815409fb4eSMark Brown #endif
9825409fb4eSMark Brown #if defined(CONFIG_SPI_MASTER)
9835409fb4eSMark Brown 	spi_unregister_driver(&wm8988_spi_driver);
9845409fb4eSMark Brown #endif
9855409fb4eSMark Brown }
9865409fb4eSMark Brown module_exit(wm8988_exit);
9875409fb4eSMark Brown 
9885409fb4eSMark Brown 
9895409fb4eSMark Brown MODULE_DESCRIPTION("ASoC WM8988 driver");
9905409fb4eSMark Brown MODULE_AUTHOR("Mark Brown <broonie@opensource.wolfsonmicro.com>");
9915409fb4eSMark Brown MODULE_LICENSE("GPL");
992