xref: /openbmc/linux/sound/pci/hda/patch_via.c (revision dda415d41882449f841f88d829dd65b6ee1c374c)
1c577b8a1SJoseph Chan /*
2c577b8a1SJoseph Chan  * Universal Interface for Intel High Definition Audio Codec
3c577b8a1SJoseph Chan  *
48e86597fSLydia Wang  * HD audio interface patch for VIA VT17xx/VT18xx/VT20xx codec
5c577b8a1SJoseph Chan  *
68e86597fSLydia Wang  *  (C) 2006-2009 VIA Technology, Inc.
78e86597fSLydia Wang  *  (C) 2006-2008 Takashi Iwai <tiwai@suse.de>
8c577b8a1SJoseph Chan  *
9c577b8a1SJoseph Chan  *  This driver is free software; you can redistribute it and/or modify
10c577b8a1SJoseph Chan  *  it under the terms of the GNU General Public License as published by
11c577b8a1SJoseph Chan  *  the Free Software Foundation; either version 2 of the License, or
12c577b8a1SJoseph Chan  *  (at your option) any later version.
13c577b8a1SJoseph Chan  *
14c577b8a1SJoseph Chan  *  This driver is distributed in the hope that it will be useful,
15c577b8a1SJoseph Chan  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
16c577b8a1SJoseph Chan  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17c577b8a1SJoseph Chan  *  GNU General Public License for more details.
18c577b8a1SJoseph Chan  *
19c577b8a1SJoseph Chan  *  You should have received a copy of the GNU General Public License
20c577b8a1SJoseph Chan  *  along with this program; if not, write to the Free Software
21c577b8a1SJoseph Chan  *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA
22c577b8a1SJoseph Chan  */
23c577b8a1SJoseph Chan 
24c577b8a1SJoseph Chan /* * * * * * * * * * * * * * Release History * * * * * * * * * * * * * * * * */
25c577b8a1SJoseph Chan /*									     */
26c577b8a1SJoseph Chan /* 2006-03-03  Lydia Wang  Create the basic patch to support VT1708 codec    */
27c577b8a1SJoseph Chan /* 2006-03-14  Lydia Wang  Modify hard code for some pin widget nid	     */
28c577b8a1SJoseph Chan /* 2006-08-02  Lydia Wang  Add support to VT1709 codec			     */
29c577b8a1SJoseph Chan /* 2006-09-08  Lydia Wang  Fix internal loopback recording source select bug */
30f7278fd0SJosepch Chan /* 2007-09-12  Lydia Wang  Add EAPD enable during driver initialization	     */
31f7278fd0SJosepch Chan /* 2007-09-17  Lydia Wang  Add VT1708B codec support			    */
3276d9b0ddSHarald Welte /* 2007-11-14  Lydia Wang  Add VT1708A codec HP and CD pin connect config    */
33fb4cb772SHarald Welte /* 2008-02-03  Lydia Wang  Fix Rear channels and Back channels inverse issue */
34d949cac1SHarald Welte /* 2008-03-06  Lydia Wang  Add VT1702 codec and VT1708S codec support	     */
3569e52a80SHarald Welte /* 2008-04-09  Lydia Wang  Add mute front speaker when HP plugin	     */
360aa62aefSHarald Welte /* 2008-04-09  Lydia Wang  Add Independent HP feature			     */
3798aa34c0SHarald Welte /* 2008-05-28  Lydia Wang  Add second S/PDIF Out support for VT1702	     */
38d7426329SHarald Welte /* 2008-09-15  Logan Li	   Add VT1708S Mic Boost workaround/backdoor	     */
398e86597fSLydia Wang /* 2009-02-16  Logan Li	   Add support for VT1718S			     */
408e86597fSLydia Wang /* 2009-03-13  Logan Li	   Add support for VT1716S			     */
418e86597fSLydia Wang /* 2009-04-14  Lydai Wang  Add support for VT1828S and VT2020		     */
428e86597fSLydia Wang /* 2009-07-08  Lydia Wang  Add support for VT2002P			     */
438e86597fSLydia Wang /* 2009-07-21  Lydia Wang  Add support for VT1812			     */
4436dd5c4aSLydia Wang /* 2009-09-19  Lydia Wang  Add support for VT1818S			     */
45c577b8a1SJoseph Chan /*									     */
46c577b8a1SJoseph Chan /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
47c577b8a1SJoseph Chan 
48c577b8a1SJoseph Chan 
49c577b8a1SJoseph Chan #include <linux/init.h>
50c577b8a1SJoseph Chan #include <linux/delay.h>
51c577b8a1SJoseph Chan #include <linux/slab.h>
52da155d5bSPaul Gortmaker #include <linux/module.h>
53c577b8a1SJoseph Chan #include <sound/core.h>
540aa62aefSHarald Welte #include <sound/asoundef.h>
55c577b8a1SJoseph Chan #include "hda_codec.h"
56c577b8a1SJoseph Chan #include "hda_local.h"
57128bc4baSTakashi Iwai #include "hda_auto_parser.h"
581835a0f9STakashi Iwai #include "hda_jack.h"
59c577b8a1SJoseph Chan 
60c577b8a1SJoseph Chan /* Pin Widget NID */
6176d9b0ddSHarald Welte #define VT1708_HP_PIN_NID	0x20
6276d9b0ddSHarald Welte #define VT1708_CD_PIN_NID	0x24
63c577b8a1SJoseph Chan 
64d7426329SHarald Welte enum VIA_HDA_CODEC {
65d7426329SHarald Welte 	UNKNOWN = -1,
66d7426329SHarald Welte 	VT1708,
67d7426329SHarald Welte 	VT1709_10CH,
68d7426329SHarald Welte 	VT1709_6CH,
69d7426329SHarald Welte 	VT1708B_8CH,
70d7426329SHarald Welte 	VT1708B_4CH,
71d7426329SHarald Welte 	VT1708S,
72518bf3baSLydia Wang 	VT1708BCE,
73d7426329SHarald Welte 	VT1702,
74eb7188caSLydia Wang 	VT1718S,
75f3db423dSLydia Wang 	VT1716S,
7625eaba2fSLydia Wang 	VT2002P,
77ab6734e7SLydia Wang 	VT1812,
7811890956SLydia Wang 	VT1802,
79d7426329SHarald Welte 	CODEC_TYPES,
80d7426329SHarald Welte };
81d7426329SHarald Welte 
8211890956SLydia Wang #define VT2002P_COMPATIBLE(spec) \
8311890956SLydia Wang 	((spec)->codec_type == VT2002P ||\
8411890956SLydia Wang 	 (spec)->codec_type == VT1812 ||\
8511890956SLydia Wang 	 (spec)->codec_type == VT1802)
8611890956SLydia Wang 
878e3679dcSTakashi Iwai #define MAX_NID_PATH_DEPTH	5
888e3679dcSTakashi Iwai 
8909a9ad69STakashi Iwai /* output-path: DAC -> ... -> pin
9009a9ad69STakashi Iwai  * idx[] contains the source index number of the next widget;
9109a9ad69STakashi Iwai  * e.g. idx[0] is the index of the DAC selected by path[1] widget
9209a9ad69STakashi Iwai  * multi[] indicates whether it's a selector widget with multi-connectors
9309a9ad69STakashi Iwai  * (i.e. the connection selection is mandatory)
9409a9ad69STakashi Iwai  * vol_ctl and mute_ctl contains the NIDs for the assigned mixers
9509a9ad69STakashi Iwai  */
964a79616dSTakashi Iwai struct nid_path {
974a79616dSTakashi Iwai 	int depth;
988e3679dcSTakashi Iwai 	hda_nid_t path[MAX_NID_PATH_DEPTH];
9909a9ad69STakashi Iwai 	unsigned char idx[MAX_NID_PATH_DEPTH];
10009a9ad69STakashi Iwai 	unsigned char multi[MAX_NID_PATH_DEPTH];
10109a9ad69STakashi Iwai 	unsigned int vol_ctl;
10209a9ad69STakashi Iwai 	unsigned int mute_ctl;
1034a79616dSTakashi Iwai };
1044a79616dSTakashi Iwai 
105a86a88eaSTakashi Iwai /* input-path */
106a86a88eaSTakashi Iwai struct via_input {
107a86a88eaSTakashi Iwai 	hda_nid_t pin;	/* input-pin or aa-mix */
108a86a88eaSTakashi Iwai 	int adc_idx;	/* ADC index to be used */
109a86a88eaSTakashi Iwai 	int mux_idx;	/* MUX index (if any) */
110a86a88eaSTakashi Iwai 	const char *label;	/* input-source label */
111a86a88eaSTakashi Iwai };
112a86a88eaSTakashi Iwai 
113de6c74f3STakashi Iwai #define VIA_MAX_ADCS	3
114de6c74f3STakashi Iwai 
1153b607e3dSTakashi Iwai enum {
1163b607e3dSTakashi Iwai 	STREAM_MULTI_OUT = (1 << 0),
1173b607e3dSTakashi Iwai 	STREAM_INDEP_HP = (1 << 1),
1183b607e3dSTakashi Iwai };
1193b607e3dSTakashi Iwai 
1201f2e99feSLydia Wang struct via_spec {
1217819d1c7STakashi Iwai 	struct hda_gen_spec gen;
1227819d1c7STakashi Iwai 
1231f2e99feSLydia Wang 	/* codec parameterization */
12490dd48a1STakashi Iwai 	const struct snd_kcontrol_new *mixers[6];
1251f2e99feSLydia Wang 	unsigned int num_mixers;
1261f2e99feSLydia Wang 
12790dd48a1STakashi Iwai 	const struct hda_verb *init_verbs[5];
1281f2e99feSLydia Wang 	unsigned int num_iverbs;
1291f2e99feSLydia Wang 
13082673bc8STakashi Iwai 	char stream_name_analog[32];
1317eb56e84STakashi Iwai 	char stream_name_hp[32];
13290dd48a1STakashi Iwai 	const struct hda_pcm_stream *stream_analog_playback;
13390dd48a1STakashi Iwai 	const struct hda_pcm_stream *stream_analog_capture;
1341f2e99feSLydia Wang 
13582673bc8STakashi Iwai 	char stream_name_digital[32];
13690dd48a1STakashi Iwai 	const struct hda_pcm_stream *stream_digital_playback;
13790dd48a1STakashi Iwai 	const struct hda_pcm_stream *stream_digital_capture;
1381f2e99feSLydia Wang 
1391f2e99feSLydia Wang 	/* playback */
1401f2e99feSLydia Wang 	struct hda_multi_out multiout;
1411f2e99feSLydia Wang 	hda_nid_t slave_dig_outs[2];
142ece8d043STakashi Iwai 	hda_nid_t hp_dac_nid;
1433214b966STakashi Iwai 	hda_nid_t speaker_dac_nid;
1443214b966STakashi Iwai 	int hp_indep_shared;	/* indep HP-DAC is shared with side ch */
1453b607e3dSTakashi Iwai 	int opened_streams;	/* STREAM_* bits */
1463b607e3dSTakashi Iwai 	int active_streams;	/* STREAM_* bits */
1473214b966STakashi Iwai 	int aamix_mode;		/* loopback is enabled for output-path? */
1481f2e99feSLydia Wang 
1493214b966STakashi Iwai 	/* Output-paths:
1503214b966STakashi Iwai 	 * There are different output-paths depending on the setup.
1513214b966STakashi Iwai 	 * out_path, hp_path and speaker_path are primary paths.  If both
1523214b966STakashi Iwai 	 * direct DAC and aa-loopback routes are available, these contain
1533214b966STakashi Iwai 	 * the former paths.  Meanwhile *_mix_path contain the paths with
1543214b966STakashi Iwai 	 * loopback mixer.  (Since the loopback is only for front channel,
1553214b966STakashi Iwai 	 * no out_mix_path for surround channels.)
1563214b966STakashi Iwai 	 * The HP output has another path, hp_indep_path, which is used in
1573214b966STakashi Iwai 	 * the independent-HP mode.
1583214b966STakashi Iwai 	 */
159de6c74f3STakashi Iwai 	struct nid_path out_path[HDA_SIDE + 1];
1603214b966STakashi Iwai 	struct nid_path out_mix_path;
1614a79616dSTakashi Iwai 	struct nid_path hp_path;
1623214b966STakashi Iwai 	struct nid_path hp_mix_path;
1633214b966STakashi Iwai 	struct nid_path hp_indep_path;
1644a918ffeSTakashi Iwai 	struct nid_path speaker_path;
1653214b966STakashi Iwai 	struct nid_path speaker_mix_path;
1664a79616dSTakashi Iwai 
1671f2e99feSLydia Wang 	/* capture */
1681f2e99feSLydia Wang 	unsigned int num_adc_nids;
169de6c74f3STakashi Iwai 	hda_nid_t adc_nids[VIA_MAX_ADCS];
170de6c74f3STakashi Iwai 	hda_nid_t mux_nids[VIA_MAX_ADCS];
171620e2b28STakashi Iwai 	hda_nid_t aa_mix_nid;
1721f2e99feSLydia Wang 	hda_nid_t dig_in_nid;
1731f2e99feSLydia Wang 
1741f2e99feSLydia Wang 	/* capture source */
175a86a88eaSTakashi Iwai 	bool dyn_adc_switch;
176a86a88eaSTakashi Iwai 	int num_inputs;
177a86a88eaSTakashi Iwai 	struct via_input inputs[AUTO_CFG_MAX_INS + 1];
178de6c74f3STakashi Iwai 	unsigned int cur_mux[VIA_MAX_ADCS];
1791f2e99feSLydia Wang 
1803b607e3dSTakashi Iwai 	/* dynamic DAC switching */
1813b607e3dSTakashi Iwai 	unsigned int cur_dac_stream_tag;
1823b607e3dSTakashi Iwai 	unsigned int cur_dac_format;
1833b607e3dSTakashi Iwai 	unsigned int cur_hp_stream_tag;
1843b607e3dSTakashi Iwai 	unsigned int cur_hp_format;
1853b607e3dSTakashi Iwai 
186a86a88eaSTakashi Iwai 	/* dynamic ADC switching */
187a86a88eaSTakashi Iwai 	hda_nid_t cur_adc;
188a86a88eaSTakashi Iwai 	unsigned int cur_adc_stream_tag;
189a86a88eaSTakashi Iwai 	unsigned int cur_adc_format;
190a86a88eaSTakashi Iwai 
1911f2e99feSLydia Wang 	/* PCM information */
1921f2e99feSLydia Wang 	struct hda_pcm pcm_rec[3];
1931f2e99feSLydia Wang 
1941f2e99feSLydia Wang 	/* dynamic controls, init_verbs and input_mux */
1951f2e99feSLydia Wang 	struct auto_pin_cfg autocfg;
1961f2e99feSLydia Wang 	struct snd_array kctls;
1971f2e99feSLydia Wang 	hda_nid_t private_dac_nids[AUTO_CFG_MAX_OUTS];
1981f2e99feSLydia Wang 
1991f2e99feSLydia Wang 	/* HP mode source */
2001f2e99feSLydia Wang 	unsigned int hp_independent_mode;
201f3db423dSLydia Wang 	unsigned int dmic_enabled;
20224088a58STakashi Iwai 	unsigned int no_pin_power_ctl;
2031f2e99feSLydia Wang 	enum VIA_HDA_CODEC codec_type;
2041f2e99feSLydia Wang 
205e9d010c2STakashi Iwai 	/* analog low-power control */
206e9d010c2STakashi Iwai 	bool alc_mode;
207e9d010c2STakashi Iwai 
208e3d7a143STakashi Iwai 	/* smart51 setup */
209e3d7a143STakashi Iwai 	unsigned int smart51_nums;
210e3d7a143STakashi Iwai 	hda_nid_t smart51_pins[2];
211e3d7a143STakashi Iwai 	int smart51_idxs[2];
212e3d7a143STakashi Iwai 	const char *smart51_labels[2];
213e3d7a143STakashi Iwai 	unsigned int smart51_enabled;
214e3d7a143STakashi Iwai 
2151f2e99feSLydia Wang 	/* work to check hp jack state */
2161f2e99feSLydia Wang 	struct hda_codec *codec;
2171f2e99feSLydia Wang 	struct delayed_work vt1708_hp_work;
218187d333eSTakashi Iwai 	int hp_work_active;
219e06e5a29STakashi Iwai 	int vt1708_jack_detect;
2201f2e99feSLydia Wang 	int vt1708_hp_present;
2213e95b9abSLydia Wang 
2223e95b9abSLydia Wang 	void (*set_widgets_power_state)(struct hda_codec *codec);
2233e95b9abSLydia Wang 
2241f2e99feSLydia Wang 	struct hda_loopback_check loopback;
22513af8e77STakashi Iwai 	int num_loopbacks;
22613af8e77STakashi Iwai 	struct hda_amp_list loopback_list[8];
227a86a88eaSTakashi Iwai 
228a86a88eaSTakashi Iwai 	/* bind capture-volume */
229a86a88eaSTakashi Iwai 	struct hda_bind_ctls *bind_cap_vol;
230a86a88eaSTakashi Iwai 	struct hda_bind_ctls *bind_cap_sw;
2313b607e3dSTakashi Iwai 
2323b607e3dSTakashi Iwai 	struct mutex config_mutex;
2331f2e99feSLydia Wang };
2341f2e99feSLydia Wang 
2350341ccd7SLydia Wang static enum VIA_HDA_CODEC get_codec_type(struct hda_codec *codec);
2365b0cb1d8SJaroslav Kysela static struct via_spec * via_new_spec(struct hda_codec *codec)
2375b0cb1d8SJaroslav Kysela {
2385b0cb1d8SJaroslav Kysela 	struct via_spec *spec;
2395b0cb1d8SJaroslav Kysela 
2405b0cb1d8SJaroslav Kysela 	spec = kzalloc(sizeof(*spec), GFP_KERNEL);
2415b0cb1d8SJaroslav Kysela 	if (spec == NULL)
2425b0cb1d8SJaroslav Kysela 		return NULL;
2435b0cb1d8SJaroslav Kysela 
244361dab3eSTakashi Iwai 	snd_array_init(&spec->kctls, sizeof(struct snd_kcontrol_new), 32);
2453b607e3dSTakashi Iwai 	mutex_init(&spec->config_mutex);
2465b0cb1d8SJaroslav Kysela 	codec->spec = spec;
2475b0cb1d8SJaroslav Kysela 	spec->codec = codec;
2480341ccd7SLydia Wang 	spec->codec_type = get_codec_type(codec);
2490341ccd7SLydia Wang 	/* VT1708BCE & VT1708S are almost same */
2500341ccd7SLydia Wang 	if (spec->codec_type == VT1708BCE)
2510341ccd7SLydia Wang 		spec->codec_type = VT1708S;
2527819d1c7STakashi Iwai 	snd_hda_gen_init(&spec->gen);
2535b0cb1d8SJaroslav Kysela 	return spec;
2545b0cb1d8SJaroslav Kysela }
2555b0cb1d8SJaroslav Kysela 
256744ff5f4SLydia Wang static enum VIA_HDA_CODEC get_codec_type(struct hda_codec *codec)
257d7426329SHarald Welte {
258744ff5f4SLydia Wang 	u32 vendor_id = codec->vendor_id;
259d7426329SHarald Welte 	u16 ven_id = vendor_id >> 16;
260d7426329SHarald Welte 	u16 dev_id = vendor_id & 0xffff;
261d7426329SHarald Welte 	enum VIA_HDA_CODEC codec_type;
262d7426329SHarald Welte 
263d7426329SHarald Welte 	/* get codec type */
264d7426329SHarald Welte 	if (ven_id != 0x1106)
265d7426329SHarald Welte 		codec_type = UNKNOWN;
266d7426329SHarald Welte 	else if (dev_id >= 0x1708 && dev_id <= 0x170b)
267d7426329SHarald Welte 		codec_type = VT1708;
268d7426329SHarald Welte 	else if (dev_id >= 0xe710 && dev_id <= 0xe713)
269d7426329SHarald Welte 		codec_type = VT1709_10CH;
270d7426329SHarald Welte 	else if (dev_id >= 0xe714 && dev_id <= 0xe717)
271d7426329SHarald Welte 		codec_type = VT1709_6CH;
272518bf3baSLydia Wang 	else if (dev_id >= 0xe720 && dev_id <= 0xe723) {
273d7426329SHarald Welte 		codec_type = VT1708B_8CH;
274518bf3baSLydia Wang 		if (snd_hda_param_read(codec, 0x16, AC_PAR_CONNLIST_LEN) == 0x7)
275518bf3baSLydia Wang 			codec_type = VT1708BCE;
276518bf3baSLydia Wang 	} else if (dev_id >= 0xe724 && dev_id <= 0xe727)
277d7426329SHarald Welte 		codec_type = VT1708B_4CH;
278d7426329SHarald Welte 	else if ((dev_id & 0xfff) == 0x397
279d7426329SHarald Welte 		 && (dev_id >> 12) < 8)
280d7426329SHarald Welte 		codec_type = VT1708S;
281d7426329SHarald Welte 	else if ((dev_id & 0xfff) == 0x398
282d7426329SHarald Welte 		 && (dev_id >> 12) < 8)
283d7426329SHarald Welte 		codec_type = VT1702;
284eb7188caSLydia Wang 	else if ((dev_id & 0xfff) == 0x428
285eb7188caSLydia Wang 		 && (dev_id >> 12) < 8)
286eb7188caSLydia Wang 		codec_type = VT1718S;
287f3db423dSLydia Wang 	else if (dev_id == 0x0433 || dev_id == 0xa721)
288f3db423dSLydia Wang 		codec_type = VT1716S;
289bb3c6bfcSLydia Wang 	else if (dev_id == 0x0441 || dev_id == 0x4441)
290bb3c6bfcSLydia Wang 		codec_type = VT1718S;
29125eaba2fSLydia Wang 	else if (dev_id == 0x0438 || dev_id == 0x4438)
29225eaba2fSLydia Wang 		codec_type = VT2002P;
293ab6734e7SLydia Wang 	else if (dev_id == 0x0448)
294ab6734e7SLydia Wang 		codec_type = VT1812;
29536dd5c4aSLydia Wang 	else if (dev_id == 0x0440)
29636dd5c4aSLydia Wang 		codec_type = VT1708S;
29711890956SLydia Wang 	else if ((dev_id & 0xfff) == 0x446)
29811890956SLydia Wang 		codec_type = VT1802;
299d7426329SHarald Welte 	else
300d7426329SHarald Welte 		codec_type = UNKNOWN;
301d7426329SHarald Welte 	return codec_type;
302d7426329SHarald Welte };
303d7426329SHarald Welte 
304ec7e7e42SLydia Wang #define VIA_JACK_EVENT		0x20
30569e52a80SHarald Welte #define VIA_HP_EVENT		0x01
3064a918ffeSTakashi Iwai #define VIA_LINE_EVENT		0x03
30769e52a80SHarald Welte 
308c577b8a1SJoseph Chan enum {
309c577b8a1SJoseph Chan 	VIA_CTL_WIDGET_VOL,
310c577b8a1SJoseph Chan 	VIA_CTL_WIDGET_MUTE,
311f5271101SLydia Wang 	VIA_CTL_WIDGET_ANALOG_MUTE,
312c577b8a1SJoseph Chan };
313c577b8a1SJoseph Chan 
314ada509ecSTakashi Iwai static void analog_low_current_mode(struct hda_codec *codec);
315ada509ecSTakashi Iwai static bool is_aa_path_mute(struct hda_codec *codec);
3161f2e99feSLydia Wang 
317187d333eSTakashi Iwai #define hp_detect_with_aa(codec) \
318187d333eSTakashi Iwai 	(snd_hda_get_bool_hint(codec, "analog_loopback_hp_detect") == 1 && \
319187d333eSTakashi Iwai 	 !is_aa_path_mute(codec))
3201f2e99feSLydia Wang 
3211f2e99feSLydia Wang static void vt1708_stop_hp_work(struct via_spec *spec)
3221f2e99feSLydia Wang {
3231f2e99feSLydia Wang 	if (spec->codec_type != VT1708 || spec->autocfg.hp_pins[0] == 0)
3241f2e99feSLydia Wang 		return;
325187d333eSTakashi Iwai 	if (spec->hp_work_active) {
326187d333eSTakashi Iwai 		snd_hda_codec_write(spec->codec, 0x1, 0, 0xf81, 1);
3275b84ba26STejun Heo 		cancel_delayed_work_sync(&spec->vt1708_hp_work);
328187d333eSTakashi Iwai 		spec->hp_work_active = 0;
329187d333eSTakashi Iwai 	}
330187d333eSTakashi Iwai }
331187d333eSTakashi Iwai 
332187d333eSTakashi Iwai static void vt1708_update_hp_work(struct via_spec *spec)
333187d333eSTakashi Iwai {
334187d333eSTakashi Iwai 	if (spec->codec_type != VT1708 || spec->autocfg.hp_pins[0] == 0)
335187d333eSTakashi Iwai 		return;
336187d333eSTakashi Iwai 	if (spec->vt1708_jack_detect &&
337187d333eSTakashi Iwai 	    (spec->active_streams || hp_detect_with_aa(spec->codec))) {
338187d333eSTakashi Iwai 		if (!spec->hp_work_active) {
339187d333eSTakashi Iwai 			snd_hda_codec_write(spec->codec, 0x1, 0, 0xf81, 0);
340187d333eSTakashi Iwai 			schedule_delayed_work(&spec->vt1708_hp_work,
341187d333eSTakashi Iwai 					      msecs_to_jiffies(100));
342187d333eSTakashi Iwai 			spec->hp_work_active = 1;
343187d333eSTakashi Iwai 		}
344187d333eSTakashi Iwai 	} else if (!hp_detect_with_aa(spec->codec))
345187d333eSTakashi Iwai 		vt1708_stop_hp_work(spec);
3461f2e99feSLydia Wang }
347f5271101SLydia Wang 
3483e95b9abSLydia Wang static void set_widgets_power_state(struct hda_codec *codec)
3493e95b9abSLydia Wang {
3503e95b9abSLydia Wang 	struct via_spec *spec = codec->spec;
3513e95b9abSLydia Wang 	if (spec->set_widgets_power_state)
3523e95b9abSLydia Wang 		spec->set_widgets_power_state(codec);
3533e95b9abSLydia Wang }
35425eaba2fSLydia Wang 
355f5271101SLydia Wang static int analog_input_switch_put(struct snd_kcontrol *kcontrol,
356f5271101SLydia Wang 				   struct snd_ctl_elem_value *ucontrol)
357f5271101SLydia Wang {
358f5271101SLydia Wang 	int change = snd_hda_mixer_amp_switch_put(kcontrol, ucontrol);
359f5271101SLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
360f5271101SLydia Wang 
3613e95b9abSLydia Wang 	set_widgets_power_state(codec);
362ada509ecSTakashi Iwai 	analog_low_current_mode(snd_kcontrol_chip(kcontrol));
363187d333eSTakashi Iwai 	vt1708_update_hp_work(codec->spec);
364f5271101SLydia Wang 	return change;
365f5271101SLydia Wang }
366f5271101SLydia Wang 
367f5271101SLydia Wang /* modify .put = snd_hda_mixer_amp_switch_put */
368f5271101SLydia Wang #define ANALOG_INPUT_MUTE						\
369f5271101SLydia Wang 	{		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,		\
370f5271101SLydia Wang 			.name = NULL,					\
371f5271101SLydia Wang 			.index = 0,					\
372f5271101SLydia Wang 			.info = snd_hda_mixer_amp_switch_info,		\
373f5271101SLydia Wang 			.get = snd_hda_mixer_amp_switch_get,		\
374f5271101SLydia Wang 			.put = analog_input_switch_put,			\
375f5271101SLydia Wang 			.private_value = HDA_COMPOSE_AMP_VAL(0, 3, 0, 0) }
376f5271101SLydia Wang 
37790dd48a1STakashi Iwai static const struct snd_kcontrol_new via_control_templates[] = {
378c577b8a1SJoseph Chan 	HDA_CODEC_VOLUME(NULL, 0, 0, 0),
379c577b8a1SJoseph Chan 	HDA_CODEC_MUTE(NULL, 0, 0, 0),
380f5271101SLydia Wang 	ANALOG_INPUT_MUTE,
381c577b8a1SJoseph Chan };
382c577b8a1SJoseph Chan 
383ab6734e7SLydia Wang 
384c577b8a1SJoseph Chan /* add dynamic controls */
385291c9e33STakashi Iwai static struct snd_kcontrol_new *__via_clone_ctl(struct via_spec *spec,
386291c9e33STakashi Iwai 				const struct snd_kcontrol_new *tmpl,
387291c9e33STakashi Iwai 				const char *name)
388c577b8a1SJoseph Chan {
389c577b8a1SJoseph Chan 	struct snd_kcontrol_new *knew;
390c577b8a1SJoseph Chan 
391603c4019STakashi Iwai 	knew = snd_array_new(&spec->kctls);
392c577b8a1SJoseph Chan 	if (!knew)
393291c9e33STakashi Iwai 		return NULL;
394291c9e33STakashi Iwai 	*knew = *tmpl;
395291c9e33STakashi Iwai 	if (!name)
396291c9e33STakashi Iwai 		name = tmpl->name;
397291c9e33STakashi Iwai 	if (name) {
398c577b8a1SJoseph Chan 		knew->name = kstrdup(name, GFP_KERNEL);
399c577b8a1SJoseph Chan 		if (!knew->name)
400291c9e33STakashi Iwai 			return NULL;
401291c9e33STakashi Iwai 	}
402291c9e33STakashi Iwai 	return knew;
403291c9e33STakashi Iwai }
404291c9e33STakashi Iwai 
405291c9e33STakashi Iwai static int __via_add_control(struct via_spec *spec, int type, const char *name,
406291c9e33STakashi Iwai 			     int idx, unsigned long val)
407291c9e33STakashi Iwai {
408291c9e33STakashi Iwai 	struct snd_kcontrol_new *knew;
409291c9e33STakashi Iwai 
410291c9e33STakashi Iwai 	knew = __via_clone_ctl(spec, &via_control_templates[type], name);
411291c9e33STakashi Iwai 	if (!knew)
412c577b8a1SJoseph Chan 		return -ENOMEM;
413d7a99cceSTakashi Iwai 	knew->index = idx;
4144d02d1b6SJaroslav Kysela 	if (get_amp_nid_(val))
4155e26dfd0SJaroslav Kysela 		knew->subdevice = HDA_SUBDEV_AMP_FLAG;
416c577b8a1SJoseph Chan 	knew->private_value = val;
417c577b8a1SJoseph Chan 	return 0;
418c577b8a1SJoseph Chan }
419c577b8a1SJoseph Chan 
4207b315bb4STakashi Iwai #define via_add_control(spec, type, name, val) \
4217b315bb4STakashi Iwai 	__via_add_control(spec, type, name, 0, val)
4227b315bb4STakashi Iwai 
423291c9e33STakashi Iwai #define via_clone_control(spec, tmpl) __via_clone_ctl(spec, tmpl, NULL)
4245b0cb1d8SJaroslav Kysela 
425603c4019STakashi Iwai static void via_free_kctls(struct hda_codec *codec)
426603c4019STakashi Iwai {
427603c4019STakashi Iwai 	struct via_spec *spec = codec->spec;
428603c4019STakashi Iwai 
429603c4019STakashi Iwai 	if (spec->kctls.list) {
430603c4019STakashi Iwai 		struct snd_kcontrol_new *kctl = spec->kctls.list;
431603c4019STakashi Iwai 		int i;
432603c4019STakashi Iwai 		for (i = 0; i < spec->kctls.used; i++)
433603c4019STakashi Iwai 			kfree(kctl[i].name);
434603c4019STakashi Iwai 	}
435603c4019STakashi Iwai 	snd_array_free(&spec->kctls);
436603c4019STakashi Iwai }
437603c4019STakashi Iwai 
438c577b8a1SJoseph Chan /* create input playback/capture controls for the given pin */
4399510e8ddSLydia Wang static int via_new_analog_input(struct via_spec *spec, const char *ctlname,
4407b315bb4STakashi Iwai 				int type_idx, int idx, int mix_nid)
441c577b8a1SJoseph Chan {
442c577b8a1SJoseph Chan 	char name[32];
443c577b8a1SJoseph Chan 	int err;
444c577b8a1SJoseph Chan 
445c577b8a1SJoseph Chan 	sprintf(name, "%s Playback Volume", ctlname);
4467b315bb4STakashi Iwai 	err = __via_add_control(spec, VIA_CTL_WIDGET_VOL, name, type_idx,
447c577b8a1SJoseph Chan 			      HDA_COMPOSE_AMP_VAL(mix_nid, 3, idx, HDA_INPUT));
448c577b8a1SJoseph Chan 	if (err < 0)
449c577b8a1SJoseph Chan 		return err;
450c577b8a1SJoseph Chan 	sprintf(name, "%s Playback Switch", ctlname);
4517b315bb4STakashi Iwai 	err = __via_add_control(spec, VIA_CTL_WIDGET_ANALOG_MUTE, name, type_idx,
452c577b8a1SJoseph Chan 			      HDA_COMPOSE_AMP_VAL(mix_nid, 3, idx, HDA_INPUT));
453c577b8a1SJoseph Chan 	if (err < 0)
454c577b8a1SJoseph Chan 		return err;
455c577b8a1SJoseph Chan 	return 0;
456c577b8a1SJoseph Chan }
457c577b8a1SJoseph Chan 
4585d41762aSTakashi Iwai #define get_connection_index(codec, mux, nid) \
4598d087c76STakashi Iwai 	snd_hda_get_conn_index(codec, mux, nid, 0)
4605d41762aSTakashi Iwai 
4618df2a312STakashi Iwai static bool check_amp_caps(struct hda_codec *codec, hda_nid_t nid, int dir,
4628df2a312STakashi Iwai 			   unsigned int mask)
4638df2a312STakashi Iwai {
464a934d5a9STakashi Iwai 	unsigned int caps;
465a934d5a9STakashi Iwai 	if (!nid)
466a934d5a9STakashi Iwai 		return false;
467a934d5a9STakashi Iwai 	caps = get_wcaps(codec, nid);
4688df2a312STakashi Iwai 	if (dir == HDA_INPUT)
4698df2a312STakashi Iwai 		caps &= AC_WCAP_IN_AMP;
4708df2a312STakashi Iwai 	else
4718df2a312STakashi Iwai 		caps &= AC_WCAP_OUT_AMP;
4728df2a312STakashi Iwai 	if (!caps)
4738df2a312STakashi Iwai 		return false;
4748df2a312STakashi Iwai 	if (query_amp_caps(codec, nid, dir) & mask)
4758df2a312STakashi Iwai 		return true;
4768df2a312STakashi Iwai 	return false;
4778df2a312STakashi Iwai }
4788df2a312STakashi Iwai 
47909a9ad69STakashi Iwai #define have_mute(codec, nid, dir) \
48009a9ad69STakashi Iwai 	check_amp_caps(codec, nid, dir, AC_AMPCAP_MUTE)
4818df2a312STakashi Iwai 
482d69607b3SLydia Wang /* enable/disable the output-route mixers */
483d69607b3SLydia Wang static void activate_output_mix(struct hda_codec *codec, struct nid_path *path,
4843214b966STakashi Iwai 				hda_nid_t mix_nid, int idx, bool enable)
485d69607b3SLydia Wang {
486d69607b3SLydia Wang 	int i, num, val;
487d69607b3SLydia Wang 
488d69607b3SLydia Wang 	if (!path)
489d69607b3SLydia Wang 		return;
49009cf03b8STakashi Iwai 	num = snd_hda_get_num_conns(codec, mix_nid);
491d69607b3SLydia Wang 	for (i = 0; i < num; i++) {
4923214b966STakashi Iwai 		if (i == idx)
493d69607b3SLydia Wang 			val = AMP_IN_UNMUTE(i);
494d69607b3SLydia Wang 		else
495d69607b3SLydia Wang 			val = AMP_IN_MUTE(i);
496d69607b3SLydia Wang 		snd_hda_codec_write(codec, mix_nid, 0,
497d69607b3SLydia Wang 				    AC_VERB_SET_AMP_GAIN_MUTE, val);
498d69607b3SLydia Wang 	}
499d69607b3SLydia Wang }
500d69607b3SLydia Wang 
50109a9ad69STakashi Iwai /* enable/disable the output-route */
50209a9ad69STakashi Iwai static void activate_output_path(struct hda_codec *codec, struct nid_path *path,
50309a9ad69STakashi Iwai 				 bool enable, bool force)
5045d41762aSTakashi Iwai {
505d69607b3SLydia Wang 	struct via_spec *spec = codec->spec;
5063214b966STakashi Iwai 	int i;
50709a9ad69STakashi Iwai 	for (i = 0; i < path->depth; i++) {
50809a9ad69STakashi Iwai 		hda_nid_t src, dst;
50909a9ad69STakashi Iwai 		int idx = path->idx[i];
51009a9ad69STakashi Iwai 		src = path->path[i];
51109a9ad69STakashi Iwai 		if (i < path->depth - 1)
51209a9ad69STakashi Iwai 			dst = path->path[i + 1];
51309a9ad69STakashi Iwai 		else
51409a9ad69STakashi Iwai 			dst = 0;
51509a9ad69STakashi Iwai 		if (enable && path->multi[i])
51609a9ad69STakashi Iwai 			snd_hda_codec_write(codec, dst, 0,
5175d41762aSTakashi Iwai 					    AC_VERB_SET_CONNECT_SEL, idx);
5183214b966STakashi Iwai 		if (!force && (dst == spec->aa_mix_nid))
519e5e14681SLydia Wang 			continue;
5203214b966STakashi Iwai 		if (have_mute(codec, dst, HDA_INPUT))
5213214b966STakashi Iwai 			activate_output_mix(codec, path, dst, idx, enable);
52209a9ad69STakashi Iwai 		if (!force && (src == path->vol_ctl || src == path->mute_ctl))
52309a9ad69STakashi Iwai 			continue;
52409a9ad69STakashi Iwai 		if (have_mute(codec, src, HDA_OUTPUT)) {
52509a9ad69STakashi Iwai 			int val = enable ? AMP_OUT_UNMUTE : AMP_OUT_MUTE;
52609a9ad69STakashi Iwai 			snd_hda_codec_write(codec, src, 0,
52709a9ad69STakashi Iwai 					    AC_VERB_SET_AMP_GAIN_MUTE, val);
52809a9ad69STakashi Iwai 		}
52909a9ad69STakashi Iwai 	}
5305d41762aSTakashi Iwai }
5315d41762aSTakashi Iwai 
5325d41762aSTakashi Iwai /* set the given pin as output */
5335d41762aSTakashi Iwai static void init_output_pin(struct hda_codec *codec, hda_nid_t pin,
5345d41762aSTakashi Iwai 			    int pin_type)
5355d41762aSTakashi Iwai {
5365d41762aSTakashi Iwai 	if (!pin)
5375d41762aSTakashi Iwai 		return;
538cdd03cedSTakashi Iwai 	snd_hda_set_pin_ctl(codec, pin, pin_type);
5395d41762aSTakashi Iwai 	if (snd_hda_query_pin_caps(codec, pin) & AC_PINCAP_EAPD)
5405d41762aSTakashi Iwai 		snd_hda_codec_write(codec, pin, 0,
541d3a11e60STakashi Iwai 				    AC_VERB_SET_EAPD_BTLENABLE, 0x02);
542c577b8a1SJoseph Chan }
543c577b8a1SJoseph Chan 
54409a9ad69STakashi Iwai static void via_auto_init_output(struct hda_codec *codec,
545a353fbb1STakashi Iwai 				 struct nid_path *path, int pin_type)
5465d41762aSTakashi Iwai {
5475d41762aSTakashi Iwai 	unsigned int caps;
548d69607b3SLydia Wang 	hda_nid_t pin;
5495d41762aSTakashi Iwai 
55009a9ad69STakashi Iwai 	if (!path->depth)
5515d41762aSTakashi Iwai 		return;
55209a9ad69STakashi Iwai 	pin = path->path[path->depth - 1];
5535d41762aSTakashi Iwai 
5545d41762aSTakashi Iwai 	init_output_pin(codec, pin, pin_type);
55577e314f7STakashi Iwai 	if (get_wcaps(codec, pin) & AC_WCAP_OUT_AMP)
5565d41762aSTakashi Iwai 		caps = query_amp_caps(codec, pin, HDA_OUTPUT);
55777e314f7STakashi Iwai 	else
55877e314f7STakashi Iwai 		caps = 0;
5595d41762aSTakashi Iwai 	if (caps & AC_AMPCAP_MUTE) {
5605d41762aSTakashi Iwai 		unsigned int val;
5615d41762aSTakashi Iwai 		val = (caps & AC_AMPCAP_OFFSET) >> AC_AMPCAP_OFFSET_SHIFT;
5625d41762aSTakashi Iwai 		snd_hda_codec_write(codec, pin, 0, AC_VERB_SET_AMP_GAIN_MUTE,
5635d41762aSTakashi Iwai 				    AMP_OUT_MUTE | val);
5645d41762aSTakashi Iwai 	}
565a353fbb1STakashi Iwai 	activate_output_path(codec, path, true, true); /* force on */
56609a9ad69STakashi Iwai }
567c577b8a1SJoseph Chan 
568c577b8a1SJoseph Chan static void via_auto_init_multi_out(struct hda_codec *codec)
569c577b8a1SJoseph Chan {
570c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
5713214b966STakashi Iwai 	struct nid_path *path;
572c577b8a1SJoseph Chan 	int i;
573c577b8a1SJoseph Chan 
5743214b966STakashi Iwai 	for (i = 0; i < spec->autocfg.line_outs + spec->smart51_nums; i++) {
5753214b966STakashi Iwai 		path = &spec->out_path[i];
5763214b966STakashi Iwai 		if (!i && spec->aamix_mode && spec->out_mix_path.depth)
5773214b966STakashi Iwai 			path = &spec->out_mix_path;
578a353fbb1STakashi Iwai 		via_auto_init_output(codec, path, PIN_OUT);
5793214b966STakashi Iwai 	}
580c577b8a1SJoseph Chan }
581c577b8a1SJoseph Chan 
582020066d1STakashi Iwai /* deactivate the inactive headphone-paths */
583020066d1STakashi Iwai static void deactivate_hp_paths(struct hda_codec *codec)
584c577b8a1SJoseph Chan {
585c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
5863214b966STakashi Iwai 	int shared = spec->hp_indep_shared;
587c577b8a1SJoseph Chan 
58809a9ad69STakashi Iwai 	if (spec->hp_independent_mode) {
58909a9ad69STakashi Iwai 		activate_output_path(codec, &spec->hp_path, false, false);
5903214b966STakashi Iwai 		activate_output_path(codec, &spec->hp_mix_path, false, false);
5913214b966STakashi Iwai 		if (shared)
5923214b966STakashi Iwai 			activate_output_path(codec, &spec->out_path[shared],
5933214b966STakashi Iwai 					     false, false);
594020066d1STakashi Iwai 	} else if (spec->aamix_mode || !spec->hp_path.depth) {
595020066d1STakashi Iwai 		activate_output_path(codec, &spec->hp_indep_path, false, false);
5963214b966STakashi Iwai 		activate_output_path(codec, &spec->hp_path, false, false);
5973214b966STakashi Iwai 	} else {
598020066d1STakashi Iwai 		activate_output_path(codec, &spec->hp_indep_path, false, false);
5993214b966STakashi Iwai 		activate_output_path(codec, &spec->hp_mix_path, false, false);
60009a9ad69STakashi Iwai 	}
60125eaba2fSLydia Wang }
602c577b8a1SJoseph Chan 
603020066d1STakashi Iwai static void via_auto_init_hp_out(struct hda_codec *codec)
604020066d1STakashi Iwai {
605020066d1STakashi Iwai 	struct via_spec *spec = codec->spec;
606020066d1STakashi Iwai 
607020066d1STakashi Iwai 	if (!spec->hp_path.depth) {
608a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->hp_mix_path, PIN_HP);
609020066d1STakashi Iwai 		return;
610020066d1STakashi Iwai 	}
611020066d1STakashi Iwai 	deactivate_hp_paths(codec);
612020066d1STakashi Iwai 	if (spec->hp_independent_mode)
613a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->hp_indep_path, PIN_HP);
614020066d1STakashi Iwai 	else if (spec->aamix_mode)
615a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->hp_mix_path, PIN_HP);
616020066d1STakashi Iwai 	else
617a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->hp_path, PIN_HP);
618020066d1STakashi Iwai }
619020066d1STakashi Iwai 
6204a918ffeSTakashi Iwai static void via_auto_init_speaker_out(struct hda_codec *codec)
6214a918ffeSTakashi Iwai {
6224a918ffeSTakashi Iwai 	struct via_spec *spec = codec->spec;
6234a918ffeSTakashi Iwai 
6243214b966STakashi Iwai 	if (!spec->autocfg.speaker_outs)
6253214b966STakashi Iwai 		return;
6263214b966STakashi Iwai 	if (!spec->speaker_path.depth) {
627a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->speaker_mix_path, PIN_OUT);
6283214b966STakashi Iwai 		return;
6293214b966STakashi Iwai 	}
6303214b966STakashi Iwai 	if (!spec->aamix_mode) {
6313214b966STakashi Iwai 		activate_output_path(codec, &spec->speaker_mix_path,
6323214b966STakashi Iwai 				     false, false);
633a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->speaker_path, PIN_OUT);
6343214b966STakashi Iwai 	} else {
6353214b966STakashi Iwai 		activate_output_path(codec, &spec->speaker_path, false, false);
636a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->speaker_mix_path, PIN_OUT);
6373214b966STakashi Iwai 	}
6384a918ffeSTakashi Iwai }
6394a918ffeSTakashi Iwai 
640f4a7828bSTakashi Iwai static bool is_smart51_pins(struct hda_codec *codec, hda_nid_t pin);
6416e969d91STakashi Iwai static void via_hp_automute(struct hda_codec *codec);
64232e0191dSClemens Ladisch 
643c577b8a1SJoseph Chan static void via_auto_init_analog_input(struct hda_codec *codec)
644c577b8a1SJoseph Chan {
645c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
6467b315bb4STakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
647096a8854STakashi Iwai 	hda_nid_t conn[HDA_MAX_CONNECTIONS];
64832e0191dSClemens Ladisch 	unsigned int ctl;
649096a8854STakashi Iwai 	int i, num_conns;
650c577b8a1SJoseph Chan 
651096a8854STakashi Iwai 	/* init ADCs */
652096a8854STakashi Iwai 	for (i = 0; i < spec->num_adc_nids; i++) {
65377e314f7STakashi Iwai 		hda_nid_t nid = spec->adc_nids[i];
65477e314f7STakashi Iwai 		if (!(get_wcaps(codec, nid) & AC_WCAP_IN_AMP) ||
65577e314f7STakashi Iwai 		    !(query_amp_caps(codec, nid, HDA_INPUT) & AC_AMPCAP_MUTE))
65677e314f7STakashi Iwai 			continue;
657096a8854STakashi Iwai 		snd_hda_codec_write(codec, spec->adc_nids[i], 0,
658096a8854STakashi Iwai 				    AC_VERB_SET_AMP_GAIN_MUTE,
659096a8854STakashi Iwai 				    AMP_IN_UNMUTE(0));
660096a8854STakashi Iwai 	}
661096a8854STakashi Iwai 
662096a8854STakashi Iwai 	/* init pins */
6637b315bb4STakashi Iwai 	for (i = 0; i < cfg->num_inputs; i++) {
6647b315bb4STakashi Iwai 		hda_nid_t nid = cfg->inputs[i].pin;
665f4a7828bSTakashi Iwai 		if (spec->smart51_enabled && is_smart51_pins(codec, nid))
66632e0191dSClemens Ladisch 			ctl = PIN_OUT;
6674740860bSTakashi Iwai 		else {
66832e0191dSClemens Ladisch 			ctl = PIN_IN;
6694740860bSTakashi Iwai 			if (cfg->inputs[i].type == AUTO_PIN_MIC)
6704740860bSTakashi Iwai 				ctl |= snd_hda_get_default_vref(codec, nid);
6714740860bSTakashi Iwai 		}
672cdd03cedSTakashi Iwai 		snd_hda_set_pin_ctl(codec, nid, ctl);
673c577b8a1SJoseph Chan 	}
674096a8854STakashi Iwai 
675096a8854STakashi Iwai 	/* init input-src */
676096a8854STakashi Iwai 	for (i = 0; i < spec->num_adc_nids; i++) {
677a86a88eaSTakashi Iwai 		int adc_idx = spec->inputs[spec->cur_mux[i]].adc_idx;
678fc1156c0STakashi Iwai 		/* secondary ADCs must have the unique MUX */
679fc1156c0STakashi Iwai 		if (i > 0 && !spec->mux_nids[i])
680fc1156c0STakashi Iwai 			break;
681a86a88eaSTakashi Iwai 		if (spec->mux_nids[adc_idx]) {
682a86a88eaSTakashi Iwai 			int mux_idx = spec->inputs[spec->cur_mux[i]].mux_idx;
683a86a88eaSTakashi Iwai 			snd_hda_codec_write(codec, spec->mux_nids[adc_idx], 0,
684096a8854STakashi Iwai 					    AC_VERB_SET_CONNECT_SEL,
685a86a88eaSTakashi Iwai 					    mux_idx);
686a86a88eaSTakashi Iwai 		}
687a86a88eaSTakashi Iwai 		if (spec->dyn_adc_switch)
688a86a88eaSTakashi Iwai 			break; /* only one input-src */
689096a8854STakashi Iwai 	}
690096a8854STakashi Iwai 
691096a8854STakashi Iwai 	/* init aa-mixer */
692096a8854STakashi Iwai 	if (!spec->aa_mix_nid)
693096a8854STakashi Iwai 		return;
694096a8854STakashi Iwai 	num_conns = snd_hda_get_connections(codec, spec->aa_mix_nid, conn,
695096a8854STakashi Iwai 					    ARRAY_SIZE(conn));
696096a8854STakashi Iwai 	for (i = 0; i < num_conns; i++) {
697096a8854STakashi Iwai 		unsigned int caps = get_wcaps(codec, conn[i]);
698096a8854STakashi Iwai 		if (get_wcaps_type(caps) == AC_WID_PIN)
699096a8854STakashi Iwai 			snd_hda_codec_write(codec, spec->aa_mix_nid, 0,
700096a8854STakashi Iwai 					    AC_VERB_SET_AMP_GAIN_MUTE,
701096a8854STakashi Iwai 					    AMP_IN_MUTE(i));
702096a8854STakashi Iwai 	}
703c577b8a1SJoseph Chan }
704f5271101SLydia Wang 
705054d867eSTakashi Iwai static void update_power_state(struct hda_codec *codec, hda_nid_t nid,
706054d867eSTakashi Iwai 			       unsigned int parm)
707054d867eSTakashi Iwai {
708054d867eSTakashi Iwai 	if (snd_hda_codec_read(codec, nid, 0,
709054d867eSTakashi Iwai 			       AC_VERB_GET_POWER_STATE, 0) == parm)
710054d867eSTakashi Iwai 		return;
711054d867eSTakashi Iwai 	snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_POWER_STATE, parm);
712054d867eSTakashi Iwai }
713054d867eSTakashi Iwai 
714f5271101SLydia Wang static void set_pin_power_state(struct hda_codec *codec, hda_nid_t nid,
715f5271101SLydia Wang 				unsigned int *affected_parm)
716f5271101SLydia Wang {
717f5271101SLydia Wang 	unsigned parm;
718f5271101SLydia Wang 	unsigned def_conf = snd_hda_codec_get_pincfg(codec, nid);
719f5271101SLydia Wang 	unsigned no_presence = (def_conf & AC_DEFCFG_MISC)
720f5271101SLydia Wang 		>> AC_DEFCFG_MISC_SHIFT
721f5271101SLydia Wang 		& AC_DEFCFG_MISC_NO_PRESENCE; /* do not support pin sense */
7221564b287SLydia Wang 	struct via_spec *spec = codec->spec;
72324088a58STakashi Iwai 	unsigned present = 0;
72424088a58STakashi Iwai 
72524088a58STakashi Iwai 	no_presence |= spec->no_pin_power_ctl;
72624088a58STakashi Iwai 	if (!no_presence)
72724088a58STakashi Iwai 		present = snd_hda_jack_detect(codec, nid);
728f4a7828bSTakashi Iwai 	if ((spec->smart51_enabled && is_smart51_pins(codec, nid))
7291564b287SLydia Wang 	    || ((no_presence || present)
7301564b287SLydia Wang 		&& get_defcfg_connect(def_conf) != AC_JACK_PORT_NONE)) {
731f5271101SLydia Wang 		*affected_parm = AC_PWRST_D0; /* if it's connected */
732f5271101SLydia Wang 		parm = AC_PWRST_D0;
733f5271101SLydia Wang 	} else
734f5271101SLydia Wang 		parm = AC_PWRST_D3;
735f5271101SLydia Wang 
736054d867eSTakashi Iwai 	update_power_state(codec, nid, parm);
737f5271101SLydia Wang }
738f5271101SLydia Wang 
73924088a58STakashi Iwai static int via_pin_power_ctl_info(struct snd_kcontrol *kcontrol,
74024088a58STakashi Iwai 				  struct snd_ctl_elem_info *uinfo)
74124088a58STakashi Iwai {
742*dda415d4STakashi Iwai 	return snd_hda_enum_bool_helper_info(kcontrol, uinfo);
74324088a58STakashi Iwai }
74424088a58STakashi Iwai 
74524088a58STakashi Iwai static int via_pin_power_ctl_get(struct snd_kcontrol *kcontrol,
74624088a58STakashi Iwai 				 struct snd_ctl_elem_value *ucontrol)
74724088a58STakashi Iwai {
74824088a58STakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
74924088a58STakashi Iwai 	struct via_spec *spec = codec->spec;
75024088a58STakashi Iwai 	ucontrol->value.enumerated.item[0] = !spec->no_pin_power_ctl;
75124088a58STakashi Iwai 	return 0;
75224088a58STakashi Iwai }
75324088a58STakashi Iwai 
75424088a58STakashi Iwai static int via_pin_power_ctl_put(struct snd_kcontrol *kcontrol,
75524088a58STakashi Iwai 				 struct snd_ctl_elem_value *ucontrol)
75624088a58STakashi Iwai {
75724088a58STakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
75824088a58STakashi Iwai 	struct via_spec *spec = codec->spec;
75924088a58STakashi Iwai 	unsigned int val = !ucontrol->value.enumerated.item[0];
76024088a58STakashi Iwai 
76124088a58STakashi Iwai 	if (val == spec->no_pin_power_ctl)
76224088a58STakashi Iwai 		return 0;
76324088a58STakashi Iwai 	spec->no_pin_power_ctl = val;
76424088a58STakashi Iwai 	set_widgets_power_state(codec);
765e9d010c2STakashi Iwai 	analog_low_current_mode(codec);
76624088a58STakashi Iwai 	return 1;
76724088a58STakashi Iwai }
76824088a58STakashi Iwai 
76924088a58STakashi Iwai static const struct snd_kcontrol_new via_pin_power_ctl_enum = {
77024088a58STakashi Iwai 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
77124088a58STakashi Iwai 	.name = "Dynamic Power-Control",
77224088a58STakashi Iwai 	.info = via_pin_power_ctl_info,
77324088a58STakashi Iwai 	.get = via_pin_power_ctl_get,
77424088a58STakashi Iwai 	.put = via_pin_power_ctl_put,
77524088a58STakashi Iwai };
77624088a58STakashi Iwai 
77724088a58STakashi Iwai 
7780aa62aefSHarald Welte static int via_independent_hp_info(struct snd_kcontrol *kcontrol,
7790aa62aefSHarald Welte 				   struct snd_ctl_elem_info *uinfo)
7800aa62aefSHarald Welte {
7818df2a312STakashi Iwai 	static const char * const texts[] = { "OFF", "ON" };
7828df2a312STakashi Iwai 
7838df2a312STakashi Iwai 	uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
7848df2a312STakashi Iwai 	uinfo->count = 1;
7858df2a312STakashi Iwai 	uinfo->value.enumerated.items = 2;
7868df2a312STakashi Iwai 	if (uinfo->value.enumerated.item >= 2)
7878df2a312STakashi Iwai 		uinfo->value.enumerated.item = 1;
7888df2a312STakashi Iwai 	strcpy(uinfo->value.enumerated.name,
7898df2a312STakashi Iwai 	       texts[uinfo->value.enumerated.item]);
7908df2a312STakashi Iwai 	return 0;
7910aa62aefSHarald Welte }
7920aa62aefSHarald Welte 
7930aa62aefSHarald Welte static int via_independent_hp_get(struct snd_kcontrol *kcontrol,
7940aa62aefSHarald Welte 				  struct snd_ctl_elem_value *ucontrol)
7950aa62aefSHarald Welte {
7960aa62aefSHarald Welte 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
797cdc1784dSLydia Wang 	struct via_spec *spec = codec->spec;
798cdc1784dSLydia Wang 
799ece8d043STakashi Iwai 	ucontrol->value.enumerated.item[0] = spec->hp_independent_mode;
800cdc1784dSLydia Wang 	return 0;
801cdc1784dSLydia Wang }
802cdc1784dSLydia Wang 
8033b607e3dSTakashi Iwai /* adjust spec->multiout setup according to the current flags */
8043b607e3dSTakashi Iwai static void setup_playback_multi_pcm(struct via_spec *spec)
8053b607e3dSTakashi Iwai {
8063b607e3dSTakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
8073b607e3dSTakashi Iwai 	spec->multiout.num_dacs = cfg->line_outs + spec->smart51_nums;
8083b607e3dSTakashi Iwai 	spec->multiout.hp_nid = 0;
8093b607e3dSTakashi Iwai 	if (!spec->hp_independent_mode) {
8103b607e3dSTakashi Iwai 		if (!spec->hp_indep_shared)
8113b607e3dSTakashi Iwai 			spec->multiout.hp_nid = spec->hp_dac_nid;
8123b607e3dSTakashi Iwai 	} else {
8133b607e3dSTakashi Iwai 		if (spec->hp_indep_shared)
8143b607e3dSTakashi Iwai 			spec->multiout.num_dacs = cfg->line_outs - 1;
8153b607e3dSTakashi Iwai 	}
8163b607e3dSTakashi Iwai }
8173b607e3dSTakashi Iwai 
8183b607e3dSTakashi Iwai /* update DAC setups according to indep-HP switch;
8193b607e3dSTakashi Iwai  * this function is called only when indep-HP is modified
8203b607e3dSTakashi Iwai  */
8213b607e3dSTakashi Iwai static void switch_indep_hp_dacs(struct hda_codec *codec)
8223b607e3dSTakashi Iwai {
8233b607e3dSTakashi Iwai 	struct via_spec *spec = codec->spec;
8243b607e3dSTakashi Iwai 	int shared = spec->hp_indep_shared;
8253b607e3dSTakashi Iwai 	hda_nid_t shared_dac, hp_dac;
8263b607e3dSTakashi Iwai 
8273b607e3dSTakashi Iwai 	if (!spec->opened_streams)
8283b607e3dSTakashi Iwai 		return;
8293b607e3dSTakashi Iwai 
8303b607e3dSTakashi Iwai 	shared_dac = shared ? spec->multiout.dac_nids[shared] : 0;
8313b607e3dSTakashi Iwai 	hp_dac = spec->hp_dac_nid;
8323b607e3dSTakashi Iwai 	if (spec->hp_independent_mode) {
8333b607e3dSTakashi Iwai 		/* switch to indep-HP mode */
8343b607e3dSTakashi Iwai 		if (spec->active_streams & STREAM_MULTI_OUT) {
8353b607e3dSTakashi Iwai 			__snd_hda_codec_cleanup_stream(codec, hp_dac, 1);
8363b607e3dSTakashi Iwai 			__snd_hda_codec_cleanup_stream(codec, shared_dac, 1);
8373b607e3dSTakashi Iwai 		}
8383b607e3dSTakashi Iwai 		if (spec->active_streams & STREAM_INDEP_HP)
8393b607e3dSTakashi Iwai 			snd_hda_codec_setup_stream(codec, hp_dac,
8403b607e3dSTakashi Iwai 						   spec->cur_hp_stream_tag, 0,
8413b607e3dSTakashi Iwai 						   spec->cur_hp_format);
8423b607e3dSTakashi Iwai 	} else {
8433b607e3dSTakashi Iwai 		/* back to HP or shared-DAC */
8443b607e3dSTakashi Iwai 		if (spec->active_streams & STREAM_INDEP_HP)
8453b607e3dSTakashi Iwai 			__snd_hda_codec_cleanup_stream(codec, hp_dac, 1);
8463b607e3dSTakashi Iwai 		if (spec->active_streams & STREAM_MULTI_OUT) {
8473b607e3dSTakashi Iwai 			hda_nid_t dac;
8483b607e3dSTakashi Iwai 			int ch;
8493b607e3dSTakashi Iwai 			if (shared_dac) { /* reset mutli-ch DAC */
8503b607e3dSTakashi Iwai 				dac = shared_dac;
8513b607e3dSTakashi Iwai 				ch = shared * 2;
8523b607e3dSTakashi Iwai 			} else { /* reset HP DAC */
8533b607e3dSTakashi Iwai 				dac = hp_dac;
8543b607e3dSTakashi Iwai 				ch = 0;
8553b607e3dSTakashi Iwai 			}
8563b607e3dSTakashi Iwai 			snd_hda_codec_setup_stream(codec, dac,
8573b607e3dSTakashi Iwai 						   spec->cur_dac_stream_tag, ch,
8583b607e3dSTakashi Iwai 						   spec->cur_dac_format);
8593b607e3dSTakashi Iwai 		}
8603b607e3dSTakashi Iwai 	}
8613b607e3dSTakashi Iwai 	setup_playback_multi_pcm(spec);
8623b607e3dSTakashi Iwai }
8633b607e3dSTakashi Iwai 
8640aa62aefSHarald Welte static int via_independent_hp_put(struct snd_kcontrol *kcontrol,
8650aa62aefSHarald Welte 				  struct snd_ctl_elem_value *ucontrol)
8660aa62aefSHarald Welte {
8670aa62aefSHarald Welte 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
8680aa62aefSHarald Welte 	struct via_spec *spec = codec->spec;
8693214b966STakashi Iwai 	int cur, shared;
8708df2a312STakashi Iwai 
8713b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
87225250505STakashi Iwai 	cur = !!ucontrol->value.enumerated.item[0];
8733b607e3dSTakashi Iwai 	if (spec->hp_independent_mode == cur) {
8743b607e3dSTakashi Iwai 		mutex_unlock(&spec->config_mutex);
87525250505STakashi Iwai 		return 0;
8763b607e3dSTakashi Iwai 	}
87725250505STakashi Iwai 	spec->hp_independent_mode = cur;
8783214b966STakashi Iwai 	shared = spec->hp_indep_shared;
879020066d1STakashi Iwai 	deactivate_hp_paths(codec);
880020066d1STakashi Iwai 	if (cur)
881020066d1STakashi Iwai 		activate_output_path(codec, &spec->hp_indep_path, true, false);
882020066d1STakashi Iwai 	else {
8833214b966STakashi Iwai 		if (shared)
8843214b966STakashi Iwai 			activate_output_path(codec, &spec->out_path[shared],
88525250505STakashi Iwai 					     true, false);
886020066d1STakashi Iwai 		if (spec->aamix_mode || !spec->hp_path.depth)
887020066d1STakashi Iwai 			activate_output_path(codec, &spec->hp_mix_path,
888020066d1STakashi Iwai 					     true, false);
889020066d1STakashi Iwai 		else
890020066d1STakashi Iwai 			activate_output_path(codec, &spec->hp_path,
891020066d1STakashi Iwai 					     true, false);
8928df2a312STakashi Iwai 	}
8930aa62aefSHarald Welte 
8943b607e3dSTakashi Iwai 	switch_indep_hp_dacs(codec);
8953b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
8963b607e3dSTakashi Iwai 
897ce0e5a9eSLydia Wang 	/* update jack power state */
8983e95b9abSLydia Wang 	set_widgets_power_state(codec);
8996e969d91STakashi Iwai 	via_hp_automute(codec);
90025250505STakashi Iwai 	return 1;
9010aa62aefSHarald Welte }
9020aa62aefSHarald Welte 
903ece8d043STakashi Iwai static const struct snd_kcontrol_new via_hp_mixer = {
9040aa62aefSHarald Welte 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
9050aa62aefSHarald Welte 	.name = "Independent HP",
9060aa62aefSHarald Welte 	.info = via_independent_hp_info,
9070aa62aefSHarald Welte 	.get = via_independent_hp_get,
9080aa62aefSHarald Welte 	.put = via_independent_hp_put,
9090aa62aefSHarald Welte };
9100aa62aefSHarald Welte 
9113d83e577STakashi Iwai static int via_hp_build(struct hda_codec *codec)
9125b0cb1d8SJaroslav Kysela {
9133d83e577STakashi Iwai 	struct via_spec *spec = codec->spec;
9145b0cb1d8SJaroslav Kysela 	struct snd_kcontrol_new *knew;
9155b0cb1d8SJaroslav Kysela 	hda_nid_t nid;
9165b0cb1d8SJaroslav Kysela 
9175b0cb1d8SJaroslav Kysela 	nid = spec->autocfg.hp_pins[0];
918ece8d043STakashi Iwai 	knew = via_clone_control(spec, &via_hp_mixer);
9193d83e577STakashi Iwai 	if (knew == NULL)
9203d83e577STakashi Iwai 		return -ENOMEM;
9213d83e577STakashi Iwai 
9225b0cb1d8SJaroslav Kysela 	knew->subdevice = HDA_SUBDEV_NID_FLAG | nid;
9235b0cb1d8SJaroslav Kysela 
9245b0cb1d8SJaroslav Kysela 	return 0;
9255b0cb1d8SJaroslav Kysela }
9265b0cb1d8SJaroslav Kysela 
9271564b287SLydia Wang static void notify_aa_path_ctls(struct hda_codec *codec)
9281564b287SLydia Wang {
929e3d7a143STakashi Iwai 	struct via_spec *spec = codec->spec;
9301564b287SLydia Wang 	int i;
9311564b287SLydia Wang 
932e3d7a143STakashi Iwai 	for (i = 0; i < spec->smart51_nums; i++) {
933e3d7a143STakashi Iwai 		struct snd_kcontrol *ctl;
934e3d7a143STakashi Iwai 		struct snd_ctl_elem_id id;
9351564b287SLydia Wang 		memset(&id, 0, sizeof(id));
9361564b287SLydia Wang 		id.iface = SNDRV_CTL_ELEM_IFACE_MIXER;
937e3d7a143STakashi Iwai 		sprintf(id.name, "%s Playback Volume", spec->smart51_labels[i]);
938525566cbSLydia Wang 		ctl = snd_hda_find_mixer_ctl(codec, id.name);
939525566cbSLydia Wang 		if (ctl)
940525566cbSLydia Wang 			snd_ctl_notify(codec->bus->card,
941525566cbSLydia Wang 					SNDRV_CTL_EVENT_MASK_VALUE,
942525566cbSLydia Wang 					&ctl->id);
9431564b287SLydia Wang 	}
9441564b287SLydia Wang }
9451564b287SLydia Wang 
9461564b287SLydia Wang static void mute_aa_path(struct hda_codec *codec, int mute)
9471564b287SLydia Wang {
9481564b287SLydia Wang 	struct via_spec *spec = codec->spec;
9491564b287SLydia Wang 	int val = mute ? HDA_AMP_MUTE : HDA_AMP_UNMUTE;
950e3d7a143STakashi Iwai 	int i;
951e3d7a143STakashi Iwai 
952e3d7a143STakashi Iwai 	/* check AA path's mute status */
953e3d7a143STakashi Iwai 	for (i = 0; i < spec->smart51_nums; i++) {
954e3d7a143STakashi Iwai 		if (spec->smart51_idxs[i] < 0)
955e3d7a143STakashi Iwai 			continue;
956e3d7a143STakashi Iwai 		snd_hda_codec_amp_stereo(codec, spec->aa_mix_nid,
957e3d7a143STakashi Iwai 					 HDA_INPUT, spec->smart51_idxs[i],
9581564b287SLydia Wang 					 HDA_AMP_MUTE, val);
9591564b287SLydia Wang 	}
9601564b287SLydia Wang }
961f4a7828bSTakashi Iwai 
962e3d7a143STakashi Iwai static bool is_smart51_pins(struct hda_codec *codec, hda_nid_t pin)
963e3d7a143STakashi Iwai {
964e3d7a143STakashi Iwai 	struct via_spec *spec = codec->spec;
965e3d7a143STakashi Iwai 	int i;
966e3d7a143STakashi Iwai 
967e3d7a143STakashi Iwai 	for (i = 0; i < spec->smart51_nums; i++)
968e3d7a143STakashi Iwai 		if (spec->smart51_pins[i] == pin)
969e3d7a143STakashi Iwai 			return true;
970e3d7a143STakashi Iwai 	return false;
971e3d7a143STakashi Iwai }
972e3d7a143STakashi Iwai 
9731564b287SLydia Wang static int via_smart51_get(struct snd_kcontrol *kcontrol,
9741564b287SLydia Wang 			   struct snd_ctl_elem_value *ucontrol)
9751564b287SLydia Wang {
9761564b287SLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
9771564b287SLydia Wang 	struct via_spec *spec = codec->spec;
9781564b287SLydia Wang 
979f2b1c9f0STakashi Iwai 	*ucontrol->value.integer.value = spec->smart51_enabled;
9801564b287SLydia Wang 	return 0;
9811564b287SLydia Wang }
9821564b287SLydia Wang 
9831564b287SLydia Wang static int via_smart51_put(struct snd_kcontrol *kcontrol,
9841564b287SLydia Wang 			   struct snd_ctl_elem_value *ucontrol)
9851564b287SLydia Wang {
9861564b287SLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
9871564b287SLydia Wang 	struct via_spec *spec = codec->spec;
9881564b287SLydia Wang 	int out_in = *ucontrol->value.integer.value
9891564b287SLydia Wang 		? AC_PINCTL_OUT_EN : AC_PINCTL_IN_EN;
9901564b287SLydia Wang 	int i;
9911564b287SLydia Wang 
992e3d7a143STakashi Iwai 	for (i = 0; i < spec->smart51_nums; i++) {
993e3d7a143STakashi Iwai 		hda_nid_t nid = spec->smart51_pins[i];
9947b315bb4STakashi Iwai 		unsigned int parm;
9957b315bb4STakashi Iwai 
9967b315bb4STakashi Iwai 		parm = snd_hda_codec_read(codec, nid, 0,
9971564b287SLydia Wang 					  AC_VERB_GET_PIN_WIDGET_CONTROL, 0);
9981564b287SLydia Wang 		parm &= ~(AC_PINCTL_IN_EN | AC_PINCTL_OUT_EN);
9991564b287SLydia Wang 		parm |= out_in;
1000cdd03cedSTakashi Iwai 		snd_hda_set_pin_ctl(codec, nid, parm);
10011564b287SLydia Wang 		if (out_in == AC_PINCTL_OUT_EN) {
10021564b287SLydia Wang 			mute_aa_path(codec, 1);
10031564b287SLydia Wang 			notify_aa_path_ctls(codec);
10041564b287SLydia Wang 		}
10051564b287SLydia Wang 	}
10061564b287SLydia Wang 	spec->smart51_enabled = *ucontrol->value.integer.value;
10073e95b9abSLydia Wang 	set_widgets_power_state(codec);
10081564b287SLydia Wang 	return 1;
10091564b287SLydia Wang }
10101564b287SLydia Wang 
10115f4b36d6STakashi Iwai static const struct snd_kcontrol_new via_smart51_mixer = {
10121564b287SLydia Wang 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
10131564b287SLydia Wang 	.name = "Smart 5.1",
10141564b287SLydia Wang 	.count = 1,
1015f2b1c9f0STakashi Iwai 	.info = snd_ctl_boolean_mono_info,
10161564b287SLydia Wang 	.get = via_smart51_get,
10171564b287SLydia Wang 	.put = via_smart51_put,
10181564b287SLydia Wang };
10191564b287SLydia Wang 
1020f4a7828bSTakashi Iwai static int via_smart51_build(struct hda_codec *codec)
10215b0cb1d8SJaroslav Kysela {
1022f4a7828bSTakashi Iwai 	struct via_spec *spec = codec->spec;
10235b0cb1d8SJaroslav Kysela 
1024e3d7a143STakashi Iwai 	if (!spec->smart51_nums)
1025cb34c207SLydia Wang 		return 0;
1026e3d7a143STakashi Iwai 	if (!via_clone_control(spec, &via_smart51_mixer))
10275b0cb1d8SJaroslav Kysela 		return -ENOMEM;
10285b0cb1d8SJaroslav Kysela 	return 0;
10295b0cb1d8SJaroslav Kysela }
10305b0cb1d8SJaroslav Kysela 
1031f5271101SLydia Wang /* check AA path's mute status */
1032ada509ecSTakashi Iwai static bool is_aa_path_mute(struct hda_codec *codec)
1033ada509ecSTakashi Iwai {
1034ada509ecSTakashi Iwai 	struct via_spec *spec = codec->spec;
1035ada509ecSTakashi Iwai 	const struct hda_amp_list *p;
1036ada509ecSTakashi Iwai 	int i, ch, v;
1037ada509ecSTakashi Iwai 
1038ada509ecSTakashi Iwai 	for (i = 0; i < spec->num_loopbacks; i++) {
1039ada509ecSTakashi Iwai 		p = &spec->loopback_list[i];
1040ada509ecSTakashi Iwai 		for (ch = 0; ch < 2; ch++) {
1041ada509ecSTakashi Iwai 			v = snd_hda_codec_amp_read(codec, p->nid, ch, p->dir,
1042ada509ecSTakashi Iwai 						   p->idx);
1043ada509ecSTakashi Iwai 			if (!(v & HDA_AMP_MUTE) && v > 0)
1044ada509ecSTakashi Iwai 				return false;
1045f5271101SLydia Wang 		}
1046f5271101SLydia Wang 	}
1047ada509ecSTakashi Iwai 	return true;
1048f5271101SLydia Wang }
1049f5271101SLydia Wang 
1050f5271101SLydia Wang /* enter/exit analog low-current mode */
1051e9d010c2STakashi Iwai static void __analog_low_current_mode(struct hda_codec *codec, bool force)
1052f5271101SLydia Wang {
1053f5271101SLydia Wang 	struct via_spec *spec = codec->spec;
1054ada509ecSTakashi Iwai 	bool enable;
1055ada509ecSTakashi Iwai 	unsigned int verb, parm;
1056f5271101SLydia Wang 
1057e9d010c2STakashi Iwai 	if (spec->no_pin_power_ctl)
1058e9d010c2STakashi Iwai 		enable = false;
1059e9d010c2STakashi Iwai 	else
106092433923STakashi Iwai 		enable = is_aa_path_mute(codec) && !spec->opened_streams;
1061e9d010c2STakashi Iwai 	if (enable == spec->alc_mode && !force)
1062e9d010c2STakashi Iwai 		return;
1063e9d010c2STakashi Iwai 	spec->alc_mode = enable;
1064f5271101SLydia Wang 
1065f5271101SLydia Wang 	/* decide low current mode's verb & parameter */
1066f5271101SLydia Wang 	switch (spec->codec_type) {
1067f5271101SLydia Wang 	case VT1708B_8CH:
1068f5271101SLydia Wang 	case VT1708B_4CH:
1069f5271101SLydia Wang 		verb = 0xf70;
1070f5271101SLydia Wang 		parm = enable ? 0x02 : 0x00; /* 0x02: 2/3x, 0x00: 1x */
1071f5271101SLydia Wang 		break;
1072f5271101SLydia Wang 	case VT1708S:
1073eb7188caSLydia Wang 	case VT1718S:
1074f3db423dSLydia Wang 	case VT1716S:
1075f5271101SLydia Wang 		verb = 0xf73;
1076f5271101SLydia Wang 		parm = enable ? 0x51 : 0xe1; /* 0x51: 4/28x, 0xe1: 1x */
1077f5271101SLydia Wang 		break;
1078f5271101SLydia Wang 	case VT1702:
1079f5271101SLydia Wang 		verb = 0xf73;
1080f5271101SLydia Wang 		parm = enable ? 0x01 : 0x1d; /* 0x01: 4/40x, 0x1d: 1x */
1081f5271101SLydia Wang 		break;
108225eaba2fSLydia Wang 	case VT2002P:
1083ab6734e7SLydia Wang 	case VT1812:
108411890956SLydia Wang 	case VT1802:
108525eaba2fSLydia Wang 		verb = 0xf93;
108625eaba2fSLydia Wang 		parm = enable ? 0x00 : 0xe0; /* 0x00: 4/40x, 0xe0: 1x */
108725eaba2fSLydia Wang 		break;
1088f5271101SLydia Wang 	default:
1089f5271101SLydia Wang 		return;		/* other codecs are not supported */
1090f5271101SLydia Wang 	}
1091f5271101SLydia Wang 	/* send verb */
1092f5271101SLydia Wang 	snd_hda_codec_write(codec, codec->afg, 0, verb, parm);
1093f5271101SLydia Wang }
1094f5271101SLydia Wang 
1095e9d010c2STakashi Iwai static void analog_low_current_mode(struct hda_codec *codec)
1096e9d010c2STakashi Iwai {
1097e9d010c2STakashi Iwai 	return __analog_low_current_mode(codec, false);
1098e9d010c2STakashi Iwai }
1099e9d010c2STakashi Iwai 
1100c577b8a1SJoseph Chan /*
1101c577b8a1SJoseph Chan  * generic initialization of ADC, input mixers and output mixers
1102c577b8a1SJoseph Chan  */
1103096a8854STakashi Iwai static const struct hda_verb vt1708_init_verbs[] = {
1104aa266fccSLydia Wang 	/* power down jack detect function */
1105aa266fccSLydia Wang 	{0x1, 0xf81, 0x1},
1106f7278fd0SJosepch Chan 	{ }
1107c577b8a1SJoseph Chan };
1108c577b8a1SJoseph Chan 
11093b607e3dSTakashi Iwai static void set_stream_open(struct hda_codec *codec, int bit, bool active)
11107eb56e84STakashi Iwai {
1111ada509ecSTakashi Iwai 	struct via_spec *spec = codec->spec;
1112ada509ecSTakashi Iwai 
1113ada509ecSTakashi Iwai 	if (active)
11143b607e3dSTakashi Iwai 		spec->opened_streams |= bit;
1115ada509ecSTakashi Iwai 	else
11163b607e3dSTakashi Iwai 		spec->opened_streams &= ~bit;
1117ada509ecSTakashi Iwai 	analog_low_current_mode(codec);
11187eb56e84STakashi Iwai }
11197eb56e84STakashi Iwai 
1120ece8d043STakashi Iwai static int via_playback_multi_pcm_open(struct hda_pcm_stream *hinfo,
1121c577b8a1SJoseph Chan 				 struct hda_codec *codec,
1122c577b8a1SJoseph Chan 				 struct snd_pcm_substream *substream)
1123c577b8a1SJoseph Chan {
1124c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
112525250505STakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
1126ada509ecSTakashi Iwai 	int err;
1127ece8d043STakashi Iwai 
112825250505STakashi Iwai 	spec->multiout.num_dacs = cfg->line_outs + spec->smart51_nums;
112925250505STakashi Iwai 	spec->multiout.max_channels = spec->multiout.num_dacs * 2;
11303b607e3dSTakashi Iwai 	set_stream_open(codec, STREAM_MULTI_OUT, true);
1131ada509ecSTakashi Iwai 	err = snd_hda_multi_out_analog_open(codec, &spec->multiout, substream,
11329a08160bSTakashi Iwai 					    hinfo);
1133ada509ecSTakashi Iwai 	if (err < 0) {
11343b607e3dSTakashi Iwai 		set_stream_open(codec, STREAM_MULTI_OUT, false);
1135ada509ecSTakashi Iwai 		return err;
1136ada509ecSTakashi Iwai 	}
1137ada509ecSTakashi Iwai 	return 0;
1138c577b8a1SJoseph Chan }
1139c577b8a1SJoseph Chan 
1140ece8d043STakashi Iwai static int via_playback_multi_pcm_close(struct hda_pcm_stream *hinfo,
11419af74210STakashi Iwai 				  struct hda_codec *codec,
11429af74210STakashi Iwai 				  struct snd_pcm_substream *substream)
11439af74210STakashi Iwai {
11443b607e3dSTakashi Iwai 	set_stream_open(codec, STREAM_MULTI_OUT, false);
11459af74210STakashi Iwai 	return 0;
11469af74210STakashi Iwai }
11479af74210STakashi Iwai 
11487eb56e84STakashi Iwai static int via_playback_hp_pcm_open(struct hda_pcm_stream *hinfo,
11497eb56e84STakashi Iwai 				    struct hda_codec *codec,
11507eb56e84STakashi Iwai 				    struct snd_pcm_substream *substream)
11517eb56e84STakashi Iwai {
11527eb56e84STakashi Iwai 	struct via_spec *spec = codec->spec;
11537eb56e84STakashi Iwai 
1154ece8d043STakashi Iwai 	if (snd_BUG_ON(!spec->hp_dac_nid))
11557eb56e84STakashi Iwai 		return -EINVAL;
11563b607e3dSTakashi Iwai 	set_stream_open(codec, STREAM_INDEP_HP, true);
1157ece8d043STakashi Iwai 	return 0;
1158ece8d043STakashi Iwai }
1159ece8d043STakashi Iwai 
1160ece8d043STakashi Iwai static int via_playback_hp_pcm_close(struct hda_pcm_stream *hinfo,
1161ece8d043STakashi Iwai 				     struct hda_codec *codec,
1162ece8d043STakashi Iwai 				     struct snd_pcm_substream *substream)
1163ece8d043STakashi Iwai {
11643b607e3dSTakashi Iwai 	set_stream_open(codec, STREAM_INDEP_HP, false);
11657eb56e84STakashi Iwai 	return 0;
11667eb56e84STakashi Iwai }
11677eb56e84STakashi Iwai 
11687eb56e84STakashi Iwai static int via_playback_multi_pcm_prepare(struct hda_pcm_stream *hinfo,
11697eb56e84STakashi Iwai 					  struct hda_codec *codec,
11700aa62aefSHarald Welte 					  unsigned int stream_tag,
11710aa62aefSHarald Welte 					  unsigned int format,
11720aa62aefSHarald Welte 					  struct snd_pcm_substream *substream)
11730aa62aefSHarald Welte {
11740aa62aefSHarald Welte 	struct via_spec *spec = codec->spec;
11750aa62aefSHarald Welte 
11763b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
11773b607e3dSTakashi Iwai 	setup_playback_multi_pcm(spec);
1178ece8d043STakashi Iwai 	snd_hda_multi_out_analog_prepare(codec, &spec->multiout, stream_tag,
1179ece8d043STakashi Iwai 					 format, substream);
11803b607e3dSTakashi Iwai 	/* remember for dynamic DAC switch with indep-HP */
11813b607e3dSTakashi Iwai 	spec->active_streams |= STREAM_MULTI_OUT;
11823b607e3dSTakashi Iwai 	spec->cur_dac_stream_tag = stream_tag;
11833b607e3dSTakashi Iwai 	spec->cur_dac_format = format;
11843b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
1185187d333eSTakashi Iwai 	vt1708_update_hp_work(spec);
11867eb56e84STakashi Iwai 	return 0;
11870aa62aefSHarald Welte }
11880aa62aefSHarald Welte 
11897eb56e84STakashi Iwai static int via_playback_hp_pcm_prepare(struct hda_pcm_stream *hinfo,
11900aa62aefSHarald Welte 				       struct hda_codec *codec,
11910aa62aefSHarald Welte 				       unsigned int stream_tag,
11920aa62aefSHarald Welte 				       unsigned int format,
11930aa62aefSHarald Welte 				       struct snd_pcm_substream *substream)
11940aa62aefSHarald Welte {
11950aa62aefSHarald Welte 	struct via_spec *spec = codec->spec;
11960aa62aefSHarald Welte 
11973b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
11983b607e3dSTakashi Iwai 	if (spec->hp_independent_mode)
1199ece8d043STakashi Iwai 		snd_hda_codec_setup_stream(codec, spec->hp_dac_nid,
1200ece8d043STakashi Iwai 					   stream_tag, 0, format);
12013b607e3dSTakashi Iwai 	spec->active_streams |= STREAM_INDEP_HP;
12023b607e3dSTakashi Iwai 	spec->cur_hp_stream_tag = stream_tag;
12033b607e3dSTakashi Iwai 	spec->cur_hp_format = format;
12043b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
1205187d333eSTakashi Iwai 	vt1708_update_hp_work(spec);
12060aa62aefSHarald Welte 	return 0;
12070aa62aefSHarald Welte }
12080aa62aefSHarald Welte 
12090aa62aefSHarald Welte static int via_playback_multi_pcm_cleanup(struct hda_pcm_stream *hinfo,
12100aa62aefSHarald Welte 				    struct hda_codec *codec,
12110aa62aefSHarald Welte 				    struct snd_pcm_substream *substream)
12120aa62aefSHarald Welte {
12130aa62aefSHarald Welte 	struct via_spec *spec = codec->spec;
12140aa62aefSHarald Welte 
12153b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
1216ece8d043STakashi Iwai 	snd_hda_multi_out_analog_cleanup(codec, &spec->multiout);
12173b607e3dSTakashi Iwai 	spec->active_streams &= ~STREAM_MULTI_OUT;
12183b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
1219187d333eSTakashi Iwai 	vt1708_update_hp_work(spec);
12207eb56e84STakashi Iwai 	return 0;
12210aa62aefSHarald Welte }
12227eb56e84STakashi Iwai 
12237eb56e84STakashi Iwai static int via_playback_hp_pcm_cleanup(struct hda_pcm_stream *hinfo,
12247eb56e84STakashi Iwai 				       struct hda_codec *codec,
12257eb56e84STakashi Iwai 				       struct snd_pcm_substream *substream)
12267eb56e84STakashi Iwai {
12277eb56e84STakashi Iwai 	struct via_spec *spec = codec->spec;
12287eb56e84STakashi Iwai 
12293b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
12303b607e3dSTakashi Iwai 	if (spec->hp_independent_mode)
1231ece8d043STakashi Iwai 		snd_hda_codec_setup_stream(codec, spec->hp_dac_nid, 0, 0, 0);
12323b607e3dSTakashi Iwai 	spec->active_streams &= ~STREAM_INDEP_HP;
12333b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
1234187d333eSTakashi Iwai 	vt1708_update_hp_work(spec);
12350aa62aefSHarald Welte 	return 0;
12360aa62aefSHarald Welte }
12370aa62aefSHarald Welte 
1238c577b8a1SJoseph Chan /*
1239c577b8a1SJoseph Chan  * Digital out
1240c577b8a1SJoseph Chan  */
1241c577b8a1SJoseph Chan static int via_dig_playback_pcm_open(struct hda_pcm_stream *hinfo,
1242c577b8a1SJoseph Chan 				     struct hda_codec *codec,
1243c577b8a1SJoseph Chan 				     struct snd_pcm_substream *substream)
1244c577b8a1SJoseph Chan {
1245c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
1246c577b8a1SJoseph Chan 	return snd_hda_multi_out_dig_open(codec, &spec->multiout);
1247c577b8a1SJoseph Chan }
1248c577b8a1SJoseph Chan 
1249c577b8a1SJoseph Chan static int via_dig_playback_pcm_close(struct hda_pcm_stream *hinfo,
1250c577b8a1SJoseph Chan 				      struct hda_codec *codec,
1251c577b8a1SJoseph Chan 				      struct snd_pcm_substream *substream)
1252c577b8a1SJoseph Chan {
1253c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
1254c577b8a1SJoseph Chan 	return snd_hda_multi_out_dig_close(codec, &spec->multiout);
1255c577b8a1SJoseph Chan }
1256c577b8a1SJoseph Chan 
12575691ec7fSHarald Welte static int via_dig_playback_pcm_prepare(struct hda_pcm_stream *hinfo,
125898aa34c0SHarald Welte 					struct hda_codec *codec,
125998aa34c0SHarald Welte 					unsigned int stream_tag,
126098aa34c0SHarald Welte 					unsigned int format,
126198aa34c0SHarald Welte 					struct snd_pcm_substream *substream)
126298aa34c0SHarald Welte {
126398aa34c0SHarald Welte 	struct via_spec *spec = codec->spec;
12649da29271STakashi Iwai 	return snd_hda_multi_out_dig_prepare(codec, &spec->multiout,
12659da29271STakashi Iwai 					     stream_tag, format, substream);
12669da29271STakashi Iwai }
12675691ec7fSHarald Welte 
12689da29271STakashi Iwai static int via_dig_playback_pcm_cleanup(struct hda_pcm_stream *hinfo,
12699da29271STakashi Iwai 					struct hda_codec *codec,
12709da29271STakashi Iwai 					struct snd_pcm_substream *substream)
12719da29271STakashi Iwai {
12729da29271STakashi Iwai 	struct via_spec *spec = codec->spec;
12739da29271STakashi Iwai 	snd_hda_multi_out_dig_cleanup(codec, &spec->multiout);
127498aa34c0SHarald Welte 	return 0;
127598aa34c0SHarald Welte }
127698aa34c0SHarald Welte 
1277c577b8a1SJoseph Chan /*
1278c577b8a1SJoseph Chan  * Analog capture
1279c577b8a1SJoseph Chan  */
1280c577b8a1SJoseph Chan static int via_capture_pcm_prepare(struct hda_pcm_stream *hinfo,
1281c577b8a1SJoseph Chan 				   struct hda_codec *codec,
1282c577b8a1SJoseph Chan 				   unsigned int stream_tag,
1283c577b8a1SJoseph Chan 				   unsigned int format,
1284c577b8a1SJoseph Chan 				   struct snd_pcm_substream *substream)
1285c577b8a1SJoseph Chan {
1286c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
1287c577b8a1SJoseph Chan 
1288c577b8a1SJoseph Chan 	snd_hda_codec_setup_stream(codec, spec->adc_nids[substream->number],
1289c577b8a1SJoseph Chan 				   stream_tag, 0, format);
1290c577b8a1SJoseph Chan 	return 0;
1291c577b8a1SJoseph Chan }
1292c577b8a1SJoseph Chan 
1293c577b8a1SJoseph Chan static int via_capture_pcm_cleanup(struct hda_pcm_stream *hinfo,
1294c577b8a1SJoseph Chan 				   struct hda_codec *codec,
1295c577b8a1SJoseph Chan 				   struct snd_pcm_substream *substream)
1296c577b8a1SJoseph Chan {
1297c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
1298888afa15STakashi Iwai 	snd_hda_codec_cleanup_stream(codec, spec->adc_nids[substream->number]);
1299c577b8a1SJoseph Chan 	return 0;
1300c577b8a1SJoseph Chan }
1301c577b8a1SJoseph Chan 
1302a86a88eaSTakashi Iwai /* analog capture with dynamic ADC switching */
1303a86a88eaSTakashi Iwai static int via_dyn_adc_capture_pcm_prepare(struct hda_pcm_stream *hinfo,
1304a86a88eaSTakashi Iwai 					   struct hda_codec *codec,
1305a86a88eaSTakashi Iwai 					   unsigned int stream_tag,
1306a86a88eaSTakashi Iwai 					   unsigned int format,
1307a86a88eaSTakashi Iwai 					   struct snd_pcm_substream *substream)
1308a86a88eaSTakashi Iwai {
1309a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
1310a86a88eaSTakashi Iwai 	int adc_idx = spec->inputs[spec->cur_mux[0]].adc_idx;
1311a86a88eaSTakashi Iwai 
13123b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
1313a86a88eaSTakashi Iwai 	spec->cur_adc = spec->adc_nids[adc_idx];
1314a86a88eaSTakashi Iwai 	spec->cur_adc_stream_tag = stream_tag;
1315a86a88eaSTakashi Iwai 	spec->cur_adc_format = format;
1316a86a88eaSTakashi Iwai 	snd_hda_codec_setup_stream(codec, spec->cur_adc, stream_tag, 0, format);
13173b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
1318a86a88eaSTakashi Iwai 	return 0;
1319a86a88eaSTakashi Iwai }
1320a86a88eaSTakashi Iwai 
1321a86a88eaSTakashi Iwai static int via_dyn_adc_capture_pcm_cleanup(struct hda_pcm_stream *hinfo,
1322a86a88eaSTakashi Iwai 					   struct hda_codec *codec,
1323a86a88eaSTakashi Iwai 					   struct snd_pcm_substream *substream)
1324a86a88eaSTakashi Iwai {
1325a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
1326a86a88eaSTakashi Iwai 
13273b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
1328a86a88eaSTakashi Iwai 	snd_hda_codec_cleanup_stream(codec, spec->cur_adc);
1329a86a88eaSTakashi Iwai 	spec->cur_adc = 0;
13303b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
1331a86a88eaSTakashi Iwai 	return 0;
1332a86a88eaSTakashi Iwai }
1333a86a88eaSTakashi Iwai 
1334a86a88eaSTakashi Iwai /* re-setup the stream if running; called from input-src put */
1335a86a88eaSTakashi Iwai static bool via_dyn_adc_pcm_resetup(struct hda_codec *codec, int cur)
1336a86a88eaSTakashi Iwai {
1337a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
1338a86a88eaSTakashi Iwai 	int adc_idx = spec->inputs[cur].adc_idx;
1339a86a88eaSTakashi Iwai 	hda_nid_t adc = spec->adc_nids[adc_idx];
13403b607e3dSTakashi Iwai 	bool ret = false;
1341a86a88eaSTakashi Iwai 
13423b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
1343a86a88eaSTakashi Iwai 	if (spec->cur_adc && spec->cur_adc != adc) {
1344a86a88eaSTakashi Iwai 		/* stream is running, let's swap the current ADC */
1345a86a88eaSTakashi Iwai 		__snd_hda_codec_cleanup_stream(codec, spec->cur_adc, 1);
1346a86a88eaSTakashi Iwai 		spec->cur_adc = adc;
1347a86a88eaSTakashi Iwai 		snd_hda_codec_setup_stream(codec, adc,
1348a86a88eaSTakashi Iwai 					   spec->cur_adc_stream_tag, 0,
1349a86a88eaSTakashi Iwai 					   spec->cur_adc_format);
13503b607e3dSTakashi Iwai 		ret = true;
1351a86a88eaSTakashi Iwai 	}
13523b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
13533b607e3dSTakashi Iwai 	return ret;
1354a86a88eaSTakashi Iwai }
1355a86a88eaSTakashi Iwai 
13569af74210STakashi Iwai static const struct hda_pcm_stream via_pcm_analog_playback = {
13577eb56e84STakashi Iwai 	.substreams = 1,
1358c577b8a1SJoseph Chan 	.channels_min = 2,
1359c577b8a1SJoseph Chan 	.channels_max = 8,
13609af74210STakashi Iwai 	/* NID is set in via_build_pcms */
1361c577b8a1SJoseph Chan 	.ops = {
1362ece8d043STakashi Iwai 		.open = via_playback_multi_pcm_open,
1363ece8d043STakashi Iwai 		.close = via_playback_multi_pcm_close,
13640aa62aefSHarald Welte 		.prepare = via_playback_multi_pcm_prepare,
13650aa62aefSHarald Welte 		.cleanup = via_playback_multi_pcm_cleanup
1366c577b8a1SJoseph Chan 	},
1367c577b8a1SJoseph Chan };
1368c577b8a1SJoseph Chan 
13697eb56e84STakashi Iwai static const struct hda_pcm_stream via_pcm_hp_playback = {
13707eb56e84STakashi Iwai 	.substreams = 1,
13717eb56e84STakashi Iwai 	.channels_min = 2,
13727eb56e84STakashi Iwai 	.channels_max = 2,
13737eb56e84STakashi Iwai 	/* NID is set in via_build_pcms */
13747eb56e84STakashi Iwai 	.ops = {
13757eb56e84STakashi Iwai 		.open = via_playback_hp_pcm_open,
1376ece8d043STakashi Iwai 		.close = via_playback_hp_pcm_close,
13777eb56e84STakashi Iwai 		.prepare = via_playback_hp_pcm_prepare,
13787eb56e84STakashi Iwai 		.cleanup = via_playback_hp_pcm_cleanup
13797eb56e84STakashi Iwai 	},
13807eb56e84STakashi Iwai };
13817eb56e84STakashi Iwai 
138290dd48a1STakashi Iwai static const struct hda_pcm_stream vt1708_pcm_analog_s16_playback = {
13837eb56e84STakashi Iwai 	.substreams = 1,
1384bc9b5623STakashi Iwai 	.channels_min = 2,
1385bc9b5623STakashi Iwai 	.channels_max = 8,
13869af74210STakashi Iwai 	/* NID is set in via_build_pcms */
1387bc9b5623STakashi Iwai 	/* We got noisy outputs on the right channel on VT1708 when
1388bc9b5623STakashi Iwai 	 * 24bit samples are used.  Until any workaround is found,
1389bc9b5623STakashi Iwai 	 * disable the 24bit format, so far.
1390bc9b5623STakashi Iwai 	 */
1391bc9b5623STakashi Iwai 	.formats = SNDRV_PCM_FMTBIT_S16_LE,
1392bc9b5623STakashi Iwai 	.ops = {
1393ece8d043STakashi Iwai 		.open = via_playback_multi_pcm_open,
1394ece8d043STakashi Iwai 		.close = via_playback_multi_pcm_close,
1395c873cc25SLydia Wang 		.prepare = via_playback_multi_pcm_prepare,
1396c873cc25SLydia Wang 		.cleanup = via_playback_multi_pcm_cleanup
1397bc9b5623STakashi Iwai 	},
1398bc9b5623STakashi Iwai };
1399bc9b5623STakashi Iwai 
14009af74210STakashi Iwai static const struct hda_pcm_stream via_pcm_analog_capture = {
14017eb56e84STakashi Iwai 	.substreams = 1, /* will be changed in via_build_pcms() */
1402c577b8a1SJoseph Chan 	.channels_min = 2,
1403c577b8a1SJoseph Chan 	.channels_max = 2,
14049af74210STakashi Iwai 	/* NID is set in via_build_pcms */
1405c577b8a1SJoseph Chan 	.ops = {
1406c577b8a1SJoseph Chan 		.prepare = via_capture_pcm_prepare,
1407c577b8a1SJoseph Chan 		.cleanup = via_capture_pcm_cleanup
1408c577b8a1SJoseph Chan 	},
1409c577b8a1SJoseph Chan };
1410c577b8a1SJoseph Chan 
1411a86a88eaSTakashi Iwai static const struct hda_pcm_stream via_pcm_dyn_adc_analog_capture = {
1412a86a88eaSTakashi Iwai 	.substreams = 1,
1413a86a88eaSTakashi Iwai 	.channels_min = 2,
1414a86a88eaSTakashi Iwai 	.channels_max = 2,
1415a86a88eaSTakashi Iwai 	/* NID is set in via_build_pcms */
1416a86a88eaSTakashi Iwai 	.ops = {
1417a86a88eaSTakashi Iwai 		.prepare = via_dyn_adc_capture_pcm_prepare,
1418a86a88eaSTakashi Iwai 		.cleanup = via_dyn_adc_capture_pcm_cleanup,
1419a86a88eaSTakashi Iwai 	},
1420a86a88eaSTakashi Iwai };
1421a86a88eaSTakashi Iwai 
14229af74210STakashi Iwai static const struct hda_pcm_stream via_pcm_digital_playback = {
1423c577b8a1SJoseph Chan 	.substreams = 1,
1424c577b8a1SJoseph Chan 	.channels_min = 2,
1425c577b8a1SJoseph Chan 	.channels_max = 2,
1426c577b8a1SJoseph Chan 	/* NID is set in via_build_pcms */
1427c577b8a1SJoseph Chan 	.ops = {
1428c577b8a1SJoseph Chan 		.open = via_dig_playback_pcm_open,
14296b97eb45STakashi Iwai 		.close = via_dig_playback_pcm_close,
14309da29271STakashi Iwai 		.prepare = via_dig_playback_pcm_prepare,
14319da29271STakashi Iwai 		.cleanup = via_dig_playback_pcm_cleanup
1432c577b8a1SJoseph Chan 	},
1433c577b8a1SJoseph Chan };
1434c577b8a1SJoseph Chan 
14359af74210STakashi Iwai static const struct hda_pcm_stream via_pcm_digital_capture = {
1436c577b8a1SJoseph Chan 	.substreams = 1,
1437c577b8a1SJoseph Chan 	.channels_min = 2,
1438c577b8a1SJoseph Chan 	.channels_max = 2,
1439c577b8a1SJoseph Chan };
1440c577b8a1SJoseph Chan 
1441370bafbdSTakashi Iwai /*
1442370bafbdSTakashi Iwai  * slave controls for virtual master
1443370bafbdSTakashi Iwai  */
14449322ca54STakashi Iwai static const char * const via_slave_pfxs[] = {
14459322ca54STakashi Iwai 	"Front", "Surround", "Center", "LFE", "Side",
1446f37bc7a8STakashi Iwai 	"Headphone", "Speaker", "Bass Speaker",
1447370bafbdSTakashi Iwai 	NULL,
1448370bafbdSTakashi Iwai };
1449370bafbdSTakashi Iwai 
1450c577b8a1SJoseph Chan static int via_build_controls(struct hda_codec *codec)
1451c577b8a1SJoseph Chan {
1452c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
14535b0cb1d8SJaroslav Kysela 	struct snd_kcontrol *kctl;
14545b0cb1d8SJaroslav Kysela 	int err, i;
1455c577b8a1SJoseph Chan 
1456b5bcc189STakashi Iwai 	spec->no_pin_power_ctl = 1;
145724088a58STakashi Iwai 	if (spec->set_widgets_power_state)
145824088a58STakashi Iwai 		if (!via_clone_control(spec, &via_pin_power_ctl_enum))
145924088a58STakashi Iwai 			return -ENOMEM;
146024088a58STakashi Iwai 
1461c577b8a1SJoseph Chan 	for (i = 0; i < spec->num_mixers; i++) {
1462c577b8a1SJoseph Chan 		err = snd_hda_add_new_ctls(codec, spec->mixers[i]);
1463c577b8a1SJoseph Chan 		if (err < 0)
1464c577b8a1SJoseph Chan 			return err;
1465c577b8a1SJoseph Chan 	}
1466c577b8a1SJoseph Chan 
1467c577b8a1SJoseph Chan 	if (spec->multiout.dig_out_nid) {
1468c577b8a1SJoseph Chan 		err = snd_hda_create_spdif_out_ctls(codec,
146974b654c9SStephen Warren 						    spec->multiout.dig_out_nid,
1470c577b8a1SJoseph Chan 						    spec->multiout.dig_out_nid);
1471c577b8a1SJoseph Chan 		if (err < 0)
1472c577b8a1SJoseph Chan 			return err;
14739a08160bSTakashi Iwai 		err = snd_hda_create_spdif_share_sw(codec,
14749a08160bSTakashi Iwai 						    &spec->multiout);
14759a08160bSTakashi Iwai 		if (err < 0)
14769a08160bSTakashi Iwai 			return err;
14779a08160bSTakashi Iwai 		spec->multiout.share_spdif = 1;
1478c577b8a1SJoseph Chan 	}
1479c577b8a1SJoseph Chan 	if (spec->dig_in_nid) {
1480c577b8a1SJoseph Chan 		err = snd_hda_create_spdif_in_ctls(codec, spec->dig_in_nid);
1481c577b8a1SJoseph Chan 		if (err < 0)
1482c577b8a1SJoseph Chan 			return err;
1483c577b8a1SJoseph Chan 	}
148417314379SLydia Wang 
1485370bafbdSTakashi Iwai 	/* if we have no master control, let's create it */
1486370bafbdSTakashi Iwai 	if (!snd_hda_find_mixer_ctl(codec, "Master Playback Volume")) {
1487370bafbdSTakashi Iwai 		unsigned int vmaster_tlv[4];
1488370bafbdSTakashi Iwai 		snd_hda_set_vmaster_tlv(codec, spec->multiout.dac_nids[0],
1489370bafbdSTakashi Iwai 					HDA_OUTPUT, vmaster_tlv);
1490370bafbdSTakashi Iwai 		err = snd_hda_add_vmaster(codec, "Master Playback Volume",
14919322ca54STakashi Iwai 					  vmaster_tlv, via_slave_pfxs,
14929322ca54STakashi Iwai 					  "Playback Volume");
1493370bafbdSTakashi Iwai 		if (err < 0)
1494370bafbdSTakashi Iwai 			return err;
1495370bafbdSTakashi Iwai 	}
1496370bafbdSTakashi Iwai 	if (!snd_hda_find_mixer_ctl(codec, "Master Playback Switch")) {
1497370bafbdSTakashi Iwai 		err = snd_hda_add_vmaster(codec, "Master Playback Switch",
14989322ca54STakashi Iwai 					  NULL, via_slave_pfxs,
14999322ca54STakashi Iwai 					  "Playback Switch");
1500370bafbdSTakashi Iwai 		if (err < 0)
1501370bafbdSTakashi Iwai 			return err;
1502370bafbdSTakashi Iwai 	}
1503370bafbdSTakashi Iwai 
15045b0cb1d8SJaroslav Kysela 	/* assign Capture Source enums to NID */
15055b0cb1d8SJaroslav Kysela 	kctl = snd_hda_find_mixer_ctl(codec, "Input Source");
15065b0cb1d8SJaroslav Kysela 	for (i = 0; kctl && i < kctl->count; i++) {
150777e314f7STakashi Iwai 		if (!spec->mux_nids[i])
150877e314f7STakashi Iwai 			continue;
150921949f00STakashi Iwai 		err = snd_hda_add_nid(codec, kctl, i, spec->mux_nids[i]);
15105b0cb1d8SJaroslav Kysela 		if (err < 0)
15115b0cb1d8SJaroslav Kysela 			return err;
15125b0cb1d8SJaroslav Kysela 	}
15135b0cb1d8SJaroslav Kysela 
1514603c4019STakashi Iwai 	via_free_kctls(codec); /* no longer needed */
151501a61e12STakashi Iwai 
151601a61e12STakashi Iwai 	err = snd_hda_jack_add_kctls(codec, &spec->autocfg);
151701a61e12STakashi Iwai 	if (err < 0)
151801a61e12STakashi Iwai 		return err;
151901a61e12STakashi Iwai 
1520c577b8a1SJoseph Chan 	return 0;
1521c577b8a1SJoseph Chan }
1522c577b8a1SJoseph Chan 
1523c577b8a1SJoseph Chan static int via_build_pcms(struct hda_codec *codec)
1524c577b8a1SJoseph Chan {
1525c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
1526c577b8a1SJoseph Chan 	struct hda_pcm *info = spec->pcm_rec;
1527c577b8a1SJoseph Chan 
1528a5973103STakashi Iwai 	codec->num_pcms = 0;
1529c577b8a1SJoseph Chan 	codec->pcm_info = info;
1530c577b8a1SJoseph Chan 
1531a5973103STakashi Iwai 	if (spec->multiout.num_dacs || spec->num_adc_nids) {
1532a5973103STakashi Iwai 		snprintf(spec->stream_name_analog,
1533a5973103STakashi Iwai 			 sizeof(spec->stream_name_analog),
153482673bc8STakashi Iwai 			 "%s Analog", codec->chip_name);
1535c577b8a1SJoseph Chan 		info->name = spec->stream_name_analog;
15369af74210STakashi Iwai 
1537a5973103STakashi Iwai 		if (spec->multiout.num_dacs) {
15389af74210STakashi Iwai 			if (!spec->stream_analog_playback)
1539a5973103STakashi Iwai 				spec->stream_analog_playback =
1540a5973103STakashi Iwai 					&via_pcm_analog_playback;
1541377ff31aSLydia Wang 			info->stream[SNDRV_PCM_STREAM_PLAYBACK] =
15429af74210STakashi Iwai 				*spec->stream_analog_playback;
1543377ff31aSLydia Wang 			info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid =
1544377ff31aSLydia Wang 				spec->multiout.dac_nids[0];
1545c577b8a1SJoseph Chan 			info->stream[SNDRV_PCM_STREAM_PLAYBACK].channels_max =
1546c577b8a1SJoseph Chan 				spec->multiout.max_channels;
1547ee81abb6STakashi Iwai 			if (spec->autocfg.line_out_type == AUTO_PIN_SPEAKER_OUT
1548ee81abb6STakashi Iwai 			    && spec->autocfg.line_outs == 2)
1549ee81abb6STakashi Iwai 				info->stream[SNDRV_PCM_STREAM_PLAYBACK].chmap =
1550ee81abb6STakashi Iwai 					snd_pcm_2_1_chmaps;
1551a5973103STakashi Iwai 		}
15529af74210STakashi Iwai 
1553a86a88eaSTakashi Iwai 		if (!spec->stream_analog_capture) {
1554a86a88eaSTakashi Iwai 			if (spec->dyn_adc_switch)
1555a86a88eaSTakashi Iwai 				spec->stream_analog_capture =
1556a86a88eaSTakashi Iwai 					&via_pcm_dyn_adc_analog_capture;
1557a86a88eaSTakashi Iwai 			else
1558a5973103STakashi Iwai 				spec->stream_analog_capture =
1559a5973103STakashi Iwai 					&via_pcm_analog_capture;
1560a86a88eaSTakashi Iwai 		}
1561a5973103STakashi Iwai 		if (spec->num_adc_nids) {
15629af74210STakashi Iwai 			info->stream[SNDRV_PCM_STREAM_CAPTURE] =
15639af74210STakashi Iwai 				*spec->stream_analog_capture;
1564a5973103STakashi Iwai 			info->stream[SNDRV_PCM_STREAM_CAPTURE].nid =
1565a5973103STakashi Iwai 				spec->adc_nids[0];
1566a86a88eaSTakashi Iwai 			if (!spec->dyn_adc_switch)
15679af74210STakashi Iwai 				info->stream[SNDRV_PCM_STREAM_CAPTURE].substreams =
15689af74210STakashi Iwai 					spec->num_adc_nids;
1569a5973103STakashi Iwai 		}
1570c577b8a1SJoseph Chan 		codec->num_pcms++;
1571c577b8a1SJoseph Chan 		info++;
1572a5973103STakashi Iwai 	}
1573a5973103STakashi Iwai 
1574a5973103STakashi Iwai 	if (spec->multiout.dig_out_nid || spec->dig_in_nid) {
157582673bc8STakashi Iwai 		snprintf(spec->stream_name_digital,
157682673bc8STakashi Iwai 			 sizeof(spec->stream_name_digital),
157782673bc8STakashi Iwai 			 "%s Digital", codec->chip_name);
1578c577b8a1SJoseph Chan 		info->name = spec->stream_name_digital;
15797ba72ba1STakashi Iwai 		info->pcm_type = HDA_PCM_TYPE_SPDIF;
1580c577b8a1SJoseph Chan 		if (spec->multiout.dig_out_nid) {
15819af74210STakashi Iwai 			if (!spec->stream_digital_playback)
15829af74210STakashi Iwai 				spec->stream_digital_playback =
15839af74210STakashi Iwai 					&via_pcm_digital_playback;
1584c577b8a1SJoseph Chan 			info->stream[SNDRV_PCM_STREAM_PLAYBACK] =
15859af74210STakashi Iwai 				*spec->stream_digital_playback;
1586c577b8a1SJoseph Chan 			info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid =
1587c577b8a1SJoseph Chan 				spec->multiout.dig_out_nid;
1588c577b8a1SJoseph Chan 		}
1589c577b8a1SJoseph Chan 		if (spec->dig_in_nid) {
15909af74210STakashi Iwai 			if (!spec->stream_digital_capture)
15919af74210STakashi Iwai 				spec->stream_digital_capture =
15929af74210STakashi Iwai 					&via_pcm_digital_capture;
1593c577b8a1SJoseph Chan 			info->stream[SNDRV_PCM_STREAM_CAPTURE] =
15949af74210STakashi Iwai 				*spec->stream_digital_capture;
1595c577b8a1SJoseph Chan 			info->stream[SNDRV_PCM_STREAM_CAPTURE].nid =
1596c577b8a1SJoseph Chan 				spec->dig_in_nid;
1597c577b8a1SJoseph Chan 		}
1598a5973103STakashi Iwai 		codec->num_pcms++;
1599a5973103STakashi Iwai 		info++;
1600c577b8a1SJoseph Chan 	}
1601c577b8a1SJoseph Chan 
1602ece8d043STakashi Iwai 	if (spec->hp_dac_nid) {
16037eb56e84STakashi Iwai 		snprintf(spec->stream_name_hp, sizeof(spec->stream_name_hp),
16047eb56e84STakashi Iwai 			 "%s HP", codec->chip_name);
16057eb56e84STakashi Iwai 		info->name = spec->stream_name_hp;
16067eb56e84STakashi Iwai 		info->stream[SNDRV_PCM_STREAM_PLAYBACK] = via_pcm_hp_playback;
16077eb56e84STakashi Iwai 		info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid =
1608ece8d043STakashi Iwai 			spec->hp_dac_nid;
1609a5973103STakashi Iwai 		codec->num_pcms++;
1610a5973103STakashi Iwai 		info++;
16117eb56e84STakashi Iwai 	}
1612c577b8a1SJoseph Chan 	return 0;
1613c577b8a1SJoseph Chan }
1614c577b8a1SJoseph Chan 
1615c577b8a1SJoseph Chan static void via_free(struct hda_codec *codec)
1616c577b8a1SJoseph Chan {
1617c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
1618c577b8a1SJoseph Chan 
1619c577b8a1SJoseph Chan 	if (!spec)
1620c577b8a1SJoseph Chan 		return;
1621c577b8a1SJoseph Chan 
1622603c4019STakashi Iwai 	via_free_kctls(codec);
16231f2e99feSLydia Wang 	vt1708_stop_hp_work(spec);
1624a86a88eaSTakashi Iwai 	kfree(spec->bind_cap_vol);
1625a86a88eaSTakashi Iwai 	kfree(spec->bind_cap_sw);
16267819d1c7STakashi Iwai 	snd_hda_gen_free(&spec->gen);
1627a86a88eaSTakashi Iwai 	kfree(spec);
1628c577b8a1SJoseph Chan }
1629c577b8a1SJoseph Chan 
163064be285bSTakashi Iwai /* mute/unmute outputs */
163164be285bSTakashi Iwai static void toggle_output_mutes(struct hda_codec *codec, int num_pins,
163264be285bSTakashi Iwai 				hda_nid_t *pins, bool mute)
163364be285bSTakashi Iwai {
163464be285bSTakashi Iwai 	int i;
163594994734STakashi Iwai 	for (i = 0; i < num_pins; i++) {
163694994734STakashi Iwai 		unsigned int parm = snd_hda_codec_read(codec, pins[i], 0,
163794994734STakashi Iwai 					  AC_VERB_GET_PIN_WIDGET_CONTROL, 0);
163894994734STakashi Iwai 		if (parm & AC_PINCTL_IN_EN)
163994994734STakashi Iwai 			continue;
164094994734STakashi Iwai 		if (mute)
164194994734STakashi Iwai 			parm &= ~AC_PINCTL_OUT_EN;
164294994734STakashi Iwai 		else
164394994734STakashi Iwai 			parm |= AC_PINCTL_OUT_EN;
1644cdd03cedSTakashi Iwai 		snd_hda_set_pin_ctl(codec, pins[i], parm);
164594994734STakashi Iwai 	}
164664be285bSTakashi Iwai }
164764be285bSTakashi Iwai 
16484a918ffeSTakashi Iwai /* mute internal speaker if line-out is plugged */
16494a918ffeSTakashi Iwai static void via_line_automute(struct hda_codec *codec, int present)
16504a918ffeSTakashi Iwai {
16514a918ffeSTakashi Iwai 	struct via_spec *spec = codec->spec;
16524a918ffeSTakashi Iwai 
16534a918ffeSTakashi Iwai 	if (!spec->autocfg.speaker_outs)
16544a918ffeSTakashi Iwai 		return;
16554a918ffeSTakashi Iwai 	if (!present)
16564a918ffeSTakashi Iwai 		present = snd_hda_jack_detect(codec,
16574a918ffeSTakashi Iwai 					      spec->autocfg.line_out_pins[0]);
16584a918ffeSTakashi Iwai 	toggle_output_mutes(codec, spec->autocfg.speaker_outs,
16594a918ffeSTakashi Iwai 			    spec->autocfg.speaker_pins,
16604a918ffeSTakashi Iwai 			    present);
16614a918ffeSTakashi Iwai }
16624a918ffeSTakashi Iwai 
166369e52a80SHarald Welte /* mute internal speaker if HP is plugged */
166469e52a80SHarald Welte static void via_hp_automute(struct hda_codec *codec)
166569e52a80SHarald Welte {
16664a918ffeSTakashi Iwai 	int present = 0;
16676e969d91STakashi Iwai 	int nums;
166869e52a80SHarald Welte 	struct via_spec *spec = codec->spec;
166969e52a80SHarald Welte 
1670187d333eSTakashi Iwai 	if (!spec->hp_independent_mode && spec->autocfg.hp_pins[0] &&
1671cf55e904SHerton Ronaldo Krzesinski 	    (spec->codec_type != VT1708 || spec->vt1708_jack_detect) &&
1672cf55e904SHerton Ronaldo Krzesinski 	    is_jack_detectable(codec, spec->autocfg.hp_pins[0]))
1673d56757abSTakashi Iwai 		present = snd_hda_jack_detect(codec, spec->autocfg.hp_pins[0]);
16746e969d91STakashi Iwai 
1675f2b1c9f0STakashi Iwai 	if (spec->smart51_enabled)
1676f2b1c9f0STakashi Iwai 		nums = spec->autocfg.line_outs + spec->smart51_nums;
1677f2b1c9f0STakashi Iwai 	else
1678f2b1c9f0STakashi Iwai 		nums = spec->autocfg.line_outs;
16796e969d91STakashi Iwai 	toggle_output_mutes(codec, nums, spec->autocfg.line_out_pins, present);
16806e969d91STakashi Iwai 
16814a918ffeSTakashi Iwai 	via_line_automute(codec, present);
1682f3db423dSLydia Wang }
1683f3db423dSLydia Wang 
16842a43952aSTakashi Iwai #ifdef CONFIG_PM
168568cb2b55STakashi Iwai static int via_suspend(struct hda_codec *codec)
16861f2e99feSLydia Wang {
16871f2e99feSLydia Wang 	struct via_spec *spec = codec->spec;
16881f2e99feSLydia Wang 	vt1708_stop_hp_work(spec);
168994c142a1SDavid Henningsson 
169094c142a1SDavid Henningsson 	if (spec->codec_type == VT1802) {
169194c142a1SDavid Henningsson 		/* Fix pop noise on headphones */
169294c142a1SDavid Henningsson 		int i;
169394c142a1SDavid Henningsson 		for (i = 0; i < spec->autocfg.hp_outs; i++)
169494c142a1SDavid Henningsson 			snd_hda_set_pin_ctl(codec, spec->autocfg.hp_pins[i], 0);
169594c142a1SDavid Henningsson 	}
169694c142a1SDavid Henningsson 
16971f2e99feSLydia Wang 	return 0;
16981f2e99feSLydia Wang }
16991f2e99feSLydia Wang #endif
17001f2e99feSLydia Wang 
170183012a7cSTakashi Iwai #ifdef CONFIG_PM
1702cb53c626STakashi Iwai static int via_check_power_status(struct hda_codec *codec, hda_nid_t nid)
1703cb53c626STakashi Iwai {
1704cb53c626STakashi Iwai 	struct via_spec *spec = codec->spec;
1705cb53c626STakashi Iwai 	return snd_hda_check_amp_list_power(codec, &spec->loopback, nid);
1706cb53c626STakashi Iwai }
1707cb53c626STakashi Iwai #endif
1708cb53c626STakashi Iwai 
1709c577b8a1SJoseph Chan /*
1710c577b8a1SJoseph Chan  */
17115d41762aSTakashi Iwai 
17125d41762aSTakashi Iwai static int via_init(struct hda_codec *codec);
17135d41762aSTakashi Iwai 
171490dd48a1STakashi Iwai static const struct hda_codec_ops via_patch_ops = {
1715c577b8a1SJoseph Chan 	.build_controls = via_build_controls,
1716c577b8a1SJoseph Chan 	.build_pcms = via_build_pcms,
1717c577b8a1SJoseph Chan 	.init = via_init,
1718c577b8a1SJoseph Chan 	.free = via_free,
17194e2d16d3SDavid Henningsson 	.unsol_event = snd_hda_jack_unsol_event,
17202a43952aSTakashi Iwai #ifdef CONFIG_PM
17211f2e99feSLydia Wang 	.suspend = via_suspend,
1722cb53c626STakashi Iwai 	.check_power_status = via_check_power_status,
1723cb53c626STakashi Iwai #endif
1724c577b8a1SJoseph Chan };
1725c577b8a1SJoseph Chan 
17264a79616dSTakashi Iwai static bool is_empty_dac(struct hda_codec *codec, hda_nid_t dac)
1727c577b8a1SJoseph Chan {
17284a79616dSTakashi Iwai 	struct via_spec *spec = codec->spec;
17294a79616dSTakashi Iwai 	int i;
17304a79616dSTakashi Iwai 
17314a79616dSTakashi Iwai 	for (i = 0; i < spec->multiout.num_dacs; i++) {
17324a79616dSTakashi Iwai 		if (spec->multiout.dac_nids[i] == dac)
17334a79616dSTakashi Iwai 			return false;
17344a79616dSTakashi Iwai 	}
1735ece8d043STakashi Iwai 	if (spec->hp_dac_nid == dac)
17364a79616dSTakashi Iwai 		return false;
17374a79616dSTakashi Iwai 	return true;
17384a79616dSTakashi Iwai }
17394a79616dSTakashi Iwai 
17408e3679dcSTakashi Iwai static bool __parse_output_path(struct hda_codec *codec, hda_nid_t nid,
17413214b966STakashi Iwai 				hda_nid_t target_dac, int with_aa_mix,
17423214b966STakashi Iwai 				struct nid_path *path, int depth)
17434a79616dSTakashi Iwai {
17443214b966STakashi Iwai 	struct via_spec *spec = codec->spec;
17454a79616dSTakashi Iwai 	hda_nid_t conn[8];
17464a79616dSTakashi Iwai 	int i, nums;
17474a79616dSTakashi Iwai 
17483214b966STakashi Iwai 	if (nid == spec->aa_mix_nid) {
17493214b966STakashi Iwai 		if (!with_aa_mix)
17503214b966STakashi Iwai 			return false;
17513214b966STakashi Iwai 		with_aa_mix = 2; /* mark aa-mix is included */
17523214b966STakashi Iwai 	}
17533214b966STakashi Iwai 
17544a79616dSTakashi Iwai 	nums = snd_hda_get_connections(codec, nid, conn, ARRAY_SIZE(conn));
17554a79616dSTakashi Iwai 	for (i = 0; i < nums; i++) {
17564a79616dSTakashi Iwai 		if (get_wcaps_type(get_wcaps(codec, conn[i])) != AC_WID_AUD_OUT)
17574a79616dSTakashi Iwai 			continue;
17583214b966STakashi Iwai 		if (conn[i] == target_dac || is_empty_dac(codec, conn[i])) {
17593214b966STakashi Iwai 			/* aa-mix is requested but not included? */
17603214b966STakashi Iwai 			if (!(spec->aa_mix_nid && with_aa_mix == 1))
176109a9ad69STakashi Iwai 				goto found;
17624a79616dSTakashi Iwai 		}
17633214b966STakashi Iwai 	}
17648e3679dcSTakashi Iwai 	if (depth >= MAX_NID_PATH_DEPTH)
17654a79616dSTakashi Iwai 		return false;
17664a79616dSTakashi Iwai 	for (i = 0; i < nums; i++) {
17674a79616dSTakashi Iwai 		unsigned int type;
17684a79616dSTakashi Iwai 		type = get_wcaps_type(get_wcaps(codec, conn[i]));
17693214b966STakashi Iwai 		if (type == AC_WID_AUD_OUT)
17704a79616dSTakashi Iwai 			continue;
17718e3679dcSTakashi Iwai 		if (__parse_output_path(codec, conn[i], target_dac,
17723214b966STakashi Iwai 					with_aa_mix, path, depth + 1))
177309a9ad69STakashi Iwai 			goto found;
17744a79616dSTakashi Iwai 	}
17754a79616dSTakashi Iwai 	return false;
177609a9ad69STakashi Iwai 
177709a9ad69STakashi Iwai  found:
177809a9ad69STakashi Iwai 	path->path[path->depth] = conn[i];
177909a9ad69STakashi Iwai 	path->idx[path->depth] = i;
178009a9ad69STakashi Iwai 	if (nums > 1 && get_wcaps_type(get_wcaps(codec, nid)) != AC_WID_AUD_MIX)
178109a9ad69STakashi Iwai 		path->multi[path->depth] = 1;
178209a9ad69STakashi Iwai 	path->depth++;
178309a9ad69STakashi Iwai 	return true;
17844a79616dSTakashi Iwai }
17854a79616dSTakashi Iwai 
17868e3679dcSTakashi Iwai static bool parse_output_path(struct hda_codec *codec, hda_nid_t nid,
17873214b966STakashi Iwai 			      hda_nid_t target_dac, int with_aa_mix,
17883214b966STakashi Iwai 			      struct nid_path *path)
17898e3679dcSTakashi Iwai {
17903214b966STakashi Iwai 	if (__parse_output_path(codec, nid, target_dac, with_aa_mix, path, 1)) {
17918e3679dcSTakashi Iwai 		path->path[path->depth] = nid;
17928e3679dcSTakashi Iwai 		path->depth++;
17933214b966STakashi Iwai 		snd_printdd("output-path: depth=%d, %02x/%02x/%02x/%02x/%02x\n",
17943214b966STakashi Iwai 			    path->depth, path->path[0], path->path[1],
17953214b966STakashi Iwai 			    path->path[2], path->path[3], path->path[4]);
17968e3679dcSTakashi Iwai 		return true;
17978e3679dcSTakashi Iwai 	}
17988e3679dcSTakashi Iwai 	return false;
17998e3679dcSTakashi Iwai }
18008e3679dcSTakashi Iwai 
18014a79616dSTakashi Iwai static int via_auto_fill_dac_nids(struct hda_codec *codec)
18024a79616dSTakashi Iwai {
18034a79616dSTakashi Iwai 	struct via_spec *spec = codec->spec;
18044a79616dSTakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
18055b376195STakashi Iwai 	int i;
1806c577b8a1SJoseph Chan 	hda_nid_t nid;
1807c577b8a1SJoseph Chan 
18085b376195STakashi Iwai 	spec->multiout.num_dacs = 0;
1809c577b8a1SJoseph Chan 	spec->multiout.dac_nids = spec->private_dac_nids;
18104a79616dSTakashi Iwai 	for (i = 0; i < cfg->line_outs; i++) {
18113214b966STakashi Iwai 		hda_nid_t dac = 0;
1812c577b8a1SJoseph Chan 		nid = cfg->line_out_pins[i];
18134a79616dSTakashi Iwai 		if (!nid)
18144a79616dSTakashi Iwai 			continue;
18153214b966STakashi Iwai 		if (parse_output_path(codec, nid, 0, 0, &spec->out_path[i]))
18163214b966STakashi Iwai 			dac = spec->out_path[i].path[0];
18173214b966STakashi Iwai 		if (!i && parse_output_path(codec, nid, dac, 1,
18183214b966STakashi Iwai 					    &spec->out_mix_path))
18193214b966STakashi Iwai 			dac = spec->out_mix_path.path[0];
18205b376195STakashi Iwai 		if (dac)
18215b376195STakashi Iwai 			spec->private_dac_nids[spec->multiout.num_dacs++] = dac;
18225c9a5615SLydia Wang 	}
18233214b966STakashi Iwai 	if (!spec->out_path[0].depth && spec->out_mix_path.depth) {
18243214b966STakashi Iwai 		spec->out_path[0] = spec->out_mix_path;
18253214b966STakashi Iwai 		spec->out_mix_path.depth = 0;
18263214b966STakashi Iwai 	}
1827c577b8a1SJoseph Chan 	return 0;
1828c577b8a1SJoseph Chan }
1829c577b8a1SJoseph Chan 
18304a79616dSTakashi Iwai static int create_ch_ctls(struct hda_codec *codec, const char *pfx,
183109a9ad69STakashi Iwai 			  int chs, bool check_dac, struct nid_path *path)
1832c577b8a1SJoseph Chan {
18334a79616dSTakashi Iwai 	struct via_spec *spec = codec->spec;
1834c577b8a1SJoseph Chan 	char name[32];
183509a9ad69STakashi Iwai 	hda_nid_t dac, pin, sel, nid;
183609a9ad69STakashi Iwai 	int err;
1837a934d5a9STakashi Iwai 
183809a9ad69STakashi Iwai 	dac = check_dac ? path->path[0] : 0;
183909a9ad69STakashi Iwai 	pin = path->path[path->depth - 1];
184009a9ad69STakashi Iwai 	sel = path->depth > 1 ? path->path[1] : 0;
1841c577b8a1SJoseph Chan 
18428df2a312STakashi Iwai 	if (dac && check_amp_caps(codec, dac, HDA_OUTPUT, AC_AMPCAP_NUM_STEPS))
18434a79616dSTakashi Iwai 		nid = dac;
18448df2a312STakashi Iwai 	else if (check_amp_caps(codec, pin, HDA_OUTPUT, AC_AMPCAP_NUM_STEPS))
18454a79616dSTakashi Iwai 		nid = pin;
1846a934d5a9STakashi Iwai 	else if (check_amp_caps(codec, sel, HDA_OUTPUT, AC_AMPCAP_NUM_STEPS))
1847a934d5a9STakashi Iwai 		nid = sel;
18484a79616dSTakashi Iwai 	else
18494a79616dSTakashi Iwai 		nid = 0;
18504a79616dSTakashi Iwai 	if (nid) {
18514a79616dSTakashi Iwai 		sprintf(name, "%s Playback Volume", pfx);
1852c577b8a1SJoseph Chan 		err = via_add_control(spec, VIA_CTL_WIDGET_VOL, name,
1853a00a5fadSLydia Wang 			      HDA_COMPOSE_AMP_VAL(nid, chs, 0, HDA_OUTPUT));
1854c577b8a1SJoseph Chan 		if (err < 0)
1855c577b8a1SJoseph Chan 			return err;
185609a9ad69STakashi Iwai 		path->vol_ctl = nid;
1857c577b8a1SJoseph Chan 	}
18584a79616dSTakashi Iwai 
18598df2a312STakashi Iwai 	if (dac && check_amp_caps(codec, dac, HDA_OUTPUT, AC_AMPCAP_MUTE))
18604a79616dSTakashi Iwai 		nid = dac;
18618df2a312STakashi Iwai 	else if (check_amp_caps(codec, pin, HDA_OUTPUT, AC_AMPCAP_MUTE))
18624a79616dSTakashi Iwai 		nid = pin;
1863a934d5a9STakashi Iwai 	else if (check_amp_caps(codec, sel, HDA_OUTPUT, AC_AMPCAP_MUTE))
1864a934d5a9STakashi Iwai 		nid = sel;
18654a79616dSTakashi Iwai 	else
18664a79616dSTakashi Iwai 		nid = 0;
18674a79616dSTakashi Iwai 	if (nid) {
18684a79616dSTakashi Iwai 		sprintf(name, "%s Playback Switch", pfx);
18694a79616dSTakashi Iwai 		err = via_add_control(spec, VIA_CTL_WIDGET_MUTE, name,
18704a79616dSTakashi Iwai 			      HDA_COMPOSE_AMP_VAL(nid, chs, 0, HDA_OUTPUT));
18714a79616dSTakashi Iwai 		if (err < 0)
18724a79616dSTakashi Iwai 			return err;
187309a9ad69STakashi Iwai 		path->mute_ctl = nid;
18744a79616dSTakashi Iwai 	}
18754a79616dSTakashi Iwai 	return 0;
18764a79616dSTakashi Iwai }
18774a79616dSTakashi Iwai 
1878f4a7828bSTakashi Iwai static void mangle_smart51(struct hda_codec *codec)
1879f4a7828bSTakashi Iwai {
1880f4a7828bSTakashi Iwai 	struct via_spec *spec = codec->spec;
1881f4a7828bSTakashi Iwai 	struct auto_pin_cfg *cfg = &spec->autocfg;
18820f98c24bSTakashi Iwai 	struct auto_pin_cfg_item *ins = cfg->inputs;
18830f98c24bSTakashi Iwai 	int i, j, nums, attr;
18840f98c24bSTakashi Iwai 	int pins[AUTO_CFG_MAX_INS];
1885f4a7828bSTakashi Iwai 
18860f98c24bSTakashi Iwai 	for (attr = INPUT_PIN_ATTR_REAR; attr >= INPUT_PIN_ATTR_NORMAL; attr--) {
18870f98c24bSTakashi Iwai 		nums = 0;
1888f4a7828bSTakashi Iwai 		for (i = 0; i < cfg->num_inputs; i++) {
18890f98c24bSTakashi Iwai 			unsigned int def;
18900f98c24bSTakashi Iwai 			if (ins[i].type > AUTO_PIN_LINE_IN)
18910f98c24bSTakashi Iwai 				continue;
18920f98c24bSTakashi Iwai 			def = snd_hda_codec_get_pincfg(codec, ins[i].pin);
18930f98c24bSTakashi Iwai 			if (snd_hda_get_input_pin_attr(def) != attr)
18940f98c24bSTakashi Iwai 				continue;
18950f98c24bSTakashi Iwai 			for (j = 0; j < nums; j++)
18960f98c24bSTakashi Iwai 				if (ins[pins[j]].type < ins[i].type) {
18970f98c24bSTakashi Iwai 					memmove(pins + j + 1, pins + j,
189821d45d2bSTakashi Iwai 						(nums - j) * sizeof(int));
18990f98c24bSTakashi Iwai 					break;
19000f98c24bSTakashi Iwai 				}
19010f98c24bSTakashi Iwai 			pins[j] = i;
1902e3d7a143STakashi Iwai 			nums++;
1903e3d7a143STakashi Iwai 		}
1904e3d7a143STakashi Iwai 		if (cfg->line_outs + nums < 3)
1905f4a7828bSTakashi Iwai 			continue;
19060f98c24bSTakashi Iwai 		for (i = 0; i < nums; i++) {
19070f98c24bSTakashi Iwai 			hda_nid_t pin = ins[pins[i]].pin;
19080f98c24bSTakashi Iwai 			spec->smart51_pins[spec->smart51_nums++] = pin;
19090f98c24bSTakashi Iwai 			cfg->line_out_pins[cfg->line_outs++] = pin;
1910f4a7828bSTakashi Iwai 			if (cfg->line_outs == 3)
1911f4a7828bSTakashi Iwai 				break;
1912f4a7828bSTakashi Iwai 		}
19130f98c24bSTakashi Iwai 		return;
19140f98c24bSTakashi Iwai 	}
1915f4a7828bSTakashi Iwai }
1916f4a7828bSTakashi Iwai 
1917020066d1STakashi Iwai static void copy_path_mixer_ctls(struct nid_path *dst, struct nid_path *src)
1918020066d1STakashi Iwai {
1919020066d1STakashi Iwai 	dst->vol_ctl = src->vol_ctl;
1920020066d1STakashi Iwai 	dst->mute_ctl = src->mute_ctl;
1921020066d1STakashi Iwai }
1922020066d1STakashi Iwai 
19234a79616dSTakashi Iwai /* add playback controls from the parsed DAC table */
19244a79616dSTakashi Iwai static int via_auto_create_multi_out_ctls(struct hda_codec *codec)
19254a79616dSTakashi Iwai {
19264a79616dSTakashi Iwai 	struct via_spec *spec = codec->spec;
1927f4a7828bSTakashi Iwai 	struct auto_pin_cfg *cfg = &spec->autocfg;
19283214b966STakashi Iwai 	struct nid_path *path;
19294a79616dSTakashi Iwai 	static const char * const chname[4] = {
193034ca8d33SDavid Henningsson 		"Front", "Surround", NULL /* "CLFE" */, "Side"
19314a79616dSTakashi Iwai 	};
19324a79616dSTakashi Iwai 	int i, idx, err;
1933f4a7828bSTakashi Iwai 	int old_line_outs;
1934f4a7828bSTakashi Iwai 
1935f4a7828bSTakashi Iwai 	/* check smart51 */
1936f4a7828bSTakashi Iwai 	old_line_outs = cfg->line_outs;
1937f4a7828bSTakashi Iwai 	if (cfg->line_outs == 1)
1938f4a7828bSTakashi Iwai 		mangle_smart51(codec);
19394a79616dSTakashi Iwai 
1940e3d7a143STakashi Iwai 	err = via_auto_fill_dac_nids(codec);
1941e3d7a143STakashi Iwai 	if (err < 0)
1942e3d7a143STakashi Iwai 		return err;
1943e3d7a143STakashi Iwai 
19445c9a5615SLydia Wang 	if (spec->multiout.num_dacs < 3) {
19455c9a5615SLydia Wang 		spec->smart51_nums = 0;
19465c9a5615SLydia Wang 		cfg->line_outs = old_line_outs;
19475c9a5615SLydia Wang 	}
19484a79616dSTakashi Iwai 	for (i = 0; i < cfg->line_outs; i++) {
19494a79616dSTakashi Iwai 		hda_nid_t pin, dac;
19504a79616dSTakashi Iwai 		pin = cfg->line_out_pins[i];
19514a79616dSTakashi Iwai 		dac = spec->multiout.dac_nids[i];
19524a79616dSTakashi Iwai 		if (!pin || !dac)
19534a79616dSTakashi Iwai 			continue;
19543214b966STakashi Iwai 		path = spec->out_path + i;
19550fe0adf8STakashi Iwai 		if (i == HDA_CLFE) {
19563214b966STakashi Iwai 			err = create_ch_ctls(codec, "Center", 1, true, path);
19574a79616dSTakashi Iwai 			if (err < 0)
19584a79616dSTakashi Iwai 				return err;
19593214b966STakashi Iwai 			err = create_ch_ctls(codec, "LFE", 2, true, path);
19604a79616dSTakashi Iwai 			if (err < 0)
19614a79616dSTakashi Iwai 				return err;
19624a79616dSTakashi Iwai 		} else {
19636aadf41dSTakashi Iwai 			const char *pfx = chname[i];
19646aadf41dSTakashi Iwai 			if (cfg->line_out_type == AUTO_PIN_SPEAKER_OUT &&
1965f37bc7a8STakashi Iwai 			    cfg->line_outs <= 2)
1966f37bc7a8STakashi Iwai 				pfx = i ? "Bass Speaker" : "Speaker";
19673214b966STakashi Iwai 			err = create_ch_ctls(codec, pfx, 3, true, path);
19684a79616dSTakashi Iwai 			if (err < 0)
19694a79616dSTakashi Iwai 				return err;
19704a79616dSTakashi Iwai 		}
1971020066d1STakashi Iwai 		if (path != spec->out_path + i)
1972020066d1STakashi Iwai 			copy_path_mixer_ctls(&spec->out_path[i], path);
1973020066d1STakashi Iwai 		if (path == spec->out_path && spec->out_mix_path.depth)
1974020066d1STakashi Iwai 			copy_path_mixer_ctls(&spec->out_mix_path, path);
19754a79616dSTakashi Iwai 	}
19764a79616dSTakashi Iwai 
19774a79616dSTakashi Iwai 	idx = get_connection_index(codec, spec->aa_mix_nid,
19784a79616dSTakashi Iwai 				   spec->multiout.dac_nids[0]);
19794a79616dSTakashi Iwai 	if (idx >= 0) {
19804a79616dSTakashi Iwai 		/* add control to mixer */
19813214b966STakashi Iwai 		const char *name;
19823214b966STakashi Iwai 		name = spec->out_mix_path.depth ?
19833214b966STakashi Iwai 			"PCM Loopback Playback Volume" : "PCM Playback Volume";
19843214b966STakashi Iwai 		err = via_add_control(spec, VIA_CTL_WIDGET_VOL, name,
19854a79616dSTakashi Iwai 				      HDA_COMPOSE_AMP_VAL(spec->aa_mix_nid, 3,
19864a79616dSTakashi Iwai 							  idx, HDA_INPUT));
19874a79616dSTakashi Iwai 		if (err < 0)
19884a79616dSTakashi Iwai 			return err;
19893214b966STakashi Iwai 		name = spec->out_mix_path.depth ?
19903214b966STakashi Iwai 			"PCM Loopback Playback Switch" : "PCM Playback Switch";
19913214b966STakashi Iwai 		err = via_add_control(spec, VIA_CTL_WIDGET_MUTE, name,
19924a79616dSTakashi Iwai 				      HDA_COMPOSE_AMP_VAL(spec->aa_mix_nid, 3,
19934a79616dSTakashi Iwai 							  idx, HDA_INPUT));
19944a79616dSTakashi Iwai 		if (err < 0)
19954a79616dSTakashi Iwai 			return err;
1996c577b8a1SJoseph Chan 	}
1997c577b8a1SJoseph Chan 
1998f4a7828bSTakashi Iwai 	cfg->line_outs = old_line_outs;
1999f4a7828bSTakashi Iwai 
2000c577b8a1SJoseph Chan 	return 0;
2001c577b8a1SJoseph Chan }
2002c577b8a1SJoseph Chan 
20034a79616dSTakashi Iwai static int via_auto_create_hp_ctls(struct hda_codec *codec, hda_nid_t pin)
2004c577b8a1SJoseph Chan {
20054a79616dSTakashi Iwai 	struct via_spec *spec = codec->spec;
200609a9ad69STakashi Iwai 	struct nid_path *path;
200718bd2c44STakashi Iwai 	bool check_dac;
20083214b966STakashi Iwai 	int i, err;
2009c577b8a1SJoseph Chan 
2010c577b8a1SJoseph Chan 	if (!pin)
2011c577b8a1SJoseph Chan 		return 0;
2012c577b8a1SJoseph Chan 
20133214b966STakashi Iwai 	if (!parse_output_path(codec, pin, 0, 0, &spec->hp_indep_path)) {
20143214b966STakashi Iwai 		for (i = HDA_SIDE; i >= HDA_CLFE; i--) {
20153214b966STakashi Iwai 			if (i < spec->multiout.num_dacs &&
201625250505STakashi Iwai 			    parse_output_path(codec, pin,
20173214b966STakashi Iwai 					      spec->multiout.dac_nids[i], 0,
20183214b966STakashi Iwai 					      &spec->hp_indep_path)) {
20193214b966STakashi Iwai 				spec->hp_indep_shared = i;
20203214b966STakashi Iwai 				break;
202125250505STakashi Iwai 			}
20223214b966STakashi Iwai 		}
20233214b966STakashi Iwai 	}
20243214b966STakashi Iwai 	if (spec->hp_indep_path.depth) {
20253214b966STakashi Iwai 		spec->hp_dac_nid = spec->hp_indep_path.path[0];
20263214b966STakashi Iwai 		if (!spec->hp_indep_shared)
20273214b966STakashi Iwai 			spec->hp_path = spec->hp_indep_path;
20283214b966STakashi Iwai 	}
20293214b966STakashi Iwai 	/* optionally check front-path w/o AA-mix */
20303214b966STakashi Iwai 	if (!spec->hp_path.depth)
20313214b966STakashi Iwai 		parse_output_path(codec, pin,
20323214b966STakashi Iwai 				  spec->multiout.dac_nids[HDA_FRONT], 0,
20333214b966STakashi Iwai 				  &spec->hp_path);
20344a79616dSTakashi Iwai 
2035ece8d043STakashi Iwai 	if (!parse_output_path(codec, pin, spec->multiout.dac_nids[HDA_FRONT],
20363214b966STakashi Iwai 			       1, &spec->hp_mix_path) && !spec->hp_path.depth)
2037ece8d043STakashi Iwai 		return 0;
2038ece8d043STakashi Iwai 
20393214b966STakashi Iwai 	if (spec->hp_path.depth) {
204009a9ad69STakashi Iwai 		path = &spec->hp_path;
204118bd2c44STakashi Iwai 		check_dac = true;
204218bd2c44STakashi Iwai 	} else {
20433214b966STakashi Iwai 		path = &spec->hp_mix_path;
204418bd2c44STakashi Iwai 		check_dac = false;
204518bd2c44STakashi Iwai 	}
204618bd2c44STakashi Iwai 	err = create_ch_ctls(codec, "Headphone", 3, check_dac, path);
20474a79616dSTakashi Iwai 	if (err < 0)
20484a79616dSTakashi Iwai 		return err;
2049020066d1STakashi Iwai 	if (check_dac)
2050020066d1STakashi Iwai 		copy_path_mixer_ctls(&spec->hp_mix_path, path);
2051020066d1STakashi Iwai 	else
2052020066d1STakashi Iwai 		copy_path_mixer_ctls(&spec->hp_path, path);
2053020066d1STakashi Iwai 	if (spec->hp_indep_path.depth)
2054020066d1STakashi Iwai 		copy_path_mixer_ctls(&spec->hp_indep_path, path);
2055c577b8a1SJoseph Chan 	return 0;
2056c577b8a1SJoseph Chan }
2057c577b8a1SJoseph Chan 
20584a918ffeSTakashi Iwai static int via_auto_create_speaker_ctls(struct hda_codec *codec)
20594a918ffeSTakashi Iwai {
20604a918ffeSTakashi Iwai 	struct via_spec *spec = codec->spec;
20613214b966STakashi Iwai 	struct nid_path *path;
20623214b966STakashi Iwai 	bool check_dac;
206381c0a78bSWang Shaoyan 	hda_nid_t pin, dac = 0;
20643214b966STakashi Iwai 	int err;
20654a918ffeSTakashi Iwai 
20664a918ffeSTakashi Iwai 	pin = spec->autocfg.speaker_pins[0];
20674a918ffeSTakashi Iwai 	if (!spec->autocfg.speaker_outs || !pin)
20684a918ffeSTakashi Iwai 		return 0;
20694a918ffeSTakashi Iwai 
20703214b966STakashi Iwai 	if (parse_output_path(codec, pin, 0, 0, &spec->speaker_path))
20718e3679dcSTakashi Iwai 		dac = spec->speaker_path.path[0];
20723214b966STakashi Iwai 	if (!dac)
20733214b966STakashi Iwai 		parse_output_path(codec, pin,
20743214b966STakashi Iwai 				  spec->multiout.dac_nids[HDA_FRONT], 0,
207509a9ad69STakashi Iwai 				  &spec->speaker_path);
20763214b966STakashi Iwai 	if (!parse_output_path(codec, pin, spec->multiout.dac_nids[HDA_FRONT],
20773214b966STakashi Iwai 			       1, &spec->speaker_mix_path) && !dac)
20783214b966STakashi Iwai 		return 0;
20794a918ffeSTakashi Iwai 
20803214b966STakashi Iwai 	/* no AA-path for front? */
20813214b966STakashi Iwai 	if (!spec->out_mix_path.depth && spec->speaker_mix_path.depth)
20823214b966STakashi Iwai 		dac = 0;
20833214b966STakashi Iwai 
20843214b966STakashi Iwai 	spec->speaker_dac_nid = dac;
20853214b966STakashi Iwai 	spec->multiout.extra_out_nid[0] = dac;
20863214b966STakashi Iwai 	if (dac) {
20873214b966STakashi Iwai 		path = &spec->speaker_path;
20883214b966STakashi Iwai 		check_dac = true;
20893214b966STakashi Iwai 	} else {
20903214b966STakashi Iwai 		path = &spec->speaker_mix_path;
20913214b966STakashi Iwai 		check_dac = false;
20923214b966STakashi Iwai 	}
20933214b966STakashi Iwai 	err = create_ch_ctls(codec, "Speaker", 3, check_dac, path);
20943214b966STakashi Iwai 	if (err < 0)
20953214b966STakashi Iwai 		return err;
2096020066d1STakashi Iwai 	if (check_dac)
2097020066d1STakashi Iwai 		copy_path_mixer_ctls(&spec->speaker_mix_path, path);
2098020066d1STakashi Iwai 	else
2099020066d1STakashi Iwai 		copy_path_mixer_ctls(&spec->speaker_path, path);
21003214b966STakashi Iwai 	return 0;
21013214b966STakashi Iwai }
21023214b966STakashi Iwai 
21033214b966STakashi Iwai #define via_aamix_ctl_info	via_pin_power_ctl_info
21043214b966STakashi Iwai 
21053214b966STakashi Iwai static int via_aamix_ctl_get(struct snd_kcontrol *kcontrol,
21063214b966STakashi Iwai 			     struct snd_ctl_elem_value *ucontrol)
21073214b966STakashi Iwai {
21083214b966STakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
21093214b966STakashi Iwai 	struct via_spec *spec = codec->spec;
21103214b966STakashi Iwai 	ucontrol->value.enumerated.item[0] = spec->aamix_mode;
21113214b966STakashi Iwai 	return 0;
21123214b966STakashi Iwai }
21133214b966STakashi Iwai 
21143214b966STakashi Iwai static void update_aamix_paths(struct hda_codec *codec, int do_mix,
21153214b966STakashi Iwai 			       struct nid_path *nomix, struct nid_path *mix)
21163214b966STakashi Iwai {
21173214b966STakashi Iwai 	if (do_mix) {
21183214b966STakashi Iwai 		activate_output_path(codec, nomix, false, false);
21193214b966STakashi Iwai 		activate_output_path(codec, mix, true, false);
21203214b966STakashi Iwai 	} else {
21213214b966STakashi Iwai 		activate_output_path(codec, mix, false, false);
21223214b966STakashi Iwai 		activate_output_path(codec, nomix, true, false);
21233214b966STakashi Iwai 	}
21243214b966STakashi Iwai }
21253214b966STakashi Iwai 
21263214b966STakashi Iwai static int via_aamix_ctl_put(struct snd_kcontrol *kcontrol,
21273214b966STakashi Iwai 			     struct snd_ctl_elem_value *ucontrol)
21283214b966STakashi Iwai {
21293214b966STakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
21303214b966STakashi Iwai 	struct via_spec *spec = codec->spec;
21313214b966STakashi Iwai 	unsigned int val = ucontrol->value.enumerated.item[0];
21323214b966STakashi Iwai 
21333214b966STakashi Iwai 	if (val == spec->aamix_mode)
21343214b966STakashi Iwai 		return 0;
21353214b966STakashi Iwai 	spec->aamix_mode = val;
21363214b966STakashi Iwai 	/* update front path */
21373214b966STakashi Iwai 	update_aamix_paths(codec, val, &spec->out_path[0], &spec->out_mix_path);
21383214b966STakashi Iwai 	/* update HP path */
21393214b966STakashi Iwai 	if (!spec->hp_independent_mode) {
21403214b966STakashi Iwai 		update_aamix_paths(codec, val, &spec->hp_path,
21413214b966STakashi Iwai 				   &spec->hp_mix_path);
21423214b966STakashi Iwai 	}
21433214b966STakashi Iwai 	/* update speaker path */
21443214b966STakashi Iwai 	update_aamix_paths(codec, val, &spec->speaker_path,
21453214b966STakashi Iwai 			   &spec->speaker_mix_path);
21463214b966STakashi Iwai 	return 1;
21473214b966STakashi Iwai }
21483214b966STakashi Iwai 
21493214b966STakashi Iwai static const struct snd_kcontrol_new via_aamix_ctl_enum = {
21503214b966STakashi Iwai 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
21513214b966STakashi Iwai 	.name = "Loopback Mixing",
21523214b966STakashi Iwai 	.info = via_aamix_ctl_info,
21533214b966STakashi Iwai 	.get = via_aamix_ctl_get,
21543214b966STakashi Iwai 	.put = via_aamix_ctl_put,
21553214b966STakashi Iwai };
21563214b966STakashi Iwai 
21573214b966STakashi Iwai static int via_auto_create_loopback_switch(struct hda_codec *codec)
21583214b966STakashi Iwai {
21593214b966STakashi Iwai 	struct via_spec *spec = codec->spec;
21603214b966STakashi Iwai 
21614808d12dSTakashi Iwai 	if (!spec->aa_mix_nid)
21624808d12dSTakashi Iwai 		return 0; /* no loopback switching available */
21634808d12dSTakashi Iwai 	if (!(spec->out_mix_path.depth || spec->hp_mix_path.depth ||
21644808d12dSTakashi Iwai 	      spec->speaker_path.depth))
21653214b966STakashi Iwai 		return 0; /* no loopback switching available */
21663214b966STakashi Iwai 	if (!via_clone_control(spec, &via_aamix_ctl_enum))
21673214b966STakashi Iwai 		return -ENOMEM;
21684a918ffeSTakashi Iwai 	return 0;
21694a918ffeSTakashi Iwai }
21704a918ffeSTakashi Iwai 
2171a766d0d7STakashi Iwai /* look for ADCs */
2172a766d0d7STakashi Iwai static int via_fill_adcs(struct hda_codec *codec)
2173a766d0d7STakashi Iwai {
2174a766d0d7STakashi Iwai 	struct via_spec *spec = codec->spec;
2175a766d0d7STakashi Iwai 	hda_nid_t nid = codec->start_nid;
2176a766d0d7STakashi Iwai 	int i;
2177a766d0d7STakashi Iwai 
2178a766d0d7STakashi Iwai 	for (i = 0; i < codec->num_nodes; i++, nid++) {
2179a766d0d7STakashi Iwai 		unsigned int wcaps = get_wcaps(codec, nid);
2180a766d0d7STakashi Iwai 		if (get_wcaps_type(wcaps) != AC_WID_AUD_IN)
2181a766d0d7STakashi Iwai 			continue;
2182a766d0d7STakashi Iwai 		if (wcaps & AC_WCAP_DIGITAL)
2183a766d0d7STakashi Iwai 			continue;
2184a766d0d7STakashi Iwai 		if (!(wcaps & AC_WCAP_CONN_LIST))
2185a766d0d7STakashi Iwai 			continue;
2186a766d0d7STakashi Iwai 		if (spec->num_adc_nids >= ARRAY_SIZE(spec->adc_nids))
2187a766d0d7STakashi Iwai 			return -ENOMEM;
2188a766d0d7STakashi Iwai 		spec->adc_nids[spec->num_adc_nids++] = nid;
2189a766d0d7STakashi Iwai 	}
2190a766d0d7STakashi Iwai 	return 0;
2191a766d0d7STakashi Iwai }
2192a766d0d7STakashi Iwai 
2193a86a88eaSTakashi Iwai /* input-src control */
2194a86a88eaSTakashi Iwai static int via_mux_enum_info(struct snd_kcontrol *kcontrol,
2195a86a88eaSTakashi Iwai 			     struct snd_ctl_elem_info *uinfo)
2196a86a88eaSTakashi Iwai {
2197a86a88eaSTakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2198a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2199a86a88eaSTakashi Iwai 
2200a86a88eaSTakashi Iwai 	uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
2201a86a88eaSTakashi Iwai 	uinfo->count = 1;
2202a86a88eaSTakashi Iwai 	uinfo->value.enumerated.items = spec->num_inputs;
2203a86a88eaSTakashi Iwai 	if (uinfo->value.enumerated.item >= spec->num_inputs)
2204a86a88eaSTakashi Iwai 		uinfo->value.enumerated.item = spec->num_inputs - 1;
2205a86a88eaSTakashi Iwai 	strcpy(uinfo->value.enumerated.name,
2206a86a88eaSTakashi Iwai 	       spec->inputs[uinfo->value.enumerated.item].label);
2207a86a88eaSTakashi Iwai 	return 0;
2208a86a88eaSTakashi Iwai }
2209a86a88eaSTakashi Iwai 
2210a86a88eaSTakashi Iwai static int via_mux_enum_get(struct snd_kcontrol *kcontrol,
2211a86a88eaSTakashi Iwai 			    struct snd_ctl_elem_value *ucontrol)
2212a86a88eaSTakashi Iwai {
2213a86a88eaSTakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2214a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2215a86a88eaSTakashi Iwai 	unsigned int idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
2216a86a88eaSTakashi Iwai 
2217a86a88eaSTakashi Iwai 	ucontrol->value.enumerated.item[0] = spec->cur_mux[idx];
2218a86a88eaSTakashi Iwai 	return 0;
2219a86a88eaSTakashi Iwai }
2220a86a88eaSTakashi Iwai 
2221a86a88eaSTakashi Iwai static int via_mux_enum_put(struct snd_kcontrol *kcontrol,
2222a86a88eaSTakashi Iwai 			    struct snd_ctl_elem_value *ucontrol)
2223a86a88eaSTakashi Iwai {
2224a86a88eaSTakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2225a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2226a86a88eaSTakashi Iwai 	unsigned int idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
2227a86a88eaSTakashi Iwai 	hda_nid_t mux;
2228a86a88eaSTakashi Iwai 	int cur;
2229a86a88eaSTakashi Iwai 
2230a86a88eaSTakashi Iwai 	cur = ucontrol->value.enumerated.item[0];
2231a86a88eaSTakashi Iwai 	if (cur < 0 || cur >= spec->num_inputs)
2232a86a88eaSTakashi Iwai 		return -EINVAL;
2233a86a88eaSTakashi Iwai 	if (spec->cur_mux[idx] == cur)
2234a86a88eaSTakashi Iwai 		return 0;
2235a86a88eaSTakashi Iwai 	spec->cur_mux[idx] = cur;
2236a86a88eaSTakashi Iwai 	if (spec->dyn_adc_switch) {
2237a86a88eaSTakashi Iwai 		int adc_idx = spec->inputs[cur].adc_idx;
2238a86a88eaSTakashi Iwai 		mux = spec->mux_nids[adc_idx];
2239a86a88eaSTakashi Iwai 		via_dyn_adc_pcm_resetup(codec, cur);
2240a86a88eaSTakashi Iwai 	} else {
2241a86a88eaSTakashi Iwai 		mux = spec->mux_nids[idx];
2242a86a88eaSTakashi Iwai 		if (snd_BUG_ON(!mux))
2243a86a88eaSTakashi Iwai 			return -EINVAL;
2244a86a88eaSTakashi Iwai 	}
2245a86a88eaSTakashi Iwai 
2246a86a88eaSTakashi Iwai 	if (mux) {
2247a86a88eaSTakashi Iwai 		/* switch to D0 beofre change index */
2248054d867eSTakashi Iwai 		update_power_state(codec, mux, AC_PWRST_D0);
2249a86a88eaSTakashi Iwai 		snd_hda_codec_write(codec, mux, 0,
2250a86a88eaSTakashi Iwai 				    AC_VERB_SET_CONNECT_SEL,
2251a86a88eaSTakashi Iwai 				    spec->inputs[cur].mux_idx);
2252a86a88eaSTakashi Iwai 	}
2253a86a88eaSTakashi Iwai 
2254a86a88eaSTakashi Iwai 	/* update jack power state */
2255a86a88eaSTakashi Iwai 	set_widgets_power_state(codec);
2256a86a88eaSTakashi Iwai 	return 0;
2257a86a88eaSTakashi Iwai }
2258a766d0d7STakashi Iwai 
2259d7a99cceSTakashi Iwai static const struct snd_kcontrol_new via_input_src_ctl = {
2260d7a99cceSTakashi Iwai 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2261d7a99cceSTakashi Iwai 	/* The multiple "Capture Source" controls confuse alsamixer
2262d7a99cceSTakashi Iwai 	 * So call somewhat different..
2263d7a99cceSTakashi Iwai 	 */
2264d7a99cceSTakashi Iwai 	/* .name = "Capture Source", */
2265d7a99cceSTakashi Iwai 	.name = "Input Source",
2266d7a99cceSTakashi Iwai 	.info = via_mux_enum_info,
2267d7a99cceSTakashi Iwai 	.get = via_mux_enum_get,
2268d7a99cceSTakashi Iwai 	.put = via_mux_enum_put,
2269d7a99cceSTakashi Iwai };
2270d7a99cceSTakashi Iwai 
2271a86a88eaSTakashi Iwai static int create_input_src_ctls(struct hda_codec *codec, int count)
2272a86a88eaSTakashi Iwai {
2273a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2274a86a88eaSTakashi Iwai 	struct snd_kcontrol_new *knew;
2275a86a88eaSTakashi Iwai 
2276a86a88eaSTakashi Iwai 	if (spec->num_inputs <= 1 || !count)
2277a86a88eaSTakashi Iwai 		return 0; /* no need for single src */
2278a86a88eaSTakashi Iwai 
2279a86a88eaSTakashi Iwai 	knew = via_clone_control(spec, &via_input_src_ctl);
2280a86a88eaSTakashi Iwai 	if (!knew)
2281a86a88eaSTakashi Iwai 		return -ENOMEM;
2282a86a88eaSTakashi Iwai 	knew->count = count;
2283a86a88eaSTakashi Iwai 	return 0;
2284a86a88eaSTakashi Iwai }
2285a86a88eaSTakashi Iwai 
2286a86a88eaSTakashi Iwai /* add the powersave loopback-list entry */
228713af8e77STakashi Iwai static void add_loopback_list(struct via_spec *spec, hda_nid_t mix, int idx)
228813af8e77STakashi Iwai {
228913af8e77STakashi Iwai 	struct hda_amp_list *list;
229013af8e77STakashi Iwai 
229113af8e77STakashi Iwai 	if (spec->num_loopbacks >= ARRAY_SIZE(spec->loopback_list) - 1)
229213af8e77STakashi Iwai 		return;
229313af8e77STakashi Iwai 	list = spec->loopback_list + spec->num_loopbacks;
229413af8e77STakashi Iwai 	list->nid = mix;
229513af8e77STakashi Iwai 	list->dir = HDA_INPUT;
229613af8e77STakashi Iwai 	list->idx = idx;
229713af8e77STakashi Iwai 	spec->num_loopbacks++;
229813af8e77STakashi Iwai 	spec->loopback.amplist = spec->loopback_list;
229913af8e77STakashi Iwai }
230013af8e77STakashi Iwai 
2301a86a88eaSTakashi Iwai static bool is_reachable_nid(struct hda_codec *codec, hda_nid_t src,
23028d087c76STakashi Iwai 			     hda_nid_t dst)
2303a86a88eaSTakashi Iwai {
23048d087c76STakashi Iwai 	return snd_hda_get_conn_index(codec, src, dst, 1) >= 0;
2305a86a88eaSTakashi Iwai }
2306a86a88eaSTakashi Iwai 
2307a86a88eaSTakashi Iwai /* add the input-route to the given pin */
2308a86a88eaSTakashi Iwai static bool add_input_route(struct hda_codec *codec, hda_nid_t pin)
2309c577b8a1SJoseph Chan {
231010a20af7STakashi Iwai 	struct via_spec *spec = codec->spec;
2311a86a88eaSTakashi Iwai 	int c, idx;
2312a86a88eaSTakashi Iwai 
2313a86a88eaSTakashi Iwai 	spec->inputs[spec->num_inputs].adc_idx = -1;
2314a86a88eaSTakashi Iwai 	spec->inputs[spec->num_inputs].pin = pin;
2315a86a88eaSTakashi Iwai 	for (c = 0; c < spec->num_adc_nids; c++) {
2316a86a88eaSTakashi Iwai 		if (spec->mux_nids[c]) {
2317a86a88eaSTakashi Iwai 			idx = get_connection_index(codec, spec->mux_nids[c],
2318a86a88eaSTakashi Iwai 						   pin);
2319a86a88eaSTakashi Iwai 			if (idx < 0)
2320a86a88eaSTakashi Iwai 				continue;
2321a86a88eaSTakashi Iwai 			spec->inputs[spec->num_inputs].mux_idx = idx;
2322a86a88eaSTakashi Iwai 		} else {
23238d087c76STakashi Iwai 			if (!is_reachable_nid(codec, spec->adc_nids[c], pin))
2324a86a88eaSTakashi Iwai 				continue;
2325a86a88eaSTakashi Iwai 		}
2326a86a88eaSTakashi Iwai 		spec->inputs[spec->num_inputs].adc_idx = c;
2327a86a88eaSTakashi Iwai 		/* Can primary ADC satisfy all inputs? */
2328a86a88eaSTakashi Iwai 		if (!spec->dyn_adc_switch &&
2329a86a88eaSTakashi Iwai 		    spec->num_inputs > 0 && spec->inputs[0].adc_idx != c) {
2330a86a88eaSTakashi Iwai 			snd_printd(KERN_INFO
2331a86a88eaSTakashi Iwai 				   "via: dynamic ADC switching enabled\n");
2332a86a88eaSTakashi Iwai 			spec->dyn_adc_switch = 1;
2333a86a88eaSTakashi Iwai 		}
2334a86a88eaSTakashi Iwai 		return true;
2335a86a88eaSTakashi Iwai 	}
2336a86a88eaSTakashi Iwai 	return false;
2337a86a88eaSTakashi Iwai }
2338a86a88eaSTakashi Iwai 
2339a86a88eaSTakashi Iwai static int get_mux_nids(struct hda_codec *codec);
2340a86a88eaSTakashi Iwai 
2341a86a88eaSTakashi Iwai /* parse input-routes; fill ADCs, MUXs and input-src entries */
2342a86a88eaSTakashi Iwai static int parse_analog_inputs(struct hda_codec *codec)
2343a86a88eaSTakashi Iwai {
2344a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2345a86a88eaSTakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
2346a86a88eaSTakashi Iwai 	int i, err;
2347a766d0d7STakashi Iwai 
2348a766d0d7STakashi Iwai 	err = via_fill_adcs(codec);
2349a766d0d7STakashi Iwai 	if (err < 0)
2350a766d0d7STakashi Iwai 		return err;
2351a766d0d7STakashi Iwai 	err = get_mux_nids(codec);
2352a766d0d7STakashi Iwai 	if (err < 0)
2353a766d0d7STakashi Iwai 		return err;
2354a766d0d7STakashi Iwai 
2355a86a88eaSTakashi Iwai 	/* fill all input-routes */
2356a86a88eaSTakashi Iwai 	for (i = 0; i < cfg->num_inputs; i++) {
2357a86a88eaSTakashi Iwai 		if (add_input_route(codec, cfg->inputs[i].pin))
2358a86a88eaSTakashi Iwai 			spec->inputs[spec->num_inputs++].label =
2359a86a88eaSTakashi Iwai 				hda_get_autocfg_input_label(codec, cfg, i);
2360a86a88eaSTakashi Iwai 	}
2361a86a88eaSTakashi Iwai 
2362a86a88eaSTakashi Iwai 	/* check for internal loopback recording */
2363a86a88eaSTakashi Iwai 	if (spec->aa_mix_nid &&
2364a86a88eaSTakashi Iwai 	    add_input_route(codec, spec->aa_mix_nid))
2365a86a88eaSTakashi Iwai 		spec->inputs[spec->num_inputs++].label = "Stereo Mixer";
2366a86a88eaSTakashi Iwai 
2367a86a88eaSTakashi Iwai 	return 0;
2368a86a88eaSTakashi Iwai }
2369a86a88eaSTakashi Iwai 
2370a86a88eaSTakashi Iwai /* create analog-loopback volume/switch controls */
2371a86a88eaSTakashi Iwai static int create_loopback_ctls(struct hda_codec *codec)
2372a86a88eaSTakashi Iwai {
2373a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2374a86a88eaSTakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
2375a86a88eaSTakashi Iwai 	const char *prev_label = NULL;
2376a86a88eaSTakashi Iwai 	int type_idx = 0;
2377a86a88eaSTakashi Iwai 	int i, j, err, idx;
2378a86a88eaSTakashi Iwai 
2379a86a88eaSTakashi Iwai 	if (!spec->aa_mix_nid)
2380a766d0d7STakashi Iwai 		return 0;
2381c577b8a1SJoseph Chan 
23827b315bb4STakashi Iwai 	for (i = 0; i < cfg->num_inputs; i++) {
2383a86a88eaSTakashi Iwai 		hda_nid_t pin = cfg->inputs[i].pin;
2384a86a88eaSTakashi Iwai 		const char *label = hda_get_autocfg_input_label(codec, cfg, i);
2385a86a88eaSTakashi Iwai 
23861e11cae1STakashi Iwai 		if (prev_label && !strcmp(label, prev_label))
23877b315bb4STakashi Iwai 			type_idx++;
23887b315bb4STakashi Iwai 		else
23897b315bb4STakashi Iwai 			type_idx = 0;
23901e11cae1STakashi Iwai 		prev_label = label;
2391a86a88eaSTakashi Iwai 		idx = get_connection_index(codec, spec->aa_mix_nid, pin);
2392a86a88eaSTakashi Iwai 		if (idx >= 0) {
239316922281SLydia Wang 			err = via_new_analog_input(spec, label, type_idx,
2394a86a88eaSTakashi Iwai 						   idx, spec->aa_mix_nid);
2395c577b8a1SJoseph Chan 			if (err < 0)
2396c577b8a1SJoseph Chan 				return err;
2397a86a88eaSTakashi Iwai 			add_loopback_list(spec, spec->aa_mix_nid, idx);
239813af8e77STakashi Iwai 		}
2399e3d7a143STakashi Iwai 
2400e3d7a143STakashi Iwai 		/* remember the label for smart51 control */
2401e3d7a143STakashi Iwai 		for (j = 0; j < spec->smart51_nums; j++) {
2402a86a88eaSTakashi Iwai 			if (spec->smart51_pins[j] == pin) {
2403e3d7a143STakashi Iwai 				spec->smart51_idxs[j] = idx;
2404e3d7a143STakashi Iwai 				spec->smart51_labels[j] = label;
2405e3d7a143STakashi Iwai 				break;
2406e3d7a143STakashi Iwai 			}
2407e3d7a143STakashi Iwai 		}
2408c577b8a1SJoseph Chan 	}
2409a86a88eaSTakashi Iwai 	return 0;
2410a86a88eaSTakashi Iwai }
2411a86a88eaSTakashi Iwai 
2412a86a88eaSTakashi Iwai /* create mic-boost controls (if present) */
2413a86a88eaSTakashi Iwai static int create_mic_boost_ctls(struct hda_codec *codec)
2414a86a88eaSTakashi Iwai {
2415a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2416a86a88eaSTakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
24178d8bbc6fSTakashi Iwai 	const char *prev_label = NULL;
24188d8bbc6fSTakashi Iwai 	int type_idx = 0;
2419a86a88eaSTakashi Iwai 	int i, err;
2420a86a88eaSTakashi Iwai 
2421a86a88eaSTakashi Iwai 	for (i = 0; i < cfg->num_inputs; i++) {
2422a86a88eaSTakashi Iwai 		hda_nid_t pin = cfg->inputs[i].pin;
2423a86a88eaSTakashi Iwai 		unsigned int caps;
2424a86a88eaSTakashi Iwai 		const char *label;
2425a86a88eaSTakashi Iwai 		char name[32];
2426a86a88eaSTakashi Iwai 
2427a86a88eaSTakashi Iwai 		if (cfg->inputs[i].type != AUTO_PIN_MIC)
2428a86a88eaSTakashi Iwai 			continue;
2429a86a88eaSTakashi Iwai 		caps = query_amp_caps(codec, pin, HDA_INPUT);
2430a86a88eaSTakashi Iwai 		if (caps == -1 || !(caps & AC_AMPCAP_NUM_STEPS))
2431a86a88eaSTakashi Iwai 			continue;
2432a86a88eaSTakashi Iwai 		label = hda_get_autocfg_input_label(codec, cfg, i);
24338d8bbc6fSTakashi Iwai 		if (prev_label && !strcmp(label, prev_label))
24348d8bbc6fSTakashi Iwai 			type_idx++;
24358d8bbc6fSTakashi Iwai 		else
24368d8bbc6fSTakashi Iwai 			type_idx = 0;
24378d8bbc6fSTakashi Iwai 		prev_label = label;
2438a86a88eaSTakashi Iwai 		snprintf(name, sizeof(name), "%s Boost Volume", label);
24398d8bbc6fSTakashi Iwai 		err = __via_add_control(spec, VIA_CTL_WIDGET_VOL, name, type_idx,
2440a86a88eaSTakashi Iwai 			      HDA_COMPOSE_AMP_VAL(pin, 3, 0, HDA_INPUT));
2441a86a88eaSTakashi Iwai 		if (err < 0)
2442a86a88eaSTakashi Iwai 			return err;
2443a86a88eaSTakashi Iwai 	}
2444a86a88eaSTakashi Iwai 	return 0;
2445a86a88eaSTakashi Iwai }
2446a86a88eaSTakashi Iwai 
2447a86a88eaSTakashi Iwai /* create capture and input-src controls for multiple streams */
2448a86a88eaSTakashi Iwai static int create_multi_adc_ctls(struct hda_codec *codec)
2449a86a88eaSTakashi Iwai {
2450a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2451a86a88eaSTakashi Iwai 	int i, err;
2452d7a99cceSTakashi Iwai 
2453d7a99cceSTakashi Iwai 	/* create capture mixer elements */
2454d7a99cceSTakashi Iwai 	for (i = 0; i < spec->num_adc_nids; i++) {
2455d7a99cceSTakashi Iwai 		hda_nid_t adc = spec->adc_nids[i];
2456d7a99cceSTakashi Iwai 		err = __via_add_control(spec, VIA_CTL_WIDGET_VOL,
2457d7a99cceSTakashi Iwai 					"Capture Volume", i,
2458d7a99cceSTakashi Iwai 					HDA_COMPOSE_AMP_VAL(adc, 3, 0,
2459d7a99cceSTakashi Iwai 							    HDA_INPUT));
2460d7a99cceSTakashi Iwai 		if (err < 0)
2461d7a99cceSTakashi Iwai 			return err;
2462d7a99cceSTakashi Iwai 		err = __via_add_control(spec, VIA_CTL_WIDGET_MUTE,
2463d7a99cceSTakashi Iwai 					"Capture Switch", i,
2464d7a99cceSTakashi Iwai 					HDA_COMPOSE_AMP_VAL(adc, 3, 0,
2465d7a99cceSTakashi Iwai 							    HDA_INPUT));
2466d7a99cceSTakashi Iwai 		if (err < 0)
2467d7a99cceSTakashi Iwai 			return err;
2468d7a99cceSTakashi Iwai 	}
2469d7a99cceSTakashi Iwai 
2470d7a99cceSTakashi Iwai 	/* input-source control */
2471d7a99cceSTakashi Iwai 	for (i = 0; i < spec->num_adc_nids; i++)
2472d7a99cceSTakashi Iwai 		if (!spec->mux_nids[i])
2473d7a99cceSTakashi Iwai 			break;
2474a86a88eaSTakashi Iwai 	err = create_input_src_ctls(codec, i);
2475d7a99cceSTakashi Iwai 	if (err < 0)
2476d7a99cceSTakashi Iwai 		return err;
2477a86a88eaSTakashi Iwai 	return 0;
2478d7a99cceSTakashi Iwai }
2479d7a99cceSTakashi Iwai 
2480a86a88eaSTakashi Iwai /* bind capture volume/switch */
2481a86a88eaSTakashi Iwai static struct snd_kcontrol_new via_bind_cap_vol_ctl =
2482a86a88eaSTakashi Iwai 	HDA_BIND_VOL("Capture Volume", 0);
2483a86a88eaSTakashi Iwai static struct snd_kcontrol_new via_bind_cap_sw_ctl =
2484a86a88eaSTakashi Iwai 	HDA_BIND_SW("Capture Switch", 0);
2485a86a88eaSTakashi Iwai 
2486a86a88eaSTakashi Iwai static int init_bind_ctl(struct via_spec *spec, struct hda_bind_ctls **ctl_ret,
2487a86a88eaSTakashi Iwai 			 struct hda_ctl_ops *ops)
2488a86a88eaSTakashi Iwai {
2489a86a88eaSTakashi Iwai 	struct hda_bind_ctls *ctl;
2490a86a88eaSTakashi Iwai 	int i;
2491a86a88eaSTakashi Iwai 
2492a86a88eaSTakashi Iwai 	ctl = kzalloc(sizeof(*ctl) + sizeof(long) * 4, GFP_KERNEL);
2493a86a88eaSTakashi Iwai 	if (!ctl)
2494a86a88eaSTakashi Iwai 		return -ENOMEM;
2495a86a88eaSTakashi Iwai 	ctl->ops = ops;
2496a86a88eaSTakashi Iwai 	for (i = 0; i < spec->num_adc_nids; i++)
2497a86a88eaSTakashi Iwai 		ctl->values[i] =
2498a86a88eaSTakashi Iwai 			HDA_COMPOSE_AMP_VAL(spec->adc_nids[i], 3, 0, HDA_INPUT);
2499a86a88eaSTakashi Iwai 	*ctl_ret = ctl;
2500a86a88eaSTakashi Iwai 	return 0;
2501a86a88eaSTakashi Iwai }
2502a86a88eaSTakashi Iwai 
2503a86a88eaSTakashi Iwai /* create capture and input-src controls for dynamic ADC-switch case */
2504a86a88eaSTakashi Iwai static int create_dyn_adc_ctls(struct hda_codec *codec)
2505a86a88eaSTakashi Iwai {
2506a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2507a86a88eaSTakashi Iwai 	struct snd_kcontrol_new *knew;
2508a86a88eaSTakashi Iwai 	int err;
2509a86a88eaSTakashi Iwai 
2510a86a88eaSTakashi Iwai 	/* set up the bind capture ctls */
2511a86a88eaSTakashi Iwai 	err = init_bind_ctl(spec, &spec->bind_cap_vol, &snd_hda_bind_vol);
2512a86a88eaSTakashi Iwai 	if (err < 0)
2513a86a88eaSTakashi Iwai 		return err;
2514a86a88eaSTakashi Iwai 	err = init_bind_ctl(spec, &spec->bind_cap_sw, &snd_hda_bind_sw);
2515a86a88eaSTakashi Iwai 	if (err < 0)
2516a86a88eaSTakashi Iwai 		return err;
2517a86a88eaSTakashi Iwai 
2518a86a88eaSTakashi Iwai 	/* create capture mixer elements */
2519a86a88eaSTakashi Iwai 	knew = via_clone_control(spec, &via_bind_cap_vol_ctl);
2520a86a88eaSTakashi Iwai 	if (!knew)
2521a86a88eaSTakashi Iwai 		return -ENOMEM;
2522a86a88eaSTakashi Iwai 	knew->private_value = (long)spec->bind_cap_vol;
2523a86a88eaSTakashi Iwai 
2524a86a88eaSTakashi Iwai 	knew = via_clone_control(spec, &via_bind_cap_sw_ctl);
2525a86a88eaSTakashi Iwai 	if (!knew)
2526a86a88eaSTakashi Iwai 		return -ENOMEM;
2527a86a88eaSTakashi Iwai 	knew->private_value = (long)spec->bind_cap_sw;
2528a86a88eaSTakashi Iwai 
2529a86a88eaSTakashi Iwai 	/* input-source control */
2530a86a88eaSTakashi Iwai 	err = create_input_src_ctls(codec, 1);
2531a86a88eaSTakashi Iwai 	if (err < 0)
2532a86a88eaSTakashi Iwai 		return err;
2533a86a88eaSTakashi Iwai 	return 0;
2534a86a88eaSTakashi Iwai }
2535a86a88eaSTakashi Iwai 
2536a86a88eaSTakashi Iwai /* parse and create capture-related stuff */
2537a86a88eaSTakashi Iwai static int via_auto_create_analog_input_ctls(struct hda_codec *codec)
2538a86a88eaSTakashi Iwai {
2539a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2540a86a88eaSTakashi Iwai 	int err;
2541a86a88eaSTakashi Iwai 
2542a86a88eaSTakashi Iwai 	err = parse_analog_inputs(codec);
2543a86a88eaSTakashi Iwai 	if (err < 0)
2544a86a88eaSTakashi Iwai 		return err;
2545a86a88eaSTakashi Iwai 	if (spec->dyn_adc_switch)
2546a86a88eaSTakashi Iwai 		err = create_dyn_adc_ctls(codec);
2547a86a88eaSTakashi Iwai 	else
2548a86a88eaSTakashi Iwai 		err = create_multi_adc_ctls(codec);
2549a86a88eaSTakashi Iwai 	if (err < 0)
2550a86a88eaSTakashi Iwai 		return err;
2551a86a88eaSTakashi Iwai 	err = create_loopback_ctls(codec);
2552a86a88eaSTakashi Iwai 	if (err < 0)
2553a86a88eaSTakashi Iwai 		return err;
2554a86a88eaSTakashi Iwai 	err = create_mic_boost_ctls(codec);
2555a86a88eaSTakashi Iwai 	if (err < 0)
2556a86a88eaSTakashi Iwai 		return err;
2557c577b8a1SJoseph Chan 	return 0;
2558c577b8a1SJoseph Chan }
2559c577b8a1SJoseph Chan 
256076d9b0ddSHarald Welte static void vt1708_set_pinconfig_connect(struct hda_codec *codec, hda_nid_t nid)
256176d9b0ddSHarald Welte {
256276d9b0ddSHarald Welte 	unsigned int def_conf;
256376d9b0ddSHarald Welte 	unsigned char seqassoc;
256476d9b0ddSHarald Welte 
25652f334f92STakashi Iwai 	def_conf = snd_hda_codec_get_pincfg(codec, nid);
256676d9b0ddSHarald Welte 	seqassoc = (unsigned char) get_defcfg_association(def_conf);
256776d9b0ddSHarald Welte 	seqassoc = (seqassoc << 4) | get_defcfg_sequence(def_conf);
256882ef9e45SLydia Wang 	if (get_defcfg_connect(def_conf) == AC_JACK_PORT_NONE
256982ef9e45SLydia Wang 	    && (seqassoc == 0xf0 || seqassoc == 0xff)) {
257076d9b0ddSHarald Welte 		def_conf = def_conf & (~(AC_JACK_PORT_BOTH << 30));
25712f334f92STakashi Iwai 		snd_hda_codec_set_pincfg(codec, nid, def_conf);
257276d9b0ddSHarald Welte 	}
257376d9b0ddSHarald Welte 
257476d9b0ddSHarald Welte 	return;
257576d9b0ddSHarald Welte }
257676d9b0ddSHarald Welte 
2577e06e5a29STakashi Iwai static int vt1708_jack_detect_get(struct snd_kcontrol *kcontrol,
25781f2e99feSLydia Wang 				     struct snd_ctl_elem_value *ucontrol)
25791f2e99feSLydia Wang {
25801f2e99feSLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
25811f2e99feSLydia Wang 	struct via_spec *spec = codec->spec;
25821f2e99feSLydia Wang 
25831f2e99feSLydia Wang 	if (spec->codec_type != VT1708)
25841f2e99feSLydia Wang 		return 0;
2585e06e5a29STakashi Iwai 	ucontrol->value.integer.value[0] = spec->vt1708_jack_detect;
25861f2e99feSLydia Wang 	return 0;
25871f2e99feSLydia Wang }
25881f2e99feSLydia Wang 
2589e06e5a29STakashi Iwai static int vt1708_jack_detect_put(struct snd_kcontrol *kcontrol,
25901f2e99feSLydia Wang 				     struct snd_ctl_elem_value *ucontrol)
25911f2e99feSLydia Wang {
25921f2e99feSLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
25931f2e99feSLydia Wang 	struct via_spec *spec = codec->spec;
2594187d333eSTakashi Iwai 	int val;
25951f2e99feSLydia Wang 
25961f2e99feSLydia Wang 	if (spec->codec_type != VT1708)
25971f2e99feSLydia Wang 		return 0;
2598187d333eSTakashi Iwai 	val = !!ucontrol->value.integer.value[0];
2599187d333eSTakashi Iwai 	if (spec->vt1708_jack_detect == val)
2600187d333eSTakashi Iwai 		return 0;
2601187d333eSTakashi Iwai 	spec->vt1708_jack_detect = val;
2602187d333eSTakashi Iwai 	if (spec->vt1708_jack_detect &&
2603187d333eSTakashi Iwai 	    snd_hda_get_bool_hint(codec, "analog_loopback_hp_detect") != 1) {
26041f2e99feSLydia Wang 		mute_aa_path(codec, 1);
26051f2e99feSLydia Wang 		notify_aa_path_ctls(codec);
26061f2e99feSLydia Wang 	}
2607187d333eSTakashi Iwai 	via_hp_automute(codec);
2608187d333eSTakashi Iwai 	vt1708_update_hp_work(spec);
2609187d333eSTakashi Iwai 	return 1;
26101f2e99feSLydia Wang }
26111f2e99feSLydia Wang 
2612e06e5a29STakashi Iwai static const struct snd_kcontrol_new vt1708_jack_detect_ctl = {
26131f2e99feSLydia Wang 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
26141f2e99feSLydia Wang 	.name = "Jack Detect",
26151f2e99feSLydia Wang 	.count = 1,
26161f2e99feSLydia Wang 	.info = snd_ctl_boolean_mono_info,
2617e06e5a29STakashi Iwai 	.get = vt1708_jack_detect_get,
2618e06e5a29STakashi Iwai 	.put = vt1708_jack_detect_put,
26191f2e99feSLydia Wang };
26201f2e99feSLydia Wang 
262112daef65STakashi Iwai static void fill_dig_outs(struct hda_codec *codec);
262212daef65STakashi Iwai static void fill_dig_in(struct hda_codec *codec);
262312daef65STakashi Iwai 
262412daef65STakashi Iwai static int via_parse_auto_config(struct hda_codec *codec)
2625c577b8a1SJoseph Chan {
2626c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
2627c577b8a1SJoseph Chan 	int err;
2628c577b8a1SJoseph Chan 
2629c577b8a1SJoseph Chan 	err = snd_hda_parse_pin_def_config(codec, &spec->autocfg, NULL);
2630c577b8a1SJoseph Chan 	if (err < 0)
2631c577b8a1SJoseph Chan 		return err;
2632c577b8a1SJoseph Chan 	if (!spec->autocfg.line_outs && !spec->autocfg.hp_pins[0])
26337f0df88cSTakashi Iwai 		return -EINVAL;
2634c577b8a1SJoseph Chan 
26354a79616dSTakashi Iwai 	err = via_auto_create_multi_out_ctls(codec);
2636c577b8a1SJoseph Chan 	if (err < 0)
2637c577b8a1SJoseph Chan 		return err;
26384a79616dSTakashi Iwai 	err = via_auto_create_hp_ctls(codec, spec->autocfg.hp_pins[0]);
2639c577b8a1SJoseph Chan 	if (err < 0)
2640c577b8a1SJoseph Chan 		return err;
26414a918ffeSTakashi Iwai 	err = via_auto_create_speaker_ctls(codec);
26424a918ffeSTakashi Iwai 	if (err < 0)
26434a918ffeSTakashi Iwai 		return err;
26443214b966STakashi Iwai 	err = via_auto_create_loopback_switch(codec);
26453214b966STakashi Iwai 	if (err < 0)
26463214b966STakashi Iwai 		return err;
2647a86a88eaSTakashi Iwai 	err = via_auto_create_analog_input_ctls(codec);
2648c577b8a1SJoseph Chan 	if (err < 0)
2649c577b8a1SJoseph Chan 		return err;
2650c577b8a1SJoseph Chan 
2651c577b8a1SJoseph Chan 	spec->multiout.max_channels = spec->multiout.num_dacs * 2;
2652c577b8a1SJoseph Chan 
265312daef65STakashi Iwai 	fill_dig_outs(codec);
265412daef65STakashi Iwai 	fill_dig_in(codec);
2655c577b8a1SJoseph Chan 
2656603c4019STakashi Iwai 	if (spec->kctls.list)
2657603c4019STakashi Iwai 		spec->mixers[spec->num_mixers++] = spec->kctls.list;
2658c577b8a1SJoseph Chan 
2659c577b8a1SJoseph Chan 
26603214b966STakashi Iwai 	if (spec->hp_dac_nid && spec->hp_mix_path.depth) {
2661ece8d043STakashi Iwai 		err = via_hp_build(codec);
2662ece8d043STakashi Iwai 		if (err < 0)
2663ece8d043STakashi Iwai 			return err;
2664ece8d043STakashi Iwai 	}
2665c577b8a1SJoseph Chan 
2666f4a7828bSTakashi Iwai 	err = via_smart51_build(codec);
2667f4a7828bSTakashi Iwai 	if (err < 0)
2668f4a7828bSTakashi Iwai 		return err;
2669f4a7828bSTakashi Iwai 
26705d41762aSTakashi Iwai 	/* assign slave outs */
26715d41762aSTakashi Iwai 	if (spec->slave_dig_outs[0])
26725d41762aSTakashi Iwai 		codec->slave_dig_outs = spec->slave_dig_outs;
26735d41762aSTakashi Iwai 
2674c577b8a1SJoseph Chan 	return 1;
2675c577b8a1SJoseph Chan }
2676c577b8a1SJoseph Chan 
26775d41762aSTakashi Iwai static void via_auto_init_dig_outs(struct hda_codec *codec)
2678c577b8a1SJoseph Chan {
267925eaba2fSLydia Wang 	struct via_spec *spec = codec->spec;
26805d41762aSTakashi Iwai 	if (spec->multiout.dig_out_nid)
26815d41762aSTakashi Iwai 		init_output_pin(codec, spec->autocfg.dig_out_pins[0], PIN_OUT);
26825d41762aSTakashi Iwai 	if (spec->slave_dig_outs[0])
26835d41762aSTakashi Iwai 		init_output_pin(codec, spec->autocfg.dig_out_pins[1], PIN_OUT);
26845d41762aSTakashi Iwai }
268525eaba2fSLydia Wang 
26865d41762aSTakashi Iwai static void via_auto_init_dig_in(struct hda_codec *codec)
26875d41762aSTakashi Iwai {
26885d41762aSTakashi Iwai 	struct via_spec *spec = codec->spec;
26895d41762aSTakashi Iwai 	if (!spec->dig_in_nid)
26905d41762aSTakashi Iwai 		return;
2691cdd03cedSTakashi Iwai 	snd_hda_set_pin_ctl(codec, spec->autocfg.dig_in_pin, PIN_IN);
26925d41762aSTakashi Iwai }
26935d41762aSTakashi Iwai 
26944e2d16d3SDavid Henningsson static void via_jack_output_event(struct hda_codec *codec, struct hda_jack_tbl *tbl)
26954e2d16d3SDavid Henningsson {
26964e2d16d3SDavid Henningsson 	set_widgets_power_state(codec);
26974e2d16d3SDavid Henningsson 	via_hp_automute(codec);
26984e2d16d3SDavid Henningsson }
26994e2d16d3SDavid Henningsson 
27004e2d16d3SDavid Henningsson static void via_jack_powerstate_event(struct hda_codec *codec, struct hda_jack_tbl *tbl)
27014e2d16d3SDavid Henningsson {
27024e2d16d3SDavid Henningsson 	set_widgets_power_state(codec);
27034e2d16d3SDavid Henningsson }
27044e2d16d3SDavid Henningsson 
27054a918ffeSTakashi Iwai /* initialize the unsolicited events */
27064a918ffeSTakashi Iwai static void via_auto_init_unsol_event(struct hda_codec *codec)
27074a918ffeSTakashi Iwai {
27084a918ffeSTakashi Iwai 	struct via_spec *spec = codec->spec;
27094a918ffeSTakashi Iwai 	struct auto_pin_cfg *cfg = &spec->autocfg;
27104a918ffeSTakashi Iwai 	unsigned int ev;
27114a918ffeSTakashi Iwai 	int i;
27124e2d16d3SDavid Henningsson 	hda_jack_callback cb;
27134a918ffeSTakashi Iwai 
27144a918ffeSTakashi Iwai 	if (cfg->hp_pins[0] && is_jack_detectable(codec, cfg->hp_pins[0]))
27154e2d16d3SDavid Henningsson 		snd_hda_jack_detect_enable_callback(codec, cfg->hp_pins[0],
27164e2d16d3SDavid Henningsson 						    VIA_HP_EVENT | VIA_JACK_EVENT,
27174e2d16d3SDavid Henningsson 						    via_jack_output_event);
27184a918ffeSTakashi Iwai 
27194a918ffeSTakashi Iwai 	if (cfg->speaker_pins[0])
27204a918ffeSTakashi Iwai 		ev = VIA_LINE_EVENT;
27214a918ffeSTakashi Iwai 	else
27224a918ffeSTakashi Iwai 		ev = 0;
27234e2d16d3SDavid Henningsson 	cb = ev ? via_jack_output_event : via_jack_powerstate_event;
27244e2d16d3SDavid Henningsson 
27254a918ffeSTakashi Iwai 	for (i = 0; i < cfg->line_outs; i++) {
27264a918ffeSTakashi Iwai 		if (cfg->line_out_pins[i] &&
27274a918ffeSTakashi Iwai 		    is_jack_detectable(codec, cfg->line_out_pins[i]))
27284e2d16d3SDavid Henningsson 			snd_hda_jack_detect_enable_callback(codec, cfg->line_out_pins[i],
27294e2d16d3SDavid Henningsson 							    ev | VIA_JACK_EVENT, cb);
27304a918ffeSTakashi Iwai 	}
27314a918ffeSTakashi Iwai 
27324a918ffeSTakashi Iwai 	for (i = 0; i < cfg->num_inputs; i++) {
27334a918ffeSTakashi Iwai 		if (is_jack_detectable(codec, cfg->inputs[i].pin))
27344e2d16d3SDavid Henningsson 			snd_hda_jack_detect_enable_callback(codec, cfg->inputs[i].pin,
27354e2d16d3SDavid Henningsson 							    VIA_JACK_EVENT,
27364e2d16d3SDavid Henningsson 							    via_jack_powerstate_event);
27374a918ffeSTakashi Iwai 	}
27384a918ffeSTakashi Iwai }
27394a918ffeSTakashi Iwai 
27405d41762aSTakashi Iwai static int via_init(struct hda_codec *codec)
27415d41762aSTakashi Iwai {
27425d41762aSTakashi Iwai 	struct via_spec *spec = codec->spec;
27435d41762aSTakashi Iwai 	int i;
27445d41762aSTakashi Iwai 
27455d41762aSTakashi Iwai 	for (i = 0; i < spec->num_iverbs; i++)
27465d41762aSTakashi Iwai 		snd_hda_sequence_write(codec, spec->init_verbs[i]);
27475d41762aSTakashi Iwai 
2748e9d010c2STakashi Iwai 	/* init power states */
2749e9d010c2STakashi Iwai 	set_widgets_power_state(codec);
2750e9d010c2STakashi Iwai 	__analog_low_current_mode(codec, true);
2751e9d010c2STakashi Iwai 
2752c577b8a1SJoseph Chan 	via_auto_init_multi_out(codec);
2753c577b8a1SJoseph Chan 	via_auto_init_hp_out(codec);
27544a918ffeSTakashi Iwai 	via_auto_init_speaker_out(codec);
2755c577b8a1SJoseph Chan 	via_auto_init_analog_input(codec);
27565d41762aSTakashi Iwai 	via_auto_init_dig_outs(codec);
27575d41762aSTakashi Iwai 	via_auto_init_dig_in(codec);
275811890956SLydia Wang 
27594a918ffeSTakashi Iwai 	via_auto_init_unsol_event(codec);
27604a918ffeSTakashi Iwai 
276125eaba2fSLydia Wang 	via_hp_automute(codec);
2762187d333eSTakashi Iwai 	vt1708_update_hp_work(spec);
276325eaba2fSLydia Wang 
2764c577b8a1SJoseph Chan 	return 0;
2765c577b8a1SJoseph Chan }
2766c577b8a1SJoseph Chan 
27671f2e99feSLydia Wang static void vt1708_update_hp_jack_state(struct work_struct *work)
27681f2e99feSLydia Wang {
27691f2e99feSLydia Wang 	struct via_spec *spec = container_of(work, struct via_spec,
27701f2e99feSLydia Wang 					     vt1708_hp_work.work);
27711f2e99feSLydia Wang 	if (spec->codec_type != VT1708)
27721f2e99feSLydia Wang 		return;
27731835a0f9STakashi Iwai 	snd_hda_jack_set_dirty_all(spec->codec);
27741f2e99feSLydia Wang 	/* if jack state toggled */
27751f2e99feSLydia Wang 	if (spec->vt1708_hp_present
2776d56757abSTakashi Iwai 	    != snd_hda_jack_detect(spec->codec, spec->autocfg.hp_pins[0])) {
27771f2e99feSLydia Wang 		spec->vt1708_hp_present ^= 1;
27781f2e99feSLydia Wang 		via_hp_automute(spec->codec);
27791f2e99feSLydia Wang 	}
2780187d333eSTakashi Iwai 	if (spec->vt1708_jack_detect)
2781187d333eSTakashi Iwai 		schedule_delayed_work(&spec->vt1708_hp_work,
2782187d333eSTakashi Iwai 				      msecs_to_jiffies(100));
27831f2e99feSLydia Wang }
27841f2e99feSLydia Wang 
2785337b9d02STakashi Iwai static int get_mux_nids(struct hda_codec *codec)
2786337b9d02STakashi Iwai {
2787337b9d02STakashi Iwai 	struct via_spec *spec = codec->spec;
2788337b9d02STakashi Iwai 	hda_nid_t nid, conn[8];
2789337b9d02STakashi Iwai 	unsigned int type;
2790337b9d02STakashi Iwai 	int i, n;
2791337b9d02STakashi Iwai 
2792337b9d02STakashi Iwai 	for (i = 0; i < spec->num_adc_nids; i++) {
2793337b9d02STakashi Iwai 		nid = spec->adc_nids[i];
2794337b9d02STakashi Iwai 		while (nid) {
2795a22d543aSTakashi Iwai 			type = get_wcaps_type(get_wcaps(codec, nid));
27961c55d521STakashi Iwai 			if (type == AC_WID_PIN)
27971c55d521STakashi Iwai 				break;
2798337b9d02STakashi Iwai 			n = snd_hda_get_connections(codec, nid, conn,
2799337b9d02STakashi Iwai 						    ARRAY_SIZE(conn));
2800337b9d02STakashi Iwai 			if (n <= 0)
2801337b9d02STakashi Iwai 				break;
2802337b9d02STakashi Iwai 			if (n > 1) {
2803337b9d02STakashi Iwai 				spec->mux_nids[i] = nid;
2804337b9d02STakashi Iwai 				break;
2805337b9d02STakashi Iwai 			}
2806337b9d02STakashi Iwai 			nid = conn[0];
2807337b9d02STakashi Iwai 		}
2808337b9d02STakashi Iwai 	}
28091c55d521STakashi Iwai 	return 0;
2810337b9d02STakashi Iwai }
2811337b9d02STakashi Iwai 
2812c577b8a1SJoseph Chan static int patch_vt1708(struct hda_codec *codec)
2813c577b8a1SJoseph Chan {
2814c577b8a1SJoseph Chan 	struct via_spec *spec;
2815c577b8a1SJoseph Chan 	int err;
2816c577b8a1SJoseph Chan 
2817c577b8a1SJoseph Chan 	/* create a codec specific record */
28185b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
2819c577b8a1SJoseph Chan 	if (spec == NULL)
2820c577b8a1SJoseph Chan 		return -ENOMEM;
2821c577b8a1SJoseph Chan 
2822620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x17;
2823620e2b28STakashi Iwai 
282412daef65STakashi Iwai 	/* Add HP and CD pin config connect bit re-config action */
282512daef65STakashi Iwai 	vt1708_set_pinconfig_connect(codec, VT1708_HP_PIN_NID);
282612daef65STakashi Iwai 	vt1708_set_pinconfig_connect(codec, VT1708_CD_PIN_NID);
282712daef65STakashi Iwai 
2828c577b8a1SJoseph Chan 	/* automatic parse from the BIOS config */
282912daef65STakashi Iwai 	err = via_parse_auto_config(codec);
2830c577b8a1SJoseph Chan 	if (err < 0) {
2831c577b8a1SJoseph Chan 		via_free(codec);
2832c577b8a1SJoseph Chan 		return err;
2833c577b8a1SJoseph Chan 	}
2834c577b8a1SJoseph Chan 
283512daef65STakashi Iwai 	/* add jack detect on/off control */
283612daef65STakashi Iwai 	if (!via_clone_control(spec, &vt1708_jack_detect_ctl))
283712daef65STakashi Iwai 		return -ENOMEM;
283812daef65STakashi Iwai 
2839bc9b5623STakashi Iwai 	/* disable 32bit format on VT1708 */
2840bc9b5623STakashi Iwai 	if (codec->vendor_id == 0x11061708)
2841bc9b5623STakashi Iwai 		spec->stream_analog_playback = &vt1708_pcm_analog_s16_playback;
2842c577b8a1SJoseph Chan 
2843e322a36dSLydia Wang 	spec->init_verbs[spec->num_iverbs++] = vt1708_init_verbs;
2844e322a36dSLydia Wang 
2845c577b8a1SJoseph Chan 	codec->patch_ops = via_patch_ops;
2846c577b8a1SJoseph Chan 
28471f2e99feSLydia Wang 	INIT_DELAYED_WORK(&spec->vt1708_hp_work, vt1708_update_hp_jack_state);
2848c577b8a1SJoseph Chan 	return 0;
2849c577b8a1SJoseph Chan }
2850c577b8a1SJoseph Chan 
2851ddd304d8STakashi Iwai static int patch_vt1709(struct hda_codec *codec)
2852c577b8a1SJoseph Chan {
2853c577b8a1SJoseph Chan 	struct via_spec *spec;
2854c577b8a1SJoseph Chan 	int err;
2855c577b8a1SJoseph Chan 
2856c577b8a1SJoseph Chan 	/* create a codec specific record */
28575b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
2858c577b8a1SJoseph Chan 	if (spec == NULL)
2859c577b8a1SJoseph Chan 		return -ENOMEM;
2860c577b8a1SJoseph Chan 
2861620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x18;
2862620e2b28STakashi Iwai 
286312daef65STakashi Iwai 	err = via_parse_auto_config(codec);
2864c577b8a1SJoseph Chan 	if (err < 0) {
2865c577b8a1SJoseph Chan 		via_free(codec);
2866c577b8a1SJoseph Chan 		return err;
2867c577b8a1SJoseph Chan 	}
2868c577b8a1SJoseph Chan 
2869c577b8a1SJoseph Chan 	codec->patch_ops = via_patch_ops;
2870c577b8a1SJoseph Chan 
2871f7278fd0SJosepch Chan 	return 0;
2872f7278fd0SJosepch Chan }
2873f7278fd0SJosepch Chan 
28743e95b9abSLydia Wang static void set_widgets_power_state_vt1708B(struct hda_codec *codec)
28753e95b9abSLydia Wang {
28763e95b9abSLydia Wang 	struct via_spec *spec = codec->spec;
28773e95b9abSLydia Wang 	int imux_is_smixer;
28783e95b9abSLydia Wang 	unsigned int parm;
28793e95b9abSLydia Wang 	int is_8ch = 0;
2880bc92df7fSLydia Wang 	if ((spec->codec_type != VT1708B_4CH) &&
2881bc92df7fSLydia Wang 	    (codec->vendor_id != 0x11064397))
28823e95b9abSLydia Wang 		is_8ch = 1;
28833e95b9abSLydia Wang 
28843e95b9abSLydia Wang 	/* SW0 (17h) = stereo mixer */
28853e95b9abSLydia Wang 	imux_is_smixer =
28863e95b9abSLydia Wang 	(snd_hda_codec_read(codec, 0x17, 0, AC_VERB_GET_CONNECT_SEL, 0x00)
28873e95b9abSLydia Wang 	 == ((spec->codec_type == VT1708S) ? 5 : 0));
28883e95b9abSLydia Wang 	/* inputs */
28893e95b9abSLydia Wang 	/* PW 1/2/5 (1ah/1bh/1eh) */
28903e95b9abSLydia Wang 	parm = AC_PWRST_D3;
28913e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1a, &parm);
28923e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1b, &parm);
28933e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1e, &parm);
28943e95b9abSLydia Wang 	if (imux_is_smixer)
28953e95b9abSLydia Wang 		parm = AC_PWRST_D0;
28963e95b9abSLydia Wang 	/* SW0 (17h), AIW 0/1 (13h/14h) */
2897054d867eSTakashi Iwai 	update_power_state(codec, 0x17, parm);
2898054d867eSTakashi Iwai 	update_power_state(codec, 0x13, parm);
2899054d867eSTakashi Iwai 	update_power_state(codec, 0x14, parm);
29003e95b9abSLydia Wang 
29013e95b9abSLydia Wang 	/* outputs */
29023e95b9abSLydia Wang 	/* PW0 (19h), SW1 (18h), AOW1 (11h) */
29033e95b9abSLydia Wang 	parm = AC_PWRST_D3;
29043e95b9abSLydia Wang 	set_pin_power_state(codec, 0x19, &parm);
29053e95b9abSLydia Wang 	if (spec->smart51_enabled)
29063e95b9abSLydia Wang 		set_pin_power_state(codec, 0x1b, &parm);
2907054d867eSTakashi Iwai 	update_power_state(codec, 0x18, parm);
2908054d867eSTakashi Iwai 	update_power_state(codec, 0x11, parm);
29093e95b9abSLydia Wang 
29103e95b9abSLydia Wang 	/* PW6 (22h), SW2 (26h), AOW2 (24h) */
29113e95b9abSLydia Wang 	if (is_8ch) {
29123e95b9abSLydia Wang 		parm = AC_PWRST_D3;
29133e95b9abSLydia Wang 		set_pin_power_state(codec, 0x22, &parm);
29143e95b9abSLydia Wang 		if (spec->smart51_enabled)
29153e95b9abSLydia Wang 			set_pin_power_state(codec, 0x1a, &parm);
2916054d867eSTakashi Iwai 		update_power_state(codec, 0x26, parm);
2917054d867eSTakashi Iwai 		update_power_state(codec, 0x24, parm);
2918bc92df7fSLydia Wang 	} else if (codec->vendor_id == 0x11064397) {
2919bc92df7fSLydia Wang 		/* PW7(23h), SW2(27h), AOW2(25h) */
2920bc92df7fSLydia Wang 		parm = AC_PWRST_D3;
2921bc92df7fSLydia Wang 		set_pin_power_state(codec, 0x23, &parm);
2922bc92df7fSLydia Wang 		if (spec->smart51_enabled)
2923bc92df7fSLydia Wang 			set_pin_power_state(codec, 0x1a, &parm);
2924054d867eSTakashi Iwai 		update_power_state(codec, 0x27, parm);
2925054d867eSTakashi Iwai 		update_power_state(codec, 0x25, parm);
29263e95b9abSLydia Wang 	}
29273e95b9abSLydia Wang 
29283e95b9abSLydia Wang 	/* PW 3/4/7 (1ch/1dh/23h) */
29293e95b9abSLydia Wang 	parm = AC_PWRST_D3;
29303e95b9abSLydia Wang 	/* force to D0 for internal Speaker */
29313e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1c, &parm);
29323e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1d, &parm);
29333e95b9abSLydia Wang 	if (is_8ch)
29343e95b9abSLydia Wang 		set_pin_power_state(codec, 0x23, &parm);
29353e95b9abSLydia Wang 
29363e95b9abSLydia Wang 	/* MW0 (16h), Sw3 (27h), AOW 0/3 (10h/25h) */
2937054d867eSTakashi Iwai 	update_power_state(codec, 0x16, imux_is_smixer ? AC_PWRST_D0 : parm);
2938054d867eSTakashi Iwai 	update_power_state(codec, 0x10, parm);
29393e95b9abSLydia Wang 	if (is_8ch) {
2940054d867eSTakashi Iwai 		update_power_state(codec, 0x25, parm);
2941054d867eSTakashi Iwai 		update_power_state(codec, 0x27, parm);
2942bc92df7fSLydia Wang 	} else if (codec->vendor_id == 0x11064397 && spec->hp_independent_mode)
2943054d867eSTakashi Iwai 		update_power_state(codec, 0x25, parm);
29443e95b9abSLydia Wang }
29453e95b9abSLydia Wang 
2946518bf3baSLydia Wang static int patch_vt1708S(struct hda_codec *codec);
2947ddd304d8STakashi Iwai static int patch_vt1708B(struct hda_codec *codec)
2948f7278fd0SJosepch Chan {
2949f7278fd0SJosepch Chan 	struct via_spec *spec;
2950f7278fd0SJosepch Chan 	int err;
2951f7278fd0SJosepch Chan 
2952518bf3baSLydia Wang 	if (get_codec_type(codec) == VT1708BCE)
2953518bf3baSLydia Wang 		return patch_vt1708S(codec);
2954ddd304d8STakashi Iwai 
2955f7278fd0SJosepch Chan 	/* create a codec specific record */
29565b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
2957f7278fd0SJosepch Chan 	if (spec == NULL)
2958f7278fd0SJosepch Chan 		return -ENOMEM;
2959f7278fd0SJosepch Chan 
2960620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x16;
2961620e2b28STakashi Iwai 
2962f7278fd0SJosepch Chan 	/* automatic parse from the BIOS config */
296312daef65STakashi Iwai 	err = via_parse_auto_config(codec);
2964f7278fd0SJosepch Chan 	if (err < 0) {
2965f7278fd0SJosepch Chan 		via_free(codec);
2966f7278fd0SJosepch Chan 		return err;
2967f7278fd0SJosepch Chan 	}
2968f7278fd0SJosepch Chan 
2969f7278fd0SJosepch Chan 	codec->patch_ops = via_patch_ops;
2970f7278fd0SJosepch Chan 
29713e95b9abSLydia Wang 	spec->set_widgets_power_state =  set_widgets_power_state_vt1708B;
29723e95b9abSLydia Wang 
2973f7278fd0SJosepch Chan 	return 0;
2974f7278fd0SJosepch Chan }
2975f7278fd0SJosepch Chan 
2976d949cac1SHarald Welte /* Patch for VT1708S */
2977096a8854STakashi Iwai static const struct hda_verb vt1708S_init_verbs[] = {
2978d7426329SHarald Welte 	/* Enable Mic Boost Volume backdoor */
2979d7426329SHarald Welte 	{0x1, 0xf98, 0x1},
2980bc7e7e5cSLydia Wang 	/* don't bybass mixer */
2981bc7e7e5cSLydia Wang 	{0x1, 0xf88, 0xc0},
2982d949cac1SHarald Welte 	{ }
2983d949cac1SHarald Welte };
2984d949cac1SHarald Welte 
29859da29271STakashi Iwai /* fill out digital output widgets; one for master and one for slave outputs */
29869da29271STakashi Iwai static void fill_dig_outs(struct hda_codec *codec)
29879da29271STakashi Iwai {
29889da29271STakashi Iwai 	struct via_spec *spec = codec->spec;
29899da29271STakashi Iwai 	int i;
29909da29271STakashi Iwai 
29919da29271STakashi Iwai 	for (i = 0; i < spec->autocfg.dig_outs; i++) {
29929da29271STakashi Iwai 		hda_nid_t nid;
29939da29271STakashi Iwai 		int conn;
29949da29271STakashi Iwai 
29959da29271STakashi Iwai 		nid = spec->autocfg.dig_out_pins[i];
29969da29271STakashi Iwai 		if (!nid)
29979da29271STakashi Iwai 			continue;
29989da29271STakashi Iwai 		conn = snd_hda_get_connections(codec, nid, &nid, 1);
29999da29271STakashi Iwai 		if (conn < 1)
30009da29271STakashi Iwai 			continue;
30019da29271STakashi Iwai 		if (!spec->multiout.dig_out_nid)
30029da29271STakashi Iwai 			spec->multiout.dig_out_nid = nid;
30039da29271STakashi Iwai 		else {
30049da29271STakashi Iwai 			spec->slave_dig_outs[0] = nid;
30059da29271STakashi Iwai 			break; /* at most two dig outs */
30069da29271STakashi Iwai 		}
30079da29271STakashi Iwai 	}
30089da29271STakashi Iwai }
30099da29271STakashi Iwai 
301012daef65STakashi Iwai static void fill_dig_in(struct hda_codec *codec)
3011d949cac1SHarald Welte {
3012d949cac1SHarald Welte 	struct via_spec *spec = codec->spec;
301312daef65STakashi Iwai 	hda_nid_t dig_nid;
301412daef65STakashi Iwai 	int i, err;
3015d949cac1SHarald Welte 
301612daef65STakashi Iwai 	if (!spec->autocfg.dig_in_pin)
301712daef65STakashi Iwai 		return;
3018d949cac1SHarald Welte 
301912daef65STakashi Iwai 	dig_nid = codec->start_nid;
302012daef65STakashi Iwai 	for (i = 0; i < codec->num_nodes; i++, dig_nid++) {
302112daef65STakashi Iwai 		unsigned int wcaps = get_wcaps(codec, dig_nid);
302212daef65STakashi Iwai 		if (get_wcaps_type(wcaps) != AC_WID_AUD_IN)
302312daef65STakashi Iwai 			continue;
302412daef65STakashi Iwai 		if (!(wcaps & AC_WCAP_DIGITAL))
302512daef65STakashi Iwai 			continue;
302612daef65STakashi Iwai 		if (!(wcaps & AC_WCAP_CONN_LIST))
302712daef65STakashi Iwai 			continue;
302812daef65STakashi Iwai 		err = get_connection_index(codec, dig_nid,
302912daef65STakashi Iwai 					   spec->autocfg.dig_in_pin);
303012daef65STakashi Iwai 		if (err >= 0) {
303112daef65STakashi Iwai 			spec->dig_in_nid = dig_nid;
303212daef65STakashi Iwai 			break;
303312daef65STakashi Iwai 		}
303412daef65STakashi Iwai 	}
3035d949cac1SHarald Welte }
3036d949cac1SHarald Welte 
30376369bcfcSLydia Wang static void override_mic_boost(struct hda_codec *codec, hda_nid_t pin,
30386369bcfcSLydia Wang 			       int offset, int num_steps, int step_size)
30396369bcfcSLydia Wang {
30406369bcfcSLydia Wang 	snd_hda_override_amp_caps(codec, pin, HDA_INPUT,
30416369bcfcSLydia Wang 				  (offset << AC_AMPCAP_OFFSET_SHIFT) |
30426369bcfcSLydia Wang 				  (num_steps << AC_AMPCAP_NUM_STEPS_SHIFT) |
30436369bcfcSLydia Wang 				  (step_size << AC_AMPCAP_STEP_SIZE_SHIFT) |
30446369bcfcSLydia Wang 				  (0 << AC_AMPCAP_MUTE_SHIFT));
30456369bcfcSLydia Wang }
30466369bcfcSLydia Wang 
3047d949cac1SHarald Welte static int patch_vt1708S(struct hda_codec *codec)
3048d949cac1SHarald Welte {
3049d949cac1SHarald Welte 	struct via_spec *spec;
3050d949cac1SHarald Welte 	int err;
3051d949cac1SHarald Welte 
3052d949cac1SHarald Welte 	/* create a codec specific record */
30535b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
3054d949cac1SHarald Welte 	if (spec == NULL)
3055d949cac1SHarald Welte 		return -ENOMEM;
3056d949cac1SHarald Welte 
3057620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x16;
3058d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x1a, 0, 3, 40);
3059d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x1e, 0, 3, 40);
3060620e2b28STakashi Iwai 
3061d949cac1SHarald Welte 	/* automatic parse from the BIOS config */
306212daef65STakashi Iwai 	err = via_parse_auto_config(codec);
3063d949cac1SHarald Welte 	if (err < 0) {
3064d949cac1SHarald Welte 		via_free(codec);
3065d949cac1SHarald Welte 		return err;
3066d949cac1SHarald Welte 	}
3067d949cac1SHarald Welte 
3068096a8854STakashi Iwai 	spec->init_verbs[spec->num_iverbs++] = vt1708S_init_verbs;
3069d949cac1SHarald Welte 
3070d949cac1SHarald Welte 	codec->patch_ops = via_patch_ops;
3071d949cac1SHarald Welte 
3072518bf3baSLydia Wang 	/* correct names for VT1708BCE */
3073518bf3baSLydia Wang 	if (get_codec_type(codec) == VT1708BCE)	{
3074518bf3baSLydia Wang 		kfree(codec->chip_name);
3075518bf3baSLydia Wang 		codec->chip_name = kstrdup("VT1708BCE", GFP_KERNEL);
3076518bf3baSLydia Wang 		snprintf(codec->bus->card->mixername,
3077518bf3baSLydia Wang 			 sizeof(codec->bus->card->mixername),
3078518bf3baSLydia Wang 			 "%s %s", codec->vendor_name, codec->chip_name);
3079970f630fSLydia Wang 	}
3080bc92df7fSLydia Wang 	/* correct names for VT1705 */
3081bc92df7fSLydia Wang 	if (codec->vendor_id == 0x11064397)	{
3082bc92df7fSLydia Wang 		kfree(codec->chip_name);
3083bc92df7fSLydia Wang 		codec->chip_name = kstrdup("VT1705", GFP_KERNEL);
3084bc92df7fSLydia Wang 		snprintf(codec->bus->card->mixername,
3085bc92df7fSLydia Wang 			 sizeof(codec->bus->card->mixername),
3086bc92df7fSLydia Wang 			 "%s %s", codec->vendor_name, codec->chip_name);
3087bc92df7fSLydia Wang 	}
30883e95b9abSLydia Wang 	spec->set_widgets_power_state =  set_widgets_power_state_vt1708B;
3089d949cac1SHarald Welte 	return 0;
3090d949cac1SHarald Welte }
3091d949cac1SHarald Welte 
3092d949cac1SHarald Welte /* Patch for VT1702 */
3093d949cac1SHarald Welte 
3094096a8854STakashi Iwai static const struct hda_verb vt1702_init_verbs[] = {
3095bc7e7e5cSLydia Wang 	/* mixer enable */
3096bc7e7e5cSLydia Wang 	{0x1, 0xF88, 0x3},
3097bc7e7e5cSLydia Wang 	/* GPIO 0~2 */
3098bc7e7e5cSLydia Wang 	{0x1, 0xF82, 0x3F},
3099d949cac1SHarald Welte 	{ }
3100d949cac1SHarald Welte };
3101d949cac1SHarald Welte 
31023e95b9abSLydia Wang static void set_widgets_power_state_vt1702(struct hda_codec *codec)
31033e95b9abSLydia Wang {
31043e95b9abSLydia Wang 	int imux_is_smixer =
31053e95b9abSLydia Wang 	snd_hda_codec_read(codec, 0x13, 0, AC_VERB_GET_CONNECT_SEL, 0x00) == 3;
31063e95b9abSLydia Wang 	unsigned int parm;
31073e95b9abSLydia Wang 	/* inputs */
31083e95b9abSLydia Wang 	/* PW 1/2/5 (14h/15h/18h) */
31093e95b9abSLydia Wang 	parm = AC_PWRST_D3;
31103e95b9abSLydia Wang 	set_pin_power_state(codec, 0x14, &parm);
31113e95b9abSLydia Wang 	set_pin_power_state(codec, 0x15, &parm);
31123e95b9abSLydia Wang 	set_pin_power_state(codec, 0x18, &parm);
31133e95b9abSLydia Wang 	if (imux_is_smixer)
31143e95b9abSLydia Wang 		parm = AC_PWRST_D0; /* SW0 (13h) = stereo mixer (idx 3) */
31153e95b9abSLydia Wang 	/* SW0 (13h), AIW 0/1/2 (12h/1fh/20h) */
3116054d867eSTakashi Iwai 	update_power_state(codec, 0x13, parm);
3117054d867eSTakashi Iwai 	update_power_state(codec, 0x12, parm);
3118054d867eSTakashi Iwai 	update_power_state(codec, 0x1f, parm);
3119054d867eSTakashi Iwai 	update_power_state(codec, 0x20, parm);
31203e95b9abSLydia Wang 
31213e95b9abSLydia Wang 	/* outputs */
31223e95b9abSLydia Wang 	/* PW 3/4 (16h/17h) */
31233e95b9abSLydia Wang 	parm = AC_PWRST_D3;
31243e95b9abSLydia Wang 	set_pin_power_state(codec, 0x17, &parm);
31253e95b9abSLydia Wang 	set_pin_power_state(codec, 0x16, &parm);
31263e95b9abSLydia Wang 	/* MW0 (1ah), AOW 0/1 (10h/1dh) */
3127054d867eSTakashi Iwai 	update_power_state(codec, 0x1a, imux_is_smixer ? AC_PWRST_D0 : parm);
3128054d867eSTakashi Iwai 	update_power_state(codec, 0x10, parm);
3129054d867eSTakashi Iwai 	update_power_state(codec, 0x1d, parm);
31303e95b9abSLydia Wang }
31313e95b9abSLydia Wang 
3132d949cac1SHarald Welte static int patch_vt1702(struct hda_codec *codec)
3133d949cac1SHarald Welte {
3134d949cac1SHarald Welte 	struct via_spec *spec;
3135d949cac1SHarald Welte 	int err;
3136d949cac1SHarald Welte 
3137d949cac1SHarald Welte 	/* create a codec specific record */
31385b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
3139d949cac1SHarald Welte 	if (spec == NULL)
3140d949cac1SHarald Welte 		return -ENOMEM;
3141d949cac1SHarald Welte 
3142620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x1a;
3143620e2b28STakashi Iwai 
314412daef65STakashi Iwai 	/* limit AA path volume to 0 dB */
314512daef65STakashi Iwai 	snd_hda_override_amp_caps(codec, 0x1A, HDA_INPUT,
314612daef65STakashi Iwai 				  (0x17 << AC_AMPCAP_OFFSET_SHIFT) |
314712daef65STakashi Iwai 				  (0x17 << AC_AMPCAP_NUM_STEPS_SHIFT) |
314812daef65STakashi Iwai 				  (0x5 << AC_AMPCAP_STEP_SIZE_SHIFT) |
314912daef65STakashi Iwai 				  (1 << AC_AMPCAP_MUTE_SHIFT));
315012daef65STakashi Iwai 
3151d949cac1SHarald Welte 	/* automatic parse from the BIOS config */
315212daef65STakashi Iwai 	err = via_parse_auto_config(codec);
3153d949cac1SHarald Welte 	if (err < 0) {
3154d949cac1SHarald Welte 		via_free(codec);
3155d949cac1SHarald Welte 		return err;
3156d949cac1SHarald Welte 	}
3157d949cac1SHarald Welte 
3158096a8854STakashi Iwai 	spec->init_verbs[spec->num_iverbs++] = vt1702_init_verbs;
3159d949cac1SHarald Welte 
3160d949cac1SHarald Welte 	codec->patch_ops = via_patch_ops;
3161d949cac1SHarald Welte 
31623e95b9abSLydia Wang 	spec->set_widgets_power_state =  set_widgets_power_state_vt1702;
3163d949cac1SHarald Welte 	return 0;
3164d949cac1SHarald Welte }
3165d949cac1SHarald Welte 
3166eb7188caSLydia Wang /* Patch for VT1718S */
3167eb7188caSLydia Wang 
3168096a8854STakashi Iwai static const struct hda_verb vt1718S_init_verbs[] = {
31694ab2d53aSLydia Wang 	/* Enable MW0 adjust Gain 5 */
31704ab2d53aSLydia Wang 	{0x1, 0xfb2, 0x10},
3171eb7188caSLydia Wang 	/* Enable Boost Volume backdoor */
3172eb7188caSLydia Wang 	{0x1, 0xf88, 0x8},
31735d41762aSTakashi Iwai 
3174eb7188caSLydia Wang 	{ }
3175eb7188caSLydia Wang };
3176eb7188caSLydia Wang 
31773e95b9abSLydia Wang static void set_widgets_power_state_vt1718S(struct hda_codec *codec)
31783e95b9abSLydia Wang {
31793e95b9abSLydia Wang 	struct via_spec *spec = codec->spec;
31803e95b9abSLydia Wang 	int imux_is_smixer;
31816162552bSTakashi Iwai 	unsigned int parm, parm2;
31823e95b9abSLydia Wang 	/* MUX6 (1eh) = stereo mixer */
31833e95b9abSLydia Wang 	imux_is_smixer =
31843e95b9abSLydia Wang 	snd_hda_codec_read(codec, 0x1e, 0, AC_VERB_GET_CONNECT_SEL, 0x00) == 5;
31853e95b9abSLydia Wang 	/* inputs */
31863e95b9abSLydia Wang 	/* PW 5/6/7 (29h/2ah/2bh) */
31873e95b9abSLydia Wang 	parm = AC_PWRST_D3;
31883e95b9abSLydia Wang 	set_pin_power_state(codec, 0x29, &parm);
31893e95b9abSLydia Wang 	set_pin_power_state(codec, 0x2a, &parm);
31903e95b9abSLydia Wang 	set_pin_power_state(codec, 0x2b, &parm);
31913e95b9abSLydia Wang 	if (imux_is_smixer)
31923e95b9abSLydia Wang 		parm = AC_PWRST_D0;
31933e95b9abSLydia Wang 	/* MUX6/7 (1eh/1fh), AIW 0/1 (10h/11h) */
3194054d867eSTakashi Iwai 	update_power_state(codec, 0x1e, parm);
3195054d867eSTakashi Iwai 	update_power_state(codec, 0x1f, parm);
3196054d867eSTakashi Iwai 	update_power_state(codec, 0x10, parm);
3197054d867eSTakashi Iwai 	update_power_state(codec, 0x11, parm);
31983e95b9abSLydia Wang 
31993e95b9abSLydia Wang 	/* outputs */
32003e95b9abSLydia Wang 	/* PW3 (27h), MW2 (1ah), AOW3 (bh) */
32013e95b9abSLydia Wang 	parm = AC_PWRST_D3;
32023e95b9abSLydia Wang 	set_pin_power_state(codec, 0x27, &parm);
3203054d867eSTakashi Iwai 	update_power_state(codec, 0x1a, parm);
32046162552bSTakashi Iwai 	parm2 = parm; /* for pin 0x0b */
32053e95b9abSLydia Wang 
32063e95b9abSLydia Wang 	/* PW2 (26h), AOW2 (ah) */
32073e95b9abSLydia Wang 	parm = AC_PWRST_D3;
32083e95b9abSLydia Wang 	set_pin_power_state(codec, 0x26, &parm);
32093e95b9abSLydia Wang 	if (spec->smart51_enabled)
32103e95b9abSLydia Wang 		set_pin_power_state(codec, 0x2b, &parm);
3211054d867eSTakashi Iwai 	update_power_state(codec, 0xa, parm);
32123e95b9abSLydia Wang 
32133e95b9abSLydia Wang 	/* PW0 (24h), AOW0 (8h) */
32143e95b9abSLydia Wang 	parm = AC_PWRST_D3;
32153e95b9abSLydia Wang 	set_pin_power_state(codec, 0x24, &parm);
32163e95b9abSLydia Wang 	if (!spec->hp_independent_mode) /* check for redirected HP */
32173e95b9abSLydia Wang 		set_pin_power_state(codec, 0x28, &parm);
3218054d867eSTakashi Iwai 	update_power_state(codec, 0x8, parm);
32196162552bSTakashi Iwai 	if (!spec->hp_independent_mode && parm2 != AC_PWRST_D3)
32206162552bSTakashi Iwai 		parm = parm2;
32216162552bSTakashi Iwai 	update_power_state(codec, 0xb, parm);
32223e95b9abSLydia Wang 	/* MW9 (21h), Mw2 (1ah), AOW0 (8h) */
3223054d867eSTakashi Iwai 	update_power_state(codec, 0x21, imux_is_smixer ? AC_PWRST_D0 : parm);
32243e95b9abSLydia Wang 
32253e95b9abSLydia Wang 	/* PW1 (25h), AOW1 (9h) */
32263e95b9abSLydia Wang 	parm = AC_PWRST_D3;
32273e95b9abSLydia Wang 	set_pin_power_state(codec, 0x25, &parm);
32283e95b9abSLydia Wang 	if (spec->smart51_enabled)
32293e95b9abSLydia Wang 		set_pin_power_state(codec, 0x2a, &parm);
3230054d867eSTakashi Iwai 	update_power_state(codec, 0x9, parm);
32313e95b9abSLydia Wang 
32323e95b9abSLydia Wang 	if (spec->hp_independent_mode) {
32333e95b9abSLydia Wang 		/* PW4 (28h), MW3 (1bh), MUX1(34h), AOW4 (ch) */
32343e95b9abSLydia Wang 		parm = AC_PWRST_D3;
32353e95b9abSLydia Wang 		set_pin_power_state(codec, 0x28, &parm);
3236054d867eSTakashi Iwai 		update_power_state(codec, 0x1b, parm);
3237054d867eSTakashi Iwai 		update_power_state(codec, 0x34, parm);
3238054d867eSTakashi Iwai 		update_power_state(codec, 0xc, parm);
32393e95b9abSLydia Wang 	}
32403e95b9abSLydia Wang }
32413e95b9abSLydia Wang 
324230b45033STakashi Iwai /* Add a connection to the primary DAC from AA-mixer for some codecs
324330b45033STakashi Iwai  * This isn't listed from the raw info, but the chip has a secret connection.
324430b45033STakashi Iwai  */
324530b45033STakashi Iwai static int add_secret_dac_path(struct hda_codec *codec)
324630b45033STakashi Iwai {
324730b45033STakashi Iwai 	struct via_spec *spec = codec->spec;
324830b45033STakashi Iwai 	int i, nums;
324930b45033STakashi Iwai 	hda_nid_t conn[8];
325030b45033STakashi Iwai 	hda_nid_t nid;
325130b45033STakashi Iwai 
325230b45033STakashi Iwai 	if (!spec->aa_mix_nid)
325330b45033STakashi Iwai 		return 0;
325430b45033STakashi Iwai 	nums = snd_hda_get_connections(codec, spec->aa_mix_nid, conn,
325530b45033STakashi Iwai 				       ARRAY_SIZE(conn) - 1);
325630b45033STakashi Iwai 	for (i = 0; i < nums; i++) {
325730b45033STakashi Iwai 		if (get_wcaps_type(get_wcaps(codec, conn[i])) == AC_WID_AUD_OUT)
325830b45033STakashi Iwai 			return 0;
325930b45033STakashi Iwai 	}
326030b45033STakashi Iwai 
326130b45033STakashi Iwai 	/* find the primary DAC and add to the connection list */
326230b45033STakashi Iwai 	nid = codec->start_nid;
326330b45033STakashi Iwai 	for (i = 0; i < codec->num_nodes; i++, nid++) {
326430b45033STakashi Iwai 		unsigned int caps = get_wcaps(codec, nid);
326530b45033STakashi Iwai 		if (get_wcaps_type(caps) == AC_WID_AUD_OUT &&
326630b45033STakashi Iwai 		    !(caps & AC_WCAP_DIGITAL)) {
326730b45033STakashi Iwai 			conn[nums++] = nid;
326830b45033STakashi Iwai 			return snd_hda_override_conn_list(codec,
326930b45033STakashi Iwai 							  spec->aa_mix_nid,
327030b45033STakashi Iwai 							  nums, conn);
327130b45033STakashi Iwai 		}
327230b45033STakashi Iwai 	}
327330b45033STakashi Iwai 	return 0;
327430b45033STakashi Iwai }
327530b45033STakashi Iwai 
327630b45033STakashi Iwai 
3277eb7188caSLydia Wang static int patch_vt1718S(struct hda_codec *codec)
3278eb7188caSLydia Wang {
3279eb7188caSLydia Wang 	struct via_spec *spec;
3280eb7188caSLydia Wang 	int err;
3281eb7188caSLydia Wang 
3282eb7188caSLydia Wang 	/* create a codec specific record */
32835b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
3284eb7188caSLydia Wang 	if (spec == NULL)
3285eb7188caSLydia Wang 		return -ENOMEM;
3286eb7188caSLydia Wang 
3287620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x21;
3288d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x2b, 0, 3, 40);
3289d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x29, 0, 3, 40);
329030b45033STakashi Iwai 	add_secret_dac_path(codec);
3291620e2b28STakashi Iwai 
3292eb7188caSLydia Wang 	/* automatic parse from the BIOS config */
329312daef65STakashi Iwai 	err = via_parse_auto_config(codec);
3294eb7188caSLydia Wang 	if (err < 0) {
3295eb7188caSLydia Wang 		via_free(codec);
3296eb7188caSLydia Wang 		return err;
3297eb7188caSLydia Wang 	}
3298eb7188caSLydia Wang 
3299096a8854STakashi Iwai 	spec->init_verbs[spec->num_iverbs++] = vt1718S_init_verbs;
3300eb7188caSLydia Wang 
3301eb7188caSLydia Wang 	codec->patch_ops = via_patch_ops;
3302eb7188caSLydia Wang 
33033e95b9abSLydia Wang 	spec->set_widgets_power_state =  set_widgets_power_state_vt1718S;
33043e95b9abSLydia Wang 
3305eb7188caSLydia Wang 	return 0;
3306eb7188caSLydia Wang }
3307f3db423dSLydia Wang 
3308f3db423dSLydia Wang /* Patch for VT1716S */
3309f3db423dSLydia Wang 
3310f3db423dSLydia Wang static int vt1716s_dmic_info(struct snd_kcontrol *kcontrol,
3311f3db423dSLydia Wang 			    struct snd_ctl_elem_info *uinfo)
3312f3db423dSLydia Wang {
3313f3db423dSLydia Wang 	uinfo->type = SNDRV_CTL_ELEM_TYPE_BOOLEAN;
3314f3db423dSLydia Wang 	uinfo->count = 1;
3315f3db423dSLydia Wang 	uinfo->value.integer.min = 0;
3316f3db423dSLydia Wang 	uinfo->value.integer.max = 1;
3317f3db423dSLydia Wang 	return 0;
3318f3db423dSLydia Wang }
3319f3db423dSLydia Wang 
3320f3db423dSLydia Wang static int vt1716s_dmic_get(struct snd_kcontrol *kcontrol,
3321f3db423dSLydia Wang 			   struct snd_ctl_elem_value *ucontrol)
3322f3db423dSLydia Wang {
3323f3db423dSLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
3324f3db423dSLydia Wang 	int index = 0;
3325f3db423dSLydia Wang 
3326f3db423dSLydia Wang 	index = snd_hda_codec_read(codec, 0x26, 0,
3327f3db423dSLydia Wang 					       AC_VERB_GET_CONNECT_SEL, 0);
3328f3db423dSLydia Wang 	if (index != -1)
3329f3db423dSLydia Wang 		*ucontrol->value.integer.value = index;
3330f3db423dSLydia Wang 
3331f3db423dSLydia Wang 	return 0;
3332f3db423dSLydia Wang }
3333f3db423dSLydia Wang 
3334f3db423dSLydia Wang static int vt1716s_dmic_put(struct snd_kcontrol *kcontrol,
3335f3db423dSLydia Wang 			   struct snd_ctl_elem_value *ucontrol)
3336f3db423dSLydia Wang {
3337f3db423dSLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
3338f3db423dSLydia Wang 	struct via_spec *spec = codec->spec;
3339f3db423dSLydia Wang 	int index = *ucontrol->value.integer.value;
3340f3db423dSLydia Wang 
3341f3db423dSLydia Wang 	snd_hda_codec_write(codec, 0x26, 0,
3342f3db423dSLydia Wang 					       AC_VERB_SET_CONNECT_SEL, index);
3343f3db423dSLydia Wang 	spec->dmic_enabled = index;
33443e95b9abSLydia Wang 	set_widgets_power_state(codec);
3345f3db423dSLydia Wang 	return 1;
3346f3db423dSLydia Wang }
3347f3db423dSLydia Wang 
334890dd48a1STakashi Iwai static const struct snd_kcontrol_new vt1716s_dmic_mixer[] = {
3349f3db423dSLydia Wang 	HDA_CODEC_VOLUME("Digital Mic Capture Volume", 0x22, 0x0, HDA_INPUT),
3350f3db423dSLydia Wang 	{
3351f3db423dSLydia Wang 	 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
3352f3db423dSLydia Wang 	 .name = "Digital Mic Capture Switch",
33535b0cb1d8SJaroslav Kysela 	 .subdevice = HDA_SUBDEV_NID_FLAG | 0x26,
3354f3db423dSLydia Wang 	 .count = 1,
3355f3db423dSLydia Wang 	 .info = vt1716s_dmic_info,
3356f3db423dSLydia Wang 	 .get = vt1716s_dmic_get,
3357f3db423dSLydia Wang 	 .put = vt1716s_dmic_put,
3358f3db423dSLydia Wang 	 },
3359f3db423dSLydia Wang 	{}			/* end */
3360f3db423dSLydia Wang };
3361f3db423dSLydia Wang 
3362f3db423dSLydia Wang 
3363f3db423dSLydia Wang /* mono-out mixer elements */
336490dd48a1STakashi Iwai static const struct snd_kcontrol_new vt1716S_mono_out_mixer[] = {
3365f3db423dSLydia Wang 	HDA_CODEC_MUTE("Mono Playback Switch", 0x2a, 0x0, HDA_OUTPUT),
3366f3db423dSLydia Wang 	{ } /* end */
3367f3db423dSLydia Wang };
3368f3db423dSLydia Wang 
3369096a8854STakashi Iwai static const struct hda_verb vt1716S_init_verbs[] = {
3370f3db423dSLydia Wang 	/* Enable Boost Volume backdoor */
3371f3db423dSLydia Wang 	{0x1, 0xf8a, 0x80},
3372f3db423dSLydia Wang 	/* don't bybass mixer */
3373f3db423dSLydia Wang 	{0x1, 0xf88, 0xc0},
3374f3db423dSLydia Wang 	/* Enable mono output */
3375f3db423dSLydia Wang 	{0x1, 0xf90, 0x08},
3376f3db423dSLydia Wang 	{ }
3377f3db423dSLydia Wang };
3378f3db423dSLydia Wang 
33793e95b9abSLydia Wang static void set_widgets_power_state_vt1716S(struct hda_codec *codec)
33803e95b9abSLydia Wang {
33813e95b9abSLydia Wang 	struct via_spec *spec = codec->spec;
33823e95b9abSLydia Wang 	int imux_is_smixer;
33833e95b9abSLydia Wang 	unsigned int parm;
33843e95b9abSLydia Wang 	unsigned int mono_out, present;
33853e95b9abSLydia Wang 	/* SW0 (17h) = stereo mixer */
33863e95b9abSLydia Wang 	imux_is_smixer =
33873e95b9abSLydia Wang 	(snd_hda_codec_read(codec, 0x17, 0,
33883e95b9abSLydia Wang 			    AC_VERB_GET_CONNECT_SEL, 0x00) ==  5);
33893e95b9abSLydia Wang 	/* inputs */
33903e95b9abSLydia Wang 	/* PW 1/2/5 (1ah/1bh/1eh) */
33913e95b9abSLydia Wang 	parm = AC_PWRST_D3;
33923e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1a, &parm);
33933e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1b, &parm);
33943e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1e, &parm);
33953e95b9abSLydia Wang 	if (imux_is_smixer)
33963e95b9abSLydia Wang 		parm = AC_PWRST_D0;
33973e95b9abSLydia Wang 	/* SW0 (17h), AIW0(13h) */
3398054d867eSTakashi Iwai 	update_power_state(codec, 0x17, parm);
3399054d867eSTakashi Iwai 	update_power_state(codec, 0x13, parm);
34003e95b9abSLydia Wang 
34013e95b9abSLydia Wang 	parm = AC_PWRST_D3;
34023e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1e, &parm);
34033e95b9abSLydia Wang 	/* PW11 (22h) */
34043e95b9abSLydia Wang 	if (spec->dmic_enabled)
34053e95b9abSLydia Wang 		set_pin_power_state(codec, 0x22, &parm);
34063e95b9abSLydia Wang 	else
3407054d867eSTakashi Iwai 		update_power_state(codec, 0x22, AC_PWRST_D3);
34083e95b9abSLydia Wang 
34093e95b9abSLydia Wang 	/* SW2(26h), AIW1(14h) */
3410054d867eSTakashi Iwai 	update_power_state(codec, 0x26, parm);
3411054d867eSTakashi Iwai 	update_power_state(codec, 0x14, parm);
34123e95b9abSLydia Wang 
34133e95b9abSLydia Wang 	/* outputs */
34143e95b9abSLydia Wang 	/* PW0 (19h), SW1 (18h), AOW1 (11h) */
34153e95b9abSLydia Wang 	parm = AC_PWRST_D3;
34163e95b9abSLydia Wang 	set_pin_power_state(codec, 0x19, &parm);
34173e95b9abSLydia Wang 	/* Smart 5.1 PW2(1bh) */
34183e95b9abSLydia Wang 	if (spec->smart51_enabled)
34193e95b9abSLydia Wang 		set_pin_power_state(codec, 0x1b, &parm);
3420054d867eSTakashi Iwai 	update_power_state(codec, 0x18, parm);
3421054d867eSTakashi Iwai 	update_power_state(codec, 0x11, parm);
34223e95b9abSLydia Wang 
34233e95b9abSLydia Wang 	/* PW7 (23h), SW3 (27h), AOW3 (25h) */
34243e95b9abSLydia Wang 	parm = AC_PWRST_D3;
34253e95b9abSLydia Wang 	set_pin_power_state(codec, 0x23, &parm);
34263e95b9abSLydia Wang 	/* Smart 5.1 PW1(1ah) */
34273e95b9abSLydia Wang 	if (spec->smart51_enabled)
34283e95b9abSLydia Wang 		set_pin_power_state(codec, 0x1a, &parm);
3429054d867eSTakashi Iwai 	update_power_state(codec, 0x27, parm);
34303e95b9abSLydia Wang 
34313e95b9abSLydia Wang 	/* Smart 5.1 PW5(1eh) */
34323e95b9abSLydia Wang 	if (spec->smart51_enabled)
34333e95b9abSLydia Wang 		set_pin_power_state(codec, 0x1e, &parm);
3434054d867eSTakashi Iwai 	update_power_state(codec, 0x25, parm);
34353e95b9abSLydia Wang 
34363e95b9abSLydia Wang 	/* Mono out */
34373e95b9abSLydia Wang 	/* SW4(28h)->MW1(29h)-> PW12 (2ah)*/
34383e95b9abSLydia Wang 	present = snd_hda_jack_detect(codec, 0x1c);
34393e95b9abSLydia Wang 
34403e95b9abSLydia Wang 	if (present)
34413e95b9abSLydia Wang 		mono_out = 0;
34423e95b9abSLydia Wang 	else {
34433e95b9abSLydia Wang 		present = snd_hda_jack_detect(codec, 0x1d);
34443e95b9abSLydia Wang 		if (!spec->hp_independent_mode && present)
34453e95b9abSLydia Wang 			mono_out = 0;
34463e95b9abSLydia Wang 		else
34473e95b9abSLydia Wang 			mono_out = 1;
34483e95b9abSLydia Wang 	}
34493e95b9abSLydia Wang 	parm = mono_out ? AC_PWRST_D0 : AC_PWRST_D3;
3450054d867eSTakashi Iwai 	update_power_state(codec, 0x28, parm);
3451054d867eSTakashi Iwai 	update_power_state(codec, 0x29, parm);
3452054d867eSTakashi Iwai 	update_power_state(codec, 0x2a, parm);
34533e95b9abSLydia Wang 
34543e95b9abSLydia Wang 	/* PW 3/4 (1ch/1dh) */
34553e95b9abSLydia Wang 	parm = AC_PWRST_D3;
34563e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1c, &parm);
34573e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1d, &parm);
34583e95b9abSLydia Wang 	/* HP Independent Mode, power on AOW3 */
34593e95b9abSLydia Wang 	if (spec->hp_independent_mode)
3460054d867eSTakashi Iwai 		update_power_state(codec, 0x25, parm);
34613e95b9abSLydia Wang 
34623e95b9abSLydia Wang 	/* force to D0 for internal Speaker */
34633e95b9abSLydia Wang 	/* MW0 (16h), AOW0 (10h) */
3464054d867eSTakashi Iwai 	update_power_state(codec, 0x16, imux_is_smixer ? AC_PWRST_D0 : parm);
3465054d867eSTakashi Iwai 	update_power_state(codec, 0x10, mono_out ? AC_PWRST_D0 : parm);
34663e95b9abSLydia Wang }
34673e95b9abSLydia Wang 
3468f3db423dSLydia Wang static int patch_vt1716S(struct hda_codec *codec)
3469f3db423dSLydia Wang {
3470f3db423dSLydia Wang 	struct via_spec *spec;
3471f3db423dSLydia Wang 	int err;
3472f3db423dSLydia Wang 
3473f3db423dSLydia Wang 	/* create a codec specific record */
34745b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
3475f3db423dSLydia Wang 	if (spec == NULL)
3476f3db423dSLydia Wang 		return -ENOMEM;
3477f3db423dSLydia Wang 
3478620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x16;
3479d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x1a, 0, 3, 40);
3480d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x1e, 0, 3, 40);
3481620e2b28STakashi Iwai 
3482f3db423dSLydia Wang 	/* automatic parse from the BIOS config */
348312daef65STakashi Iwai 	err = via_parse_auto_config(codec);
3484f3db423dSLydia Wang 	if (err < 0) {
3485f3db423dSLydia Wang 		via_free(codec);
3486f3db423dSLydia Wang 		return err;
3487f3db423dSLydia Wang 	}
3488f3db423dSLydia Wang 
3489096a8854STakashi Iwai 	spec->init_verbs[spec->num_iverbs++]  = vt1716S_init_verbs;
3490f3db423dSLydia Wang 
3491f3db423dSLydia Wang 	spec->mixers[spec->num_mixers] = vt1716s_dmic_mixer;
3492f3db423dSLydia Wang 	spec->num_mixers++;
3493f3db423dSLydia Wang 
3494f3db423dSLydia Wang 	spec->mixers[spec->num_mixers++] = vt1716S_mono_out_mixer;
3495f3db423dSLydia Wang 
3496f3db423dSLydia Wang 	codec->patch_ops = via_patch_ops;
3497f3db423dSLydia Wang 
34983e95b9abSLydia Wang 	spec->set_widgets_power_state = set_widgets_power_state_vt1716S;
3499f3db423dSLydia Wang 	return 0;
3500f3db423dSLydia Wang }
350125eaba2fSLydia Wang 
350225eaba2fSLydia Wang /* for vt2002P */
350325eaba2fSLydia Wang 
3504096a8854STakashi Iwai static const struct hda_verb vt2002P_init_verbs[] = {
3505eadb9a80SLydia Wang 	/* Class-D speaker related verbs */
3506eadb9a80SLydia Wang 	{0x1, 0xfe0, 0x4},
3507eadb9a80SLydia Wang 	{0x1, 0xfe9, 0x80},
3508eadb9a80SLydia Wang 	{0x1, 0xfe2, 0x22},
350925eaba2fSLydia Wang 	/* Enable Boost Volume backdoor */
351025eaba2fSLydia Wang 	{0x1, 0xfb9, 0x24},
351125eaba2fSLydia Wang 	/* Enable AOW0 to MW9 */
351225eaba2fSLydia Wang 	{0x1, 0xfb8, 0x88},
351325eaba2fSLydia Wang 	{ }
351425eaba2fSLydia Wang };
35154a918ffeSTakashi Iwai 
3516096a8854STakashi Iwai static const struct hda_verb vt1802_init_verbs[] = {
351711890956SLydia Wang 	/* Enable Boost Volume backdoor */
351811890956SLydia Wang 	{0x1, 0xfb9, 0x24},
351911890956SLydia Wang 	/* Enable AOW0 to MW9 */
352011890956SLydia Wang 	{0x1, 0xfb8, 0x88},
352111890956SLydia Wang 	{ }
352211890956SLydia Wang };
352325eaba2fSLydia Wang 
35243e95b9abSLydia Wang static void set_widgets_power_state_vt2002P(struct hda_codec *codec)
35253e95b9abSLydia Wang {
35263e95b9abSLydia Wang 	struct via_spec *spec = codec->spec;
35273e95b9abSLydia Wang 	int imux_is_smixer;
35283e95b9abSLydia Wang 	unsigned int parm;
35293e95b9abSLydia Wang 	unsigned int present;
35303e95b9abSLydia Wang 	/* MUX9 (1eh) = stereo mixer */
35313e95b9abSLydia Wang 	imux_is_smixer =
35323e95b9abSLydia Wang 	snd_hda_codec_read(codec, 0x1e, 0, AC_VERB_GET_CONNECT_SEL, 0x00) == 3;
35333e95b9abSLydia Wang 	/* inputs */
35343e95b9abSLydia Wang 	/* PW 5/6/7 (29h/2ah/2bh) */
35353e95b9abSLydia Wang 	parm = AC_PWRST_D3;
35363e95b9abSLydia Wang 	set_pin_power_state(codec, 0x29, &parm);
35373e95b9abSLydia Wang 	set_pin_power_state(codec, 0x2a, &parm);
35383e95b9abSLydia Wang 	set_pin_power_state(codec, 0x2b, &parm);
35393e95b9abSLydia Wang 	parm = AC_PWRST_D0;
35403e95b9abSLydia Wang 	/* MUX9/10 (1eh/1fh), AIW 0/1 (10h/11h) */
3541054d867eSTakashi Iwai 	update_power_state(codec, 0x1e, parm);
3542054d867eSTakashi Iwai 	update_power_state(codec, 0x1f, parm);
3543054d867eSTakashi Iwai 	update_power_state(codec, 0x10, parm);
3544054d867eSTakashi Iwai 	update_power_state(codec, 0x11, parm);
35453e95b9abSLydia Wang 
35463e95b9abSLydia Wang 	/* outputs */
35473e95b9abSLydia Wang 	/* AOW0 (8h)*/
3548054d867eSTakashi Iwai 	update_power_state(codec, 0x8, parm);
35493e95b9abSLydia Wang 
355011890956SLydia Wang 	if (spec->codec_type == VT1802) {
355111890956SLydia Wang 		/* PW4 (28h), MW4 (18h), MUX4(38h) */
355211890956SLydia Wang 		parm = AC_PWRST_D3;
355311890956SLydia Wang 		set_pin_power_state(codec, 0x28, &parm);
3554054d867eSTakashi Iwai 		update_power_state(codec, 0x18, parm);
3555054d867eSTakashi Iwai 		update_power_state(codec, 0x38, parm);
355611890956SLydia Wang 	} else {
35573e95b9abSLydia Wang 		/* PW4 (26h), MW4 (1ch), MUX4(37h) */
35583e95b9abSLydia Wang 		parm = AC_PWRST_D3;
35593e95b9abSLydia Wang 		set_pin_power_state(codec, 0x26, &parm);
3560054d867eSTakashi Iwai 		update_power_state(codec, 0x1c, parm);
3561054d867eSTakashi Iwai 		update_power_state(codec, 0x37, parm);
356211890956SLydia Wang 	}
35633e95b9abSLydia Wang 
356411890956SLydia Wang 	if (spec->codec_type == VT1802) {
356511890956SLydia Wang 		/* PW1 (25h), MW1 (15h), MUX1(35h), AOW1 (9h) */
356611890956SLydia Wang 		parm = AC_PWRST_D3;
356711890956SLydia Wang 		set_pin_power_state(codec, 0x25, &parm);
3568054d867eSTakashi Iwai 		update_power_state(codec, 0x15, parm);
3569054d867eSTakashi Iwai 		update_power_state(codec, 0x35, parm);
357011890956SLydia Wang 	} else {
35713e95b9abSLydia Wang 		/* PW1 (25h), MW1 (19h), MUX1(35h), AOW1 (9h) */
35723e95b9abSLydia Wang 		parm = AC_PWRST_D3;
35733e95b9abSLydia Wang 		set_pin_power_state(codec, 0x25, &parm);
3574054d867eSTakashi Iwai 		update_power_state(codec, 0x19, parm);
3575054d867eSTakashi Iwai 		update_power_state(codec, 0x35, parm);
357611890956SLydia Wang 	}
35773e95b9abSLydia Wang 
35783e95b9abSLydia Wang 	if (spec->hp_independent_mode)
3579054d867eSTakashi Iwai 		update_power_state(codec, 0x9, AC_PWRST_D0);
35803e95b9abSLydia Wang 
35813e95b9abSLydia Wang 	/* Class-D */
35823e95b9abSLydia Wang 	/* PW0 (24h), MW0(18h/14h), MUX0(34h) */
35833e95b9abSLydia Wang 	present = snd_hda_jack_detect(codec, 0x25);
35843e95b9abSLydia Wang 
35853e95b9abSLydia Wang 	parm = AC_PWRST_D3;
35863e95b9abSLydia Wang 	set_pin_power_state(codec, 0x24, &parm);
35873e95b9abSLydia Wang 	parm = present ? AC_PWRST_D3 : AC_PWRST_D0;
358811890956SLydia Wang 	if (spec->codec_type == VT1802)
3589054d867eSTakashi Iwai 		update_power_state(codec, 0x14, parm);
359011890956SLydia Wang 	else
3591054d867eSTakashi Iwai 		update_power_state(codec, 0x18, parm);
3592054d867eSTakashi Iwai 	update_power_state(codec, 0x34, parm);
35933e95b9abSLydia Wang 
35943e95b9abSLydia Wang 	/* Mono Out */
35953e95b9abSLydia Wang 	present = snd_hda_jack_detect(codec, 0x26);
35963e95b9abSLydia Wang 
35973e95b9abSLydia Wang 	parm = present ? AC_PWRST_D3 : AC_PWRST_D0;
359811890956SLydia Wang 	if (spec->codec_type == VT1802) {
359911890956SLydia Wang 		/* PW15 (33h), MW8(1ch), MUX8(3ch) */
3600054d867eSTakashi Iwai 		update_power_state(codec, 0x33, parm);
3601054d867eSTakashi Iwai 		update_power_state(codec, 0x1c, parm);
3602054d867eSTakashi Iwai 		update_power_state(codec, 0x3c, parm);
360311890956SLydia Wang 	} else {
36043e95b9abSLydia Wang 		/* PW15 (31h), MW8(17h), MUX8(3bh) */
3605054d867eSTakashi Iwai 		update_power_state(codec, 0x31, parm);
3606054d867eSTakashi Iwai 		update_power_state(codec, 0x17, parm);
3607054d867eSTakashi Iwai 		update_power_state(codec, 0x3b, parm);
360811890956SLydia Wang 	}
36093e95b9abSLydia Wang 	/* MW9 (21h) */
36103e95b9abSLydia Wang 	if (imux_is_smixer || !is_aa_path_mute(codec))
3611054d867eSTakashi Iwai 		update_power_state(codec, 0x21, AC_PWRST_D0);
36123e95b9abSLydia Wang 	else
3613054d867eSTakashi Iwai 		update_power_state(codec, 0x21, AC_PWRST_D3);
36143e95b9abSLydia Wang }
361525eaba2fSLydia Wang 
36164b527b65SDavid Henningsson /*
36174b527b65SDavid Henningsson  * pin fix-up
36184b527b65SDavid Henningsson  */
36194b527b65SDavid Henningsson enum {
36204b527b65SDavid Henningsson 	VIA_FIXUP_INTMIC_BOOST,
3621d5266125STakashi Iwai 	VIA_FIXUP_ASUS_G75,
36224b527b65SDavid Henningsson };
36234b527b65SDavid Henningsson 
36244b527b65SDavid Henningsson static void via_fixup_intmic_boost(struct hda_codec *codec,
36254b527b65SDavid Henningsson 				  const struct hda_fixup *fix, int action)
36264b527b65SDavid Henningsson {
36274b527b65SDavid Henningsson 	if (action == HDA_FIXUP_ACT_PRE_PROBE)
36284b527b65SDavid Henningsson 		override_mic_boost(codec, 0x30, 0, 2, 40);
36294b527b65SDavid Henningsson }
36304b527b65SDavid Henningsson 
36314b527b65SDavid Henningsson static const struct hda_fixup via_fixups[] = {
36324b527b65SDavid Henningsson 	[VIA_FIXUP_INTMIC_BOOST] = {
36334b527b65SDavid Henningsson 		.type = HDA_FIXUP_FUNC,
36344b527b65SDavid Henningsson 		.v.func = via_fixup_intmic_boost,
36354b527b65SDavid Henningsson 	},
3636d5266125STakashi Iwai 	[VIA_FIXUP_ASUS_G75] = {
3637d5266125STakashi Iwai 		.type = HDA_FIXUP_PINS,
3638d5266125STakashi Iwai 		.v.pins = (const struct hda_pintbl[]) {
3639d5266125STakashi Iwai 			/* set 0x24 and 0x33 as speakers */
3640d5266125STakashi Iwai 			{ 0x24, 0x991301f0 },
3641d5266125STakashi Iwai 			{ 0x33, 0x991301f1 }, /* subwoofer */
3642d5266125STakashi Iwai 			{ }
3643d5266125STakashi Iwai 		}
3644d5266125STakashi Iwai 	},
36454b527b65SDavid Henningsson };
36464b527b65SDavid Henningsson 
36474b527b65SDavid Henningsson static const struct snd_pci_quirk vt2002p_fixups[] = {
3648d5266125STakashi Iwai 	SND_PCI_QUIRK(0x1043, 0x1487, "Asus G75", VIA_FIXUP_ASUS_G75),
36494b527b65SDavid Henningsson 	SND_PCI_QUIRK(0x1043, 0x8532, "Asus X202E", VIA_FIXUP_INTMIC_BOOST),
36504b527b65SDavid Henningsson 	{}
36514b527b65SDavid Henningsson };
36524b527b65SDavid Henningsson 
3653ef4da458STakashi Iwai /* NIDs 0x24 and 0x33 on VT1802 have connections to non-existing NID 0x3e
3654ef4da458STakashi Iwai  * Replace this with mixer NID 0x1c
3655ef4da458STakashi Iwai  */
3656ef4da458STakashi Iwai static void fix_vt1802_connections(struct hda_codec *codec)
3657ef4da458STakashi Iwai {
3658ef4da458STakashi Iwai 	static hda_nid_t conn_24[] = { 0x14, 0x1c };
3659ef4da458STakashi Iwai 	static hda_nid_t conn_33[] = { 0x1c };
3660ef4da458STakashi Iwai 
3661ef4da458STakashi Iwai 	snd_hda_override_conn_list(codec, 0x24, ARRAY_SIZE(conn_24), conn_24);
3662ef4da458STakashi Iwai 	snd_hda_override_conn_list(codec, 0x33, ARRAY_SIZE(conn_33), conn_33);
3663ef4da458STakashi Iwai }
3664ef4da458STakashi Iwai 
366525eaba2fSLydia Wang /* patch for vt2002P */
366625eaba2fSLydia Wang static int patch_vt2002P(struct hda_codec *codec)
366725eaba2fSLydia Wang {
366825eaba2fSLydia Wang 	struct via_spec *spec;
366925eaba2fSLydia Wang 	int err;
367025eaba2fSLydia Wang 
367125eaba2fSLydia Wang 	/* create a codec specific record */
36725b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
367325eaba2fSLydia Wang 	if (spec == NULL)
367425eaba2fSLydia Wang 		return -ENOMEM;
367525eaba2fSLydia Wang 
3676620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x21;
3677d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x2b, 0, 3, 40);
3678d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x29, 0, 3, 40);
3679ef4da458STakashi Iwai 	if (spec->codec_type == VT1802)
3680ef4da458STakashi Iwai 		fix_vt1802_connections(codec);
368130b45033STakashi Iwai 	add_secret_dac_path(codec);
3682620e2b28STakashi Iwai 
36834b527b65SDavid Henningsson 	snd_hda_pick_fixup(codec, NULL, vt2002p_fixups, via_fixups);
36844b527b65SDavid Henningsson 	snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
36854b527b65SDavid Henningsson 
368625eaba2fSLydia Wang 	/* automatic parse from the BIOS config */
368712daef65STakashi Iwai 	err = via_parse_auto_config(codec);
368825eaba2fSLydia Wang 	if (err < 0) {
368925eaba2fSLydia Wang 		via_free(codec);
369025eaba2fSLydia Wang 		return err;
369125eaba2fSLydia Wang 	}
369225eaba2fSLydia Wang 
369311890956SLydia Wang 	if (spec->codec_type == VT1802)
36944a918ffeSTakashi Iwai 		spec->init_verbs[spec->num_iverbs++] = vt1802_init_verbs;
369511890956SLydia Wang 	else
36964a918ffeSTakashi Iwai 		spec->init_verbs[spec->num_iverbs++] = vt2002P_init_verbs;
369711890956SLydia Wang 
369825eaba2fSLydia Wang 	codec->patch_ops = via_patch_ops;
369925eaba2fSLydia Wang 
37003e95b9abSLydia Wang 	spec->set_widgets_power_state =  set_widgets_power_state_vt2002P;
370125eaba2fSLydia Wang 	return 0;
370225eaba2fSLydia Wang }
3703ab6734e7SLydia Wang 
3704ab6734e7SLydia Wang /* for vt1812 */
3705ab6734e7SLydia Wang 
3706096a8854STakashi Iwai static const struct hda_verb vt1812_init_verbs[] = {
3707ab6734e7SLydia Wang 	/* Enable Boost Volume backdoor */
3708ab6734e7SLydia Wang 	{0x1, 0xfb9, 0x24},
3709ab6734e7SLydia Wang 	/* Enable AOW0 to MW9 */
3710ab6734e7SLydia Wang 	{0x1, 0xfb8, 0xa8},
3711ab6734e7SLydia Wang 	{ }
3712ab6734e7SLydia Wang };
3713ab6734e7SLydia Wang 
37143e95b9abSLydia Wang static void set_widgets_power_state_vt1812(struct hda_codec *codec)
37153e95b9abSLydia Wang {
37163e95b9abSLydia Wang 	struct via_spec *spec = codec->spec;
37173e95b9abSLydia Wang 	unsigned int parm;
37183e95b9abSLydia Wang 	unsigned int present;
37193e95b9abSLydia Wang 	/* inputs */
37203e95b9abSLydia Wang 	/* PW 5/6/7 (29h/2ah/2bh) */
37213e95b9abSLydia Wang 	parm = AC_PWRST_D3;
37223e95b9abSLydia Wang 	set_pin_power_state(codec, 0x29, &parm);
37233e95b9abSLydia Wang 	set_pin_power_state(codec, 0x2a, &parm);
37243e95b9abSLydia Wang 	set_pin_power_state(codec, 0x2b, &parm);
37253e95b9abSLydia Wang 	parm = AC_PWRST_D0;
37263e95b9abSLydia Wang 	/* MUX10/11 (1eh/1fh), AIW 0/1 (10h/11h) */
3727054d867eSTakashi Iwai 	update_power_state(codec, 0x1e, parm);
3728054d867eSTakashi Iwai 	update_power_state(codec, 0x1f, parm);
3729054d867eSTakashi Iwai 	update_power_state(codec, 0x10, parm);
3730054d867eSTakashi Iwai 	update_power_state(codec, 0x11, parm);
37313e95b9abSLydia Wang 
37323e95b9abSLydia Wang 	/* outputs */
37333e95b9abSLydia Wang 	/* AOW0 (8h)*/
3734054d867eSTakashi Iwai 	update_power_state(codec, 0x8, AC_PWRST_D0);
37353e95b9abSLydia Wang 
37363e95b9abSLydia Wang 	/* PW4 (28h), MW4 (18h), MUX4(38h) */
37373e95b9abSLydia Wang 	parm = AC_PWRST_D3;
37383e95b9abSLydia Wang 	set_pin_power_state(codec, 0x28, &parm);
3739054d867eSTakashi Iwai 	update_power_state(codec, 0x18, parm);
3740054d867eSTakashi Iwai 	update_power_state(codec, 0x38, parm);
37413e95b9abSLydia Wang 
37423e95b9abSLydia Wang 	/* PW1 (25h), MW1 (15h), MUX1(35h), AOW1 (9h) */
37433e95b9abSLydia Wang 	parm = AC_PWRST_D3;
37443e95b9abSLydia Wang 	set_pin_power_state(codec, 0x25, &parm);
3745054d867eSTakashi Iwai 	update_power_state(codec, 0x15, parm);
3746054d867eSTakashi Iwai 	update_power_state(codec, 0x35, parm);
37473e95b9abSLydia Wang 	if (spec->hp_independent_mode)
3748054d867eSTakashi Iwai 		update_power_state(codec, 0x9, AC_PWRST_D0);
37493e95b9abSLydia Wang 
37503e95b9abSLydia Wang 	/* Internal Speaker */
37513e95b9abSLydia Wang 	/* PW0 (24h), MW0(14h), MUX0(34h) */
37523e95b9abSLydia Wang 	present = snd_hda_jack_detect(codec, 0x25);
37533e95b9abSLydia Wang 
37543e95b9abSLydia Wang 	parm = AC_PWRST_D3;
37553e95b9abSLydia Wang 	set_pin_power_state(codec, 0x24, &parm);
37563e95b9abSLydia Wang 	if (present) {
3757054d867eSTakashi Iwai 		update_power_state(codec, 0x14, AC_PWRST_D3);
3758054d867eSTakashi Iwai 		update_power_state(codec, 0x34, AC_PWRST_D3);
37593e95b9abSLydia Wang 	} else {
3760054d867eSTakashi Iwai 		update_power_state(codec, 0x14, AC_PWRST_D0);
3761054d867eSTakashi Iwai 		update_power_state(codec, 0x34, AC_PWRST_D0);
37623e95b9abSLydia Wang 	}
37633e95b9abSLydia Wang 
37643e95b9abSLydia Wang 
37653e95b9abSLydia Wang 	/* Mono Out */
37663e95b9abSLydia Wang 	/* PW13 (31h), MW13(1ch), MUX13(3ch), MW14(3eh) */
37673e95b9abSLydia Wang 	present = snd_hda_jack_detect(codec, 0x28);
37683e95b9abSLydia Wang 
37693e95b9abSLydia Wang 	parm = AC_PWRST_D3;
37703e95b9abSLydia Wang 	set_pin_power_state(codec, 0x31, &parm);
37713e95b9abSLydia Wang 	if (present) {
3772054d867eSTakashi Iwai 		update_power_state(codec, 0x1c, AC_PWRST_D3);
3773054d867eSTakashi Iwai 		update_power_state(codec, 0x3c, AC_PWRST_D3);
3774054d867eSTakashi Iwai 		update_power_state(codec, 0x3e, AC_PWRST_D3);
37753e95b9abSLydia Wang 	} else {
3776054d867eSTakashi Iwai 		update_power_state(codec, 0x1c, AC_PWRST_D0);
3777054d867eSTakashi Iwai 		update_power_state(codec, 0x3c, AC_PWRST_D0);
3778054d867eSTakashi Iwai 		update_power_state(codec, 0x3e, AC_PWRST_D0);
37793e95b9abSLydia Wang 	}
37803e95b9abSLydia Wang 
37813e95b9abSLydia Wang 	/* PW15 (33h), MW15 (1dh), MUX15(3dh) */
37823e95b9abSLydia Wang 	parm = AC_PWRST_D3;
37833e95b9abSLydia Wang 	set_pin_power_state(codec, 0x33, &parm);
3784054d867eSTakashi Iwai 	update_power_state(codec, 0x1d, parm);
3785054d867eSTakashi Iwai 	update_power_state(codec, 0x3d, parm);
37863e95b9abSLydia Wang 
37873e95b9abSLydia Wang }
3788ab6734e7SLydia Wang 
3789ab6734e7SLydia Wang /* patch for vt1812 */
3790ab6734e7SLydia Wang static int patch_vt1812(struct hda_codec *codec)
3791ab6734e7SLydia Wang {
3792ab6734e7SLydia Wang 	struct via_spec *spec;
3793ab6734e7SLydia Wang 	int err;
3794ab6734e7SLydia Wang 
3795ab6734e7SLydia Wang 	/* create a codec specific record */
37965b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
3797ab6734e7SLydia Wang 	if (spec == NULL)
3798ab6734e7SLydia Wang 		return -ENOMEM;
3799ab6734e7SLydia Wang 
3800620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x21;
3801d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x2b, 0, 3, 40);
3802d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x29, 0, 3, 40);
380330b45033STakashi Iwai 	add_secret_dac_path(codec);
3804620e2b28STakashi Iwai 
3805ab6734e7SLydia Wang 	/* automatic parse from the BIOS config */
380612daef65STakashi Iwai 	err = via_parse_auto_config(codec);
3807ab6734e7SLydia Wang 	if (err < 0) {
3808ab6734e7SLydia Wang 		via_free(codec);
3809ab6734e7SLydia Wang 		return err;
3810ab6734e7SLydia Wang 	}
3811ab6734e7SLydia Wang 
3812096a8854STakashi Iwai 	spec->init_verbs[spec->num_iverbs++]  = vt1812_init_verbs;
3813ab6734e7SLydia Wang 
3814ab6734e7SLydia Wang 	codec->patch_ops = via_patch_ops;
3815ab6734e7SLydia Wang 
38163e95b9abSLydia Wang 	spec->set_widgets_power_state =  set_widgets_power_state_vt1812;
3817ab6734e7SLydia Wang 	return 0;
3818ab6734e7SLydia Wang }
3819ab6734e7SLydia Wang 
3820c577b8a1SJoseph Chan /*
3821c577b8a1SJoseph Chan  * patch entries
3822c577b8a1SJoseph Chan  */
382390dd48a1STakashi Iwai static const struct hda_codec_preset snd_hda_preset_via[] = {
38243218c178STakashi Iwai 	{ .id = 0x11061708, .name = "VT1708", .patch = patch_vt1708},
38253218c178STakashi Iwai 	{ .id = 0x11061709, .name = "VT1708", .patch = patch_vt1708},
38263218c178STakashi Iwai 	{ .id = 0x1106170a, .name = "VT1708", .patch = patch_vt1708},
38273218c178STakashi Iwai 	{ .id = 0x1106170b, .name = "VT1708", .patch = patch_vt1708},
38283218c178STakashi Iwai 	{ .id = 0x1106e710, .name = "VT1709 10-Ch",
3829ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
38303218c178STakashi Iwai 	{ .id = 0x1106e711, .name = "VT1709 10-Ch",
3831ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
38323218c178STakashi Iwai 	{ .id = 0x1106e712, .name = "VT1709 10-Ch",
3833ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
38343218c178STakashi Iwai 	{ .id = 0x1106e713, .name = "VT1709 10-Ch",
3835ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
38363218c178STakashi Iwai 	{ .id = 0x1106e714, .name = "VT1709 6-Ch",
3837ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
38383218c178STakashi Iwai 	{ .id = 0x1106e715, .name = "VT1709 6-Ch",
3839ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
38403218c178STakashi Iwai 	{ .id = 0x1106e716, .name = "VT1709 6-Ch",
3841ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
38423218c178STakashi Iwai 	{ .id = 0x1106e717, .name = "VT1709 6-Ch",
3843ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
38443218c178STakashi Iwai 	{ .id = 0x1106e720, .name = "VT1708B 8-Ch",
3845ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
38463218c178STakashi Iwai 	{ .id = 0x1106e721, .name = "VT1708B 8-Ch",
3847ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
38483218c178STakashi Iwai 	{ .id = 0x1106e722, .name = "VT1708B 8-Ch",
3849ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
38503218c178STakashi Iwai 	{ .id = 0x1106e723, .name = "VT1708B 8-Ch",
3851ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
38523218c178STakashi Iwai 	{ .id = 0x1106e724, .name = "VT1708B 4-Ch",
3853ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
38543218c178STakashi Iwai 	{ .id = 0x1106e725, .name = "VT1708B 4-Ch",
3855ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
38563218c178STakashi Iwai 	{ .id = 0x1106e726, .name = "VT1708B 4-Ch",
3857ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
38583218c178STakashi Iwai 	{ .id = 0x1106e727, .name = "VT1708B 4-Ch",
3859ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
38603218c178STakashi Iwai 	{ .id = 0x11060397, .name = "VT1708S",
3861d949cac1SHarald Welte 	  .patch = patch_vt1708S},
38623218c178STakashi Iwai 	{ .id = 0x11061397, .name = "VT1708S",
3863d949cac1SHarald Welte 	  .patch = patch_vt1708S},
38643218c178STakashi Iwai 	{ .id = 0x11062397, .name = "VT1708S",
3865d949cac1SHarald Welte 	  .patch = patch_vt1708S},
38663218c178STakashi Iwai 	{ .id = 0x11063397, .name = "VT1708S",
3867d949cac1SHarald Welte 	  .patch = patch_vt1708S},
3868bc92df7fSLydia Wang 	{ .id = 0x11064397, .name = "VT1705",
3869d949cac1SHarald Welte 	  .patch = patch_vt1708S},
38703218c178STakashi Iwai 	{ .id = 0x11065397, .name = "VT1708S",
3871d949cac1SHarald Welte 	  .patch = patch_vt1708S},
38723218c178STakashi Iwai 	{ .id = 0x11066397, .name = "VT1708S",
3873d949cac1SHarald Welte 	  .patch = patch_vt1708S},
38743218c178STakashi Iwai 	{ .id = 0x11067397, .name = "VT1708S",
3875d949cac1SHarald Welte 	  .patch = patch_vt1708S},
38763218c178STakashi Iwai 	{ .id = 0x11060398, .name = "VT1702",
3877d949cac1SHarald Welte 	  .patch = patch_vt1702},
38783218c178STakashi Iwai 	{ .id = 0x11061398, .name = "VT1702",
3879d949cac1SHarald Welte 	  .patch = patch_vt1702},
38803218c178STakashi Iwai 	{ .id = 0x11062398, .name = "VT1702",
3881d949cac1SHarald Welte 	  .patch = patch_vt1702},
38823218c178STakashi Iwai 	{ .id = 0x11063398, .name = "VT1702",
3883d949cac1SHarald Welte 	  .patch = patch_vt1702},
38843218c178STakashi Iwai 	{ .id = 0x11064398, .name = "VT1702",
3885d949cac1SHarald Welte 	  .patch = patch_vt1702},
38863218c178STakashi Iwai 	{ .id = 0x11065398, .name = "VT1702",
3887d949cac1SHarald Welte 	  .patch = patch_vt1702},
38883218c178STakashi Iwai 	{ .id = 0x11066398, .name = "VT1702",
3889d949cac1SHarald Welte 	  .patch = patch_vt1702},
38903218c178STakashi Iwai 	{ .id = 0x11067398, .name = "VT1702",
3891d949cac1SHarald Welte 	  .patch = patch_vt1702},
3892eb7188caSLydia Wang 	{ .id = 0x11060428, .name = "VT1718S",
3893eb7188caSLydia Wang 	  .patch = patch_vt1718S},
3894eb7188caSLydia Wang 	{ .id = 0x11064428, .name = "VT1718S",
3895eb7188caSLydia Wang 	  .patch = patch_vt1718S},
3896bb3c6bfcSLydia Wang 	{ .id = 0x11060441, .name = "VT2020",
3897bb3c6bfcSLydia Wang 	  .patch = patch_vt1718S},
3898bb3c6bfcSLydia Wang 	{ .id = 0x11064441, .name = "VT1828S",
3899bb3c6bfcSLydia Wang 	  .patch = patch_vt1718S},
3900f3db423dSLydia Wang 	{ .id = 0x11060433, .name = "VT1716S",
3901f3db423dSLydia Wang 	  .patch = patch_vt1716S},
3902f3db423dSLydia Wang 	{ .id = 0x1106a721, .name = "VT1716S",
3903f3db423dSLydia Wang 	  .patch = patch_vt1716S},
390425eaba2fSLydia Wang 	{ .id = 0x11060438, .name = "VT2002P", .patch = patch_vt2002P},
390525eaba2fSLydia Wang 	{ .id = 0x11064438, .name = "VT2002P", .patch = patch_vt2002P},
3906ab6734e7SLydia Wang 	{ .id = 0x11060448, .name = "VT1812", .patch = patch_vt1812},
390736dd5c4aSLydia Wang 	{ .id = 0x11060440, .name = "VT1818S",
390836dd5c4aSLydia Wang 	  .patch = patch_vt1708S},
390911890956SLydia Wang 	{ .id = 0x11060446, .name = "VT1802",
391011890956SLydia Wang 		.patch = patch_vt2002P},
391111890956SLydia Wang 	{ .id = 0x11068446, .name = "VT1802",
391211890956SLydia Wang 		.patch = patch_vt2002P},
3913c577b8a1SJoseph Chan 	{} /* terminator */
3914c577b8a1SJoseph Chan };
39151289e9e8STakashi Iwai 
39161289e9e8STakashi Iwai MODULE_ALIAS("snd-hda-codec-id:1106*");
39171289e9e8STakashi Iwai 
39181289e9e8STakashi Iwai static struct hda_codec_preset_list via_list = {
39191289e9e8STakashi Iwai 	.preset = snd_hda_preset_via,
39201289e9e8STakashi Iwai 	.owner = THIS_MODULE,
39211289e9e8STakashi Iwai };
39221289e9e8STakashi Iwai 
39231289e9e8STakashi Iwai MODULE_LICENSE("GPL");
39241289e9e8STakashi Iwai MODULE_DESCRIPTION("VIA HD-audio codec");
39251289e9e8STakashi Iwai 
39261289e9e8STakashi Iwai static int __init patch_via_init(void)
39271289e9e8STakashi Iwai {
39281289e9e8STakashi Iwai 	return snd_hda_add_codec_preset(&via_list);
39291289e9e8STakashi Iwai }
39301289e9e8STakashi Iwai 
39311289e9e8STakashi Iwai static void __exit patch_via_exit(void)
39321289e9e8STakashi Iwai {
39331289e9e8STakashi Iwai 	snd_hda_delete_codec_preset(&via_list);
39341289e9e8STakashi Iwai }
39351289e9e8STakashi Iwai 
39361289e9e8STakashi Iwai module_init(patch_via_init)
39371289e9e8STakashi Iwai module_exit(patch_via_exit)
3938