xref: /openbmc/linux/sound/pci/hda/patch_via.c (revision cf55e904516947597d75fd3844acc24891a95772)
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 {
1211f2e99feSLydia Wang 	/* codec parameterization */
12290dd48a1STakashi Iwai 	const struct snd_kcontrol_new *mixers[6];
1231f2e99feSLydia Wang 	unsigned int num_mixers;
1241f2e99feSLydia Wang 
12590dd48a1STakashi Iwai 	const struct hda_verb *init_verbs[5];
1261f2e99feSLydia Wang 	unsigned int num_iverbs;
1271f2e99feSLydia Wang 
12882673bc8STakashi Iwai 	char stream_name_analog[32];
1297eb56e84STakashi Iwai 	char stream_name_hp[32];
13090dd48a1STakashi Iwai 	const struct hda_pcm_stream *stream_analog_playback;
13190dd48a1STakashi Iwai 	const struct hda_pcm_stream *stream_analog_capture;
1321f2e99feSLydia Wang 
13382673bc8STakashi Iwai 	char stream_name_digital[32];
13490dd48a1STakashi Iwai 	const struct hda_pcm_stream *stream_digital_playback;
13590dd48a1STakashi Iwai 	const struct hda_pcm_stream *stream_digital_capture;
1361f2e99feSLydia Wang 
1371f2e99feSLydia Wang 	/* playback */
1381f2e99feSLydia Wang 	struct hda_multi_out multiout;
1391f2e99feSLydia Wang 	hda_nid_t slave_dig_outs[2];
140ece8d043STakashi Iwai 	hda_nid_t hp_dac_nid;
1413214b966STakashi Iwai 	hda_nid_t speaker_dac_nid;
1423214b966STakashi Iwai 	int hp_indep_shared;	/* indep HP-DAC is shared with side ch */
1433b607e3dSTakashi Iwai 	int opened_streams;	/* STREAM_* bits */
1443b607e3dSTakashi Iwai 	int active_streams;	/* STREAM_* bits */
1453214b966STakashi Iwai 	int aamix_mode;		/* loopback is enabled for output-path? */
1461f2e99feSLydia Wang 
1473214b966STakashi Iwai 	/* Output-paths:
1483214b966STakashi Iwai 	 * There are different output-paths depending on the setup.
1493214b966STakashi Iwai 	 * out_path, hp_path and speaker_path are primary paths.  If both
1503214b966STakashi Iwai 	 * direct DAC and aa-loopback routes are available, these contain
1513214b966STakashi Iwai 	 * the former paths.  Meanwhile *_mix_path contain the paths with
1523214b966STakashi Iwai 	 * loopback mixer.  (Since the loopback is only for front channel,
1533214b966STakashi Iwai 	 * no out_mix_path for surround channels.)
1543214b966STakashi Iwai 	 * The HP output has another path, hp_indep_path, which is used in
1553214b966STakashi Iwai 	 * the independent-HP mode.
1563214b966STakashi Iwai 	 */
157de6c74f3STakashi Iwai 	struct nid_path out_path[HDA_SIDE + 1];
1583214b966STakashi Iwai 	struct nid_path out_mix_path;
1594a79616dSTakashi Iwai 	struct nid_path hp_path;
1603214b966STakashi Iwai 	struct nid_path hp_mix_path;
1613214b966STakashi Iwai 	struct nid_path hp_indep_path;
1624a918ffeSTakashi Iwai 	struct nid_path speaker_path;
1633214b966STakashi Iwai 	struct nid_path speaker_mix_path;
1644a79616dSTakashi Iwai 
1651f2e99feSLydia Wang 	/* capture */
1661f2e99feSLydia Wang 	unsigned int num_adc_nids;
167de6c74f3STakashi Iwai 	hda_nid_t adc_nids[VIA_MAX_ADCS];
168de6c74f3STakashi Iwai 	hda_nid_t mux_nids[VIA_MAX_ADCS];
169620e2b28STakashi Iwai 	hda_nid_t aa_mix_nid;
1701f2e99feSLydia Wang 	hda_nid_t dig_in_nid;
1711f2e99feSLydia Wang 
1721f2e99feSLydia Wang 	/* capture source */
173a86a88eaSTakashi Iwai 	bool dyn_adc_switch;
174a86a88eaSTakashi Iwai 	int num_inputs;
175a86a88eaSTakashi Iwai 	struct via_input inputs[AUTO_CFG_MAX_INS + 1];
176de6c74f3STakashi Iwai 	unsigned int cur_mux[VIA_MAX_ADCS];
1771f2e99feSLydia Wang 
1783b607e3dSTakashi Iwai 	/* dynamic DAC switching */
1793b607e3dSTakashi Iwai 	unsigned int cur_dac_stream_tag;
1803b607e3dSTakashi Iwai 	unsigned int cur_dac_format;
1813b607e3dSTakashi Iwai 	unsigned int cur_hp_stream_tag;
1823b607e3dSTakashi Iwai 	unsigned int cur_hp_format;
1833b607e3dSTakashi Iwai 
184a86a88eaSTakashi Iwai 	/* dynamic ADC switching */
185a86a88eaSTakashi Iwai 	hda_nid_t cur_adc;
186a86a88eaSTakashi Iwai 	unsigned int cur_adc_stream_tag;
187a86a88eaSTakashi Iwai 	unsigned int cur_adc_format;
188a86a88eaSTakashi Iwai 
1891f2e99feSLydia Wang 	/* PCM information */
1901f2e99feSLydia Wang 	struct hda_pcm pcm_rec[3];
1911f2e99feSLydia Wang 
1921f2e99feSLydia Wang 	/* dynamic controls, init_verbs and input_mux */
1931f2e99feSLydia Wang 	struct auto_pin_cfg autocfg;
1941f2e99feSLydia Wang 	struct snd_array kctls;
1951f2e99feSLydia Wang 	hda_nid_t private_dac_nids[AUTO_CFG_MAX_OUTS];
1961f2e99feSLydia Wang 
1971f2e99feSLydia Wang 	/* HP mode source */
1981f2e99feSLydia Wang 	unsigned int hp_independent_mode;
199f3db423dSLydia Wang 	unsigned int dmic_enabled;
20024088a58STakashi Iwai 	unsigned int no_pin_power_ctl;
2011f2e99feSLydia Wang 	enum VIA_HDA_CODEC codec_type;
2021f2e99feSLydia Wang 
203e9d010c2STakashi Iwai 	/* analog low-power control */
204e9d010c2STakashi Iwai 	bool alc_mode;
205e9d010c2STakashi Iwai 
206e3d7a143STakashi Iwai 	/* smart51 setup */
207e3d7a143STakashi Iwai 	unsigned int smart51_nums;
208e3d7a143STakashi Iwai 	hda_nid_t smart51_pins[2];
209e3d7a143STakashi Iwai 	int smart51_idxs[2];
210e3d7a143STakashi Iwai 	const char *smart51_labels[2];
211e3d7a143STakashi Iwai 	unsigned int smart51_enabled;
212e3d7a143STakashi Iwai 
2131f2e99feSLydia Wang 	/* work to check hp jack state */
2141f2e99feSLydia Wang 	struct hda_codec *codec;
2151f2e99feSLydia Wang 	struct delayed_work vt1708_hp_work;
216187d333eSTakashi Iwai 	int hp_work_active;
217e06e5a29STakashi Iwai 	int vt1708_jack_detect;
2181f2e99feSLydia Wang 	int vt1708_hp_present;
2193e95b9abSLydia Wang 
2203e95b9abSLydia Wang 	void (*set_widgets_power_state)(struct hda_codec *codec);
2213e95b9abSLydia Wang 
2221f2e99feSLydia Wang 	struct hda_loopback_check loopback;
22313af8e77STakashi Iwai 	int num_loopbacks;
22413af8e77STakashi Iwai 	struct hda_amp_list loopback_list[8];
225a86a88eaSTakashi Iwai 
226a86a88eaSTakashi Iwai 	/* bind capture-volume */
227a86a88eaSTakashi Iwai 	struct hda_bind_ctls *bind_cap_vol;
228a86a88eaSTakashi Iwai 	struct hda_bind_ctls *bind_cap_sw;
2293b607e3dSTakashi Iwai 
2303b607e3dSTakashi Iwai 	struct mutex config_mutex;
2311f2e99feSLydia Wang };
2321f2e99feSLydia Wang 
2330341ccd7SLydia Wang static enum VIA_HDA_CODEC get_codec_type(struct hda_codec *codec);
2345b0cb1d8SJaroslav Kysela static struct via_spec * via_new_spec(struct hda_codec *codec)
2355b0cb1d8SJaroslav Kysela {
2365b0cb1d8SJaroslav Kysela 	struct via_spec *spec;
2375b0cb1d8SJaroslav Kysela 
2385b0cb1d8SJaroslav Kysela 	spec = kzalloc(sizeof(*spec), GFP_KERNEL);
2395b0cb1d8SJaroslav Kysela 	if (spec == NULL)
2405b0cb1d8SJaroslav Kysela 		return NULL;
2415b0cb1d8SJaroslav Kysela 
2423b607e3dSTakashi Iwai 	mutex_init(&spec->config_mutex);
2435b0cb1d8SJaroslav Kysela 	codec->spec = spec;
2445b0cb1d8SJaroslav Kysela 	spec->codec = codec;
2450341ccd7SLydia Wang 	spec->codec_type = get_codec_type(codec);
2460341ccd7SLydia Wang 	/* VT1708BCE & VT1708S are almost same */
2470341ccd7SLydia Wang 	if (spec->codec_type == VT1708BCE)
2480341ccd7SLydia Wang 		spec->codec_type = VT1708S;
2495b0cb1d8SJaroslav Kysela 	return spec;
2505b0cb1d8SJaroslav Kysela }
2515b0cb1d8SJaroslav Kysela 
252744ff5f4SLydia Wang static enum VIA_HDA_CODEC get_codec_type(struct hda_codec *codec)
253d7426329SHarald Welte {
254744ff5f4SLydia Wang 	u32 vendor_id = codec->vendor_id;
255d7426329SHarald Welte 	u16 ven_id = vendor_id >> 16;
256d7426329SHarald Welte 	u16 dev_id = vendor_id & 0xffff;
257d7426329SHarald Welte 	enum VIA_HDA_CODEC codec_type;
258d7426329SHarald Welte 
259d7426329SHarald Welte 	/* get codec type */
260d7426329SHarald Welte 	if (ven_id != 0x1106)
261d7426329SHarald Welte 		codec_type = UNKNOWN;
262d7426329SHarald Welte 	else if (dev_id >= 0x1708 && dev_id <= 0x170b)
263d7426329SHarald Welte 		codec_type = VT1708;
264d7426329SHarald Welte 	else if (dev_id >= 0xe710 && dev_id <= 0xe713)
265d7426329SHarald Welte 		codec_type = VT1709_10CH;
266d7426329SHarald Welte 	else if (dev_id >= 0xe714 && dev_id <= 0xe717)
267d7426329SHarald Welte 		codec_type = VT1709_6CH;
268518bf3baSLydia Wang 	else if (dev_id >= 0xe720 && dev_id <= 0xe723) {
269d7426329SHarald Welte 		codec_type = VT1708B_8CH;
270518bf3baSLydia Wang 		if (snd_hda_param_read(codec, 0x16, AC_PAR_CONNLIST_LEN) == 0x7)
271518bf3baSLydia Wang 			codec_type = VT1708BCE;
272518bf3baSLydia Wang 	} else if (dev_id >= 0xe724 && dev_id <= 0xe727)
273d7426329SHarald Welte 		codec_type = VT1708B_4CH;
274d7426329SHarald Welte 	else if ((dev_id & 0xfff) == 0x397
275d7426329SHarald Welte 		 && (dev_id >> 12) < 8)
276d7426329SHarald Welte 		codec_type = VT1708S;
277d7426329SHarald Welte 	else if ((dev_id & 0xfff) == 0x398
278d7426329SHarald Welte 		 && (dev_id >> 12) < 8)
279d7426329SHarald Welte 		codec_type = VT1702;
280eb7188caSLydia Wang 	else if ((dev_id & 0xfff) == 0x428
281eb7188caSLydia Wang 		 && (dev_id >> 12) < 8)
282eb7188caSLydia Wang 		codec_type = VT1718S;
283f3db423dSLydia Wang 	else if (dev_id == 0x0433 || dev_id == 0xa721)
284f3db423dSLydia Wang 		codec_type = VT1716S;
285bb3c6bfcSLydia Wang 	else if (dev_id == 0x0441 || dev_id == 0x4441)
286bb3c6bfcSLydia Wang 		codec_type = VT1718S;
28725eaba2fSLydia Wang 	else if (dev_id == 0x0438 || dev_id == 0x4438)
28825eaba2fSLydia Wang 		codec_type = VT2002P;
289ab6734e7SLydia Wang 	else if (dev_id == 0x0448)
290ab6734e7SLydia Wang 		codec_type = VT1812;
29136dd5c4aSLydia Wang 	else if (dev_id == 0x0440)
29236dd5c4aSLydia Wang 		codec_type = VT1708S;
29311890956SLydia Wang 	else if ((dev_id & 0xfff) == 0x446)
29411890956SLydia Wang 		codec_type = VT1802;
295d7426329SHarald Welte 	else
296d7426329SHarald Welte 		codec_type = UNKNOWN;
297d7426329SHarald Welte 	return codec_type;
298d7426329SHarald Welte };
299d7426329SHarald Welte 
300ec7e7e42SLydia Wang #define VIA_JACK_EVENT		0x20
30169e52a80SHarald Welte #define VIA_HP_EVENT		0x01
30269e52a80SHarald Welte #define VIA_GPIO_EVENT		0x02
3034a918ffeSTakashi Iwai #define VIA_LINE_EVENT		0x03
30469e52a80SHarald Welte 
305c577b8a1SJoseph Chan enum {
306c577b8a1SJoseph Chan 	VIA_CTL_WIDGET_VOL,
307c577b8a1SJoseph Chan 	VIA_CTL_WIDGET_MUTE,
308f5271101SLydia Wang 	VIA_CTL_WIDGET_ANALOG_MUTE,
309c577b8a1SJoseph Chan };
310c577b8a1SJoseph Chan 
311ada509ecSTakashi Iwai static void analog_low_current_mode(struct hda_codec *codec);
312ada509ecSTakashi Iwai static bool is_aa_path_mute(struct hda_codec *codec);
3131f2e99feSLydia Wang 
314187d333eSTakashi Iwai #define hp_detect_with_aa(codec) \
315187d333eSTakashi Iwai 	(snd_hda_get_bool_hint(codec, "analog_loopback_hp_detect") == 1 && \
316187d333eSTakashi Iwai 	 !is_aa_path_mute(codec))
3171f2e99feSLydia Wang 
3181f2e99feSLydia Wang static void vt1708_stop_hp_work(struct via_spec *spec)
3191f2e99feSLydia Wang {
3201f2e99feSLydia Wang 	if (spec->codec_type != VT1708 || spec->autocfg.hp_pins[0] == 0)
3211f2e99feSLydia Wang 		return;
322187d333eSTakashi Iwai 	if (spec->hp_work_active) {
323187d333eSTakashi Iwai 		snd_hda_codec_write(spec->codec, 0x1, 0, 0xf81, 1);
3245b84ba26STejun Heo 		cancel_delayed_work_sync(&spec->vt1708_hp_work);
325187d333eSTakashi Iwai 		spec->hp_work_active = 0;
326187d333eSTakashi Iwai 	}
327187d333eSTakashi Iwai }
328187d333eSTakashi Iwai 
329187d333eSTakashi Iwai static void vt1708_update_hp_work(struct via_spec *spec)
330187d333eSTakashi Iwai {
331187d333eSTakashi Iwai 	if (spec->codec_type != VT1708 || spec->autocfg.hp_pins[0] == 0)
332187d333eSTakashi Iwai 		return;
333187d333eSTakashi Iwai 	if (spec->vt1708_jack_detect &&
334187d333eSTakashi Iwai 	    (spec->active_streams || hp_detect_with_aa(spec->codec))) {
335187d333eSTakashi Iwai 		if (!spec->hp_work_active) {
336187d333eSTakashi Iwai 			snd_hda_codec_write(spec->codec, 0x1, 0, 0xf81, 0);
337187d333eSTakashi Iwai 			schedule_delayed_work(&spec->vt1708_hp_work,
338187d333eSTakashi Iwai 					      msecs_to_jiffies(100));
339187d333eSTakashi Iwai 			spec->hp_work_active = 1;
340187d333eSTakashi Iwai 		}
341187d333eSTakashi Iwai 	} else if (!hp_detect_with_aa(spec->codec))
342187d333eSTakashi Iwai 		vt1708_stop_hp_work(spec);
3431f2e99feSLydia Wang }
344f5271101SLydia Wang 
3453e95b9abSLydia Wang static void set_widgets_power_state(struct hda_codec *codec)
3463e95b9abSLydia Wang {
3473e95b9abSLydia Wang 	struct via_spec *spec = codec->spec;
3483e95b9abSLydia Wang 	if (spec->set_widgets_power_state)
3493e95b9abSLydia Wang 		spec->set_widgets_power_state(codec);
3503e95b9abSLydia Wang }
35125eaba2fSLydia Wang 
352f5271101SLydia Wang static int analog_input_switch_put(struct snd_kcontrol *kcontrol,
353f5271101SLydia Wang 				   struct snd_ctl_elem_value *ucontrol)
354f5271101SLydia Wang {
355f5271101SLydia Wang 	int change = snd_hda_mixer_amp_switch_put(kcontrol, ucontrol);
356f5271101SLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
357f5271101SLydia Wang 
3583e95b9abSLydia Wang 	set_widgets_power_state(codec);
359ada509ecSTakashi Iwai 	analog_low_current_mode(snd_kcontrol_chip(kcontrol));
360187d333eSTakashi Iwai 	vt1708_update_hp_work(codec->spec);
361f5271101SLydia Wang 	return change;
362f5271101SLydia Wang }
363f5271101SLydia Wang 
364f5271101SLydia Wang /* modify .put = snd_hda_mixer_amp_switch_put */
365f5271101SLydia Wang #define ANALOG_INPUT_MUTE						\
366f5271101SLydia Wang 	{		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,		\
367f5271101SLydia Wang 			.name = NULL,					\
368f5271101SLydia Wang 			.index = 0,					\
369f5271101SLydia Wang 			.info = snd_hda_mixer_amp_switch_info,		\
370f5271101SLydia Wang 			.get = snd_hda_mixer_amp_switch_get,		\
371f5271101SLydia Wang 			.put = analog_input_switch_put,			\
372f5271101SLydia Wang 			.private_value = HDA_COMPOSE_AMP_VAL(0, 3, 0, 0) }
373f5271101SLydia Wang 
37490dd48a1STakashi Iwai static const struct snd_kcontrol_new via_control_templates[] = {
375c577b8a1SJoseph Chan 	HDA_CODEC_VOLUME(NULL, 0, 0, 0),
376c577b8a1SJoseph Chan 	HDA_CODEC_MUTE(NULL, 0, 0, 0),
377f5271101SLydia Wang 	ANALOG_INPUT_MUTE,
378c577b8a1SJoseph Chan };
379c577b8a1SJoseph Chan 
380ab6734e7SLydia Wang 
381c577b8a1SJoseph Chan /* add dynamic controls */
382291c9e33STakashi Iwai static struct snd_kcontrol_new *__via_clone_ctl(struct via_spec *spec,
383291c9e33STakashi Iwai 				const struct snd_kcontrol_new *tmpl,
384291c9e33STakashi Iwai 				const char *name)
385c577b8a1SJoseph Chan {
386c577b8a1SJoseph Chan 	struct snd_kcontrol_new *knew;
387c577b8a1SJoseph Chan 
388603c4019STakashi Iwai 	snd_array_init(&spec->kctls, sizeof(*knew), 32);
389603c4019STakashi Iwai 	knew = snd_array_new(&spec->kctls);
390c577b8a1SJoseph Chan 	if (!knew)
391291c9e33STakashi Iwai 		return NULL;
392291c9e33STakashi Iwai 	*knew = *tmpl;
393291c9e33STakashi Iwai 	if (!name)
394291c9e33STakashi Iwai 		name = tmpl->name;
395291c9e33STakashi Iwai 	if (name) {
396c577b8a1SJoseph Chan 		knew->name = kstrdup(name, GFP_KERNEL);
397c577b8a1SJoseph Chan 		if (!knew->name)
398291c9e33STakashi Iwai 			return NULL;
399291c9e33STakashi Iwai 	}
400291c9e33STakashi Iwai 	return knew;
401291c9e33STakashi Iwai }
402291c9e33STakashi Iwai 
403291c9e33STakashi Iwai static int __via_add_control(struct via_spec *spec, int type, const char *name,
404291c9e33STakashi Iwai 			     int idx, unsigned long val)
405291c9e33STakashi Iwai {
406291c9e33STakashi Iwai 	struct snd_kcontrol_new *knew;
407291c9e33STakashi Iwai 
408291c9e33STakashi Iwai 	knew = __via_clone_ctl(spec, &via_control_templates[type], name);
409291c9e33STakashi Iwai 	if (!knew)
410c577b8a1SJoseph Chan 		return -ENOMEM;
411d7a99cceSTakashi Iwai 	knew->index = idx;
4124d02d1b6SJaroslav Kysela 	if (get_amp_nid_(val))
4135e26dfd0SJaroslav Kysela 		knew->subdevice = HDA_SUBDEV_AMP_FLAG;
414c577b8a1SJoseph Chan 	knew->private_value = val;
415c577b8a1SJoseph Chan 	return 0;
416c577b8a1SJoseph Chan }
417c577b8a1SJoseph Chan 
4187b315bb4STakashi Iwai #define via_add_control(spec, type, name, val) \
4197b315bb4STakashi Iwai 	__via_add_control(spec, type, name, 0, val)
4207b315bb4STakashi Iwai 
421291c9e33STakashi Iwai #define via_clone_control(spec, tmpl) __via_clone_ctl(spec, tmpl, NULL)
4225b0cb1d8SJaroslav Kysela 
423603c4019STakashi Iwai static void via_free_kctls(struct hda_codec *codec)
424603c4019STakashi Iwai {
425603c4019STakashi Iwai 	struct via_spec *spec = codec->spec;
426603c4019STakashi Iwai 
427603c4019STakashi Iwai 	if (spec->kctls.list) {
428603c4019STakashi Iwai 		struct snd_kcontrol_new *kctl = spec->kctls.list;
429603c4019STakashi Iwai 		int i;
430603c4019STakashi Iwai 		for (i = 0; i < spec->kctls.used; i++)
431603c4019STakashi Iwai 			kfree(kctl[i].name);
432603c4019STakashi Iwai 	}
433603c4019STakashi Iwai 	snd_array_free(&spec->kctls);
434603c4019STakashi Iwai }
435603c4019STakashi Iwai 
436c577b8a1SJoseph Chan /* create input playback/capture controls for the given pin */
4379510e8ddSLydia Wang static int via_new_analog_input(struct via_spec *spec, const char *ctlname,
4387b315bb4STakashi Iwai 				int type_idx, int idx, int mix_nid)
439c577b8a1SJoseph Chan {
440c577b8a1SJoseph Chan 	char name[32];
441c577b8a1SJoseph Chan 	int err;
442c577b8a1SJoseph Chan 
443c577b8a1SJoseph Chan 	sprintf(name, "%s Playback Volume", ctlname);
4447b315bb4STakashi Iwai 	err = __via_add_control(spec, VIA_CTL_WIDGET_VOL, name, type_idx,
445c577b8a1SJoseph Chan 			      HDA_COMPOSE_AMP_VAL(mix_nid, 3, idx, HDA_INPUT));
446c577b8a1SJoseph Chan 	if (err < 0)
447c577b8a1SJoseph Chan 		return err;
448c577b8a1SJoseph Chan 	sprintf(name, "%s Playback Switch", ctlname);
4497b315bb4STakashi Iwai 	err = __via_add_control(spec, VIA_CTL_WIDGET_ANALOG_MUTE, name, type_idx,
450c577b8a1SJoseph Chan 			      HDA_COMPOSE_AMP_VAL(mix_nid, 3, idx, HDA_INPUT));
451c577b8a1SJoseph Chan 	if (err < 0)
452c577b8a1SJoseph Chan 		return err;
453c577b8a1SJoseph Chan 	return 0;
454c577b8a1SJoseph Chan }
455c577b8a1SJoseph Chan 
4565d41762aSTakashi Iwai #define get_connection_index(codec, mux, nid) \
4578d087c76STakashi Iwai 	snd_hda_get_conn_index(codec, mux, nid, 0)
4585d41762aSTakashi Iwai 
4598df2a312STakashi Iwai static bool check_amp_caps(struct hda_codec *codec, hda_nid_t nid, int dir,
4608df2a312STakashi Iwai 			   unsigned int mask)
4618df2a312STakashi Iwai {
462a934d5a9STakashi Iwai 	unsigned int caps;
463a934d5a9STakashi Iwai 	if (!nid)
464a934d5a9STakashi Iwai 		return false;
465a934d5a9STakashi Iwai 	caps = get_wcaps(codec, nid);
4668df2a312STakashi Iwai 	if (dir == HDA_INPUT)
4678df2a312STakashi Iwai 		caps &= AC_WCAP_IN_AMP;
4688df2a312STakashi Iwai 	else
4698df2a312STakashi Iwai 		caps &= AC_WCAP_OUT_AMP;
4708df2a312STakashi Iwai 	if (!caps)
4718df2a312STakashi Iwai 		return false;
4728df2a312STakashi Iwai 	if (query_amp_caps(codec, nid, dir) & mask)
4738df2a312STakashi Iwai 		return true;
4748df2a312STakashi Iwai 	return false;
4758df2a312STakashi Iwai }
4768df2a312STakashi Iwai 
47709a9ad69STakashi Iwai #define have_mute(codec, nid, dir) \
47809a9ad69STakashi Iwai 	check_amp_caps(codec, nid, dir, AC_AMPCAP_MUTE)
4798df2a312STakashi Iwai 
480d69607b3SLydia Wang /* enable/disable the output-route mixers */
481d69607b3SLydia Wang static void activate_output_mix(struct hda_codec *codec, struct nid_path *path,
4823214b966STakashi Iwai 				hda_nid_t mix_nid, int idx, bool enable)
483d69607b3SLydia Wang {
484d69607b3SLydia Wang 	int i, num, val;
485d69607b3SLydia Wang 
486d69607b3SLydia Wang 	if (!path)
487d69607b3SLydia Wang 		return;
48809cf03b8STakashi Iwai 	num = snd_hda_get_num_conns(codec, mix_nid);
489d69607b3SLydia Wang 	for (i = 0; i < num; i++) {
4903214b966STakashi Iwai 		if (i == idx)
491d69607b3SLydia Wang 			val = AMP_IN_UNMUTE(i);
492d69607b3SLydia Wang 		else
493d69607b3SLydia Wang 			val = AMP_IN_MUTE(i);
494d69607b3SLydia Wang 		snd_hda_codec_write(codec, mix_nid, 0,
495d69607b3SLydia Wang 				    AC_VERB_SET_AMP_GAIN_MUTE, val);
496d69607b3SLydia Wang 	}
497d69607b3SLydia Wang }
498d69607b3SLydia Wang 
49909a9ad69STakashi Iwai /* enable/disable the output-route */
50009a9ad69STakashi Iwai static void activate_output_path(struct hda_codec *codec, struct nid_path *path,
50109a9ad69STakashi Iwai 				 bool enable, bool force)
5025d41762aSTakashi Iwai {
503d69607b3SLydia Wang 	struct via_spec *spec = codec->spec;
5043214b966STakashi Iwai 	int i;
50509a9ad69STakashi Iwai 	for (i = 0; i < path->depth; i++) {
50609a9ad69STakashi Iwai 		hda_nid_t src, dst;
50709a9ad69STakashi Iwai 		int idx = path->idx[i];
50809a9ad69STakashi Iwai 		src = path->path[i];
50909a9ad69STakashi Iwai 		if (i < path->depth - 1)
51009a9ad69STakashi Iwai 			dst = path->path[i + 1];
51109a9ad69STakashi Iwai 		else
51209a9ad69STakashi Iwai 			dst = 0;
51309a9ad69STakashi Iwai 		if (enable && path->multi[i])
51409a9ad69STakashi Iwai 			snd_hda_codec_write(codec, dst, 0,
5155d41762aSTakashi Iwai 					    AC_VERB_SET_CONNECT_SEL, idx);
5163214b966STakashi Iwai 		if (!force && (dst == spec->aa_mix_nid))
517e5e14681SLydia Wang 			continue;
5183214b966STakashi Iwai 		if (have_mute(codec, dst, HDA_INPUT))
5193214b966STakashi Iwai 			activate_output_mix(codec, path, dst, idx, enable);
52009a9ad69STakashi Iwai 		if (!force && (src == path->vol_ctl || src == path->mute_ctl))
52109a9ad69STakashi Iwai 			continue;
52209a9ad69STakashi Iwai 		if (have_mute(codec, src, HDA_OUTPUT)) {
52309a9ad69STakashi Iwai 			int val = enable ? AMP_OUT_UNMUTE : AMP_OUT_MUTE;
52409a9ad69STakashi Iwai 			snd_hda_codec_write(codec, src, 0,
52509a9ad69STakashi Iwai 					    AC_VERB_SET_AMP_GAIN_MUTE, val);
52609a9ad69STakashi Iwai 		}
52709a9ad69STakashi Iwai 	}
5285d41762aSTakashi Iwai }
5295d41762aSTakashi Iwai 
5305d41762aSTakashi Iwai /* set the given pin as output */
5315d41762aSTakashi Iwai static void init_output_pin(struct hda_codec *codec, hda_nid_t pin,
5325d41762aSTakashi Iwai 			    int pin_type)
5335d41762aSTakashi Iwai {
5345d41762aSTakashi Iwai 	if (!pin)
5355d41762aSTakashi Iwai 		return;
536cdd03cedSTakashi Iwai 	snd_hda_set_pin_ctl(codec, pin, pin_type);
5375d41762aSTakashi Iwai 	if (snd_hda_query_pin_caps(codec, pin) & AC_PINCAP_EAPD)
5385d41762aSTakashi Iwai 		snd_hda_codec_write(codec, pin, 0,
539d3a11e60STakashi Iwai 				    AC_VERB_SET_EAPD_BTLENABLE, 0x02);
540c577b8a1SJoseph Chan }
541c577b8a1SJoseph Chan 
54209a9ad69STakashi Iwai static void via_auto_init_output(struct hda_codec *codec,
543a353fbb1STakashi Iwai 				 struct nid_path *path, int pin_type)
5445d41762aSTakashi Iwai {
5455d41762aSTakashi Iwai 	unsigned int caps;
546d69607b3SLydia Wang 	hda_nid_t pin;
5475d41762aSTakashi Iwai 
54809a9ad69STakashi Iwai 	if (!path->depth)
5495d41762aSTakashi Iwai 		return;
55009a9ad69STakashi Iwai 	pin = path->path[path->depth - 1];
5515d41762aSTakashi Iwai 
5525d41762aSTakashi Iwai 	init_output_pin(codec, pin, pin_type);
55377e314f7STakashi Iwai 	if (get_wcaps(codec, pin) & AC_WCAP_OUT_AMP)
5545d41762aSTakashi Iwai 		caps = query_amp_caps(codec, pin, HDA_OUTPUT);
55577e314f7STakashi Iwai 	else
55677e314f7STakashi Iwai 		caps = 0;
5575d41762aSTakashi Iwai 	if (caps & AC_AMPCAP_MUTE) {
5585d41762aSTakashi Iwai 		unsigned int val;
5595d41762aSTakashi Iwai 		val = (caps & AC_AMPCAP_OFFSET) >> AC_AMPCAP_OFFSET_SHIFT;
5605d41762aSTakashi Iwai 		snd_hda_codec_write(codec, pin, 0, AC_VERB_SET_AMP_GAIN_MUTE,
5615d41762aSTakashi Iwai 				    AMP_OUT_MUTE | val);
5625d41762aSTakashi Iwai 	}
563a353fbb1STakashi Iwai 	activate_output_path(codec, path, true, true); /* force on */
56409a9ad69STakashi Iwai }
565c577b8a1SJoseph Chan 
566c577b8a1SJoseph Chan static void via_auto_init_multi_out(struct hda_codec *codec)
567c577b8a1SJoseph Chan {
568c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
5693214b966STakashi Iwai 	struct nid_path *path;
570c577b8a1SJoseph Chan 	int i;
571c577b8a1SJoseph Chan 
5723214b966STakashi Iwai 	for (i = 0; i < spec->autocfg.line_outs + spec->smart51_nums; i++) {
5733214b966STakashi Iwai 		path = &spec->out_path[i];
5743214b966STakashi Iwai 		if (!i && spec->aamix_mode && spec->out_mix_path.depth)
5753214b966STakashi Iwai 			path = &spec->out_mix_path;
576a353fbb1STakashi Iwai 		via_auto_init_output(codec, path, PIN_OUT);
5773214b966STakashi Iwai 	}
578c577b8a1SJoseph Chan }
579c577b8a1SJoseph Chan 
580020066d1STakashi Iwai /* deactivate the inactive headphone-paths */
581020066d1STakashi Iwai static void deactivate_hp_paths(struct hda_codec *codec)
582c577b8a1SJoseph Chan {
583c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
5843214b966STakashi Iwai 	int shared = spec->hp_indep_shared;
585c577b8a1SJoseph Chan 
58609a9ad69STakashi Iwai 	if (spec->hp_independent_mode) {
58709a9ad69STakashi Iwai 		activate_output_path(codec, &spec->hp_path, false, false);
5883214b966STakashi Iwai 		activate_output_path(codec, &spec->hp_mix_path, false, false);
5893214b966STakashi Iwai 		if (shared)
5903214b966STakashi Iwai 			activate_output_path(codec, &spec->out_path[shared],
5913214b966STakashi Iwai 					     false, false);
592020066d1STakashi Iwai 	} else if (spec->aamix_mode || !spec->hp_path.depth) {
593020066d1STakashi Iwai 		activate_output_path(codec, &spec->hp_indep_path, false, false);
5943214b966STakashi Iwai 		activate_output_path(codec, &spec->hp_path, false, false);
5953214b966STakashi Iwai 	} else {
596020066d1STakashi Iwai 		activate_output_path(codec, &spec->hp_indep_path, false, false);
5973214b966STakashi Iwai 		activate_output_path(codec, &spec->hp_mix_path, false, false);
59809a9ad69STakashi Iwai 	}
59925eaba2fSLydia Wang }
600c577b8a1SJoseph Chan 
601020066d1STakashi Iwai static void via_auto_init_hp_out(struct hda_codec *codec)
602020066d1STakashi Iwai {
603020066d1STakashi Iwai 	struct via_spec *spec = codec->spec;
604020066d1STakashi Iwai 
605020066d1STakashi Iwai 	if (!spec->hp_path.depth) {
606a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->hp_mix_path, PIN_HP);
607020066d1STakashi Iwai 		return;
608020066d1STakashi Iwai 	}
609020066d1STakashi Iwai 	deactivate_hp_paths(codec);
610020066d1STakashi Iwai 	if (spec->hp_independent_mode)
611a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->hp_indep_path, PIN_HP);
612020066d1STakashi Iwai 	else if (spec->aamix_mode)
613a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->hp_mix_path, PIN_HP);
614020066d1STakashi Iwai 	else
615a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->hp_path, PIN_HP);
616020066d1STakashi Iwai }
617020066d1STakashi Iwai 
6184a918ffeSTakashi Iwai static void via_auto_init_speaker_out(struct hda_codec *codec)
6194a918ffeSTakashi Iwai {
6204a918ffeSTakashi Iwai 	struct via_spec *spec = codec->spec;
6214a918ffeSTakashi Iwai 
6223214b966STakashi Iwai 	if (!spec->autocfg.speaker_outs)
6233214b966STakashi Iwai 		return;
6243214b966STakashi Iwai 	if (!spec->speaker_path.depth) {
625a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->speaker_mix_path, PIN_OUT);
6263214b966STakashi Iwai 		return;
6273214b966STakashi Iwai 	}
6283214b966STakashi Iwai 	if (!spec->aamix_mode) {
6293214b966STakashi Iwai 		activate_output_path(codec, &spec->speaker_mix_path,
6303214b966STakashi Iwai 				     false, false);
631a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->speaker_path, PIN_OUT);
6323214b966STakashi Iwai 	} else {
6333214b966STakashi Iwai 		activate_output_path(codec, &spec->speaker_path, false, false);
634a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->speaker_mix_path, PIN_OUT);
6353214b966STakashi Iwai 	}
6364a918ffeSTakashi Iwai }
6374a918ffeSTakashi Iwai 
638f4a7828bSTakashi Iwai static bool is_smart51_pins(struct hda_codec *codec, hda_nid_t pin);
6396e969d91STakashi Iwai static void via_hp_automute(struct hda_codec *codec);
64032e0191dSClemens Ladisch 
641c577b8a1SJoseph Chan static void via_auto_init_analog_input(struct hda_codec *codec)
642c577b8a1SJoseph Chan {
643c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
6447b315bb4STakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
645096a8854STakashi Iwai 	hda_nid_t conn[HDA_MAX_CONNECTIONS];
64632e0191dSClemens Ladisch 	unsigned int ctl;
647096a8854STakashi Iwai 	int i, num_conns;
648c577b8a1SJoseph Chan 
649096a8854STakashi Iwai 	/* init ADCs */
650096a8854STakashi Iwai 	for (i = 0; i < spec->num_adc_nids; i++) {
65177e314f7STakashi Iwai 		hda_nid_t nid = spec->adc_nids[i];
65277e314f7STakashi Iwai 		if (!(get_wcaps(codec, nid) & AC_WCAP_IN_AMP) ||
65377e314f7STakashi Iwai 		    !(query_amp_caps(codec, nid, HDA_INPUT) & AC_AMPCAP_MUTE))
65477e314f7STakashi Iwai 			continue;
655096a8854STakashi Iwai 		snd_hda_codec_write(codec, spec->adc_nids[i], 0,
656096a8854STakashi Iwai 				    AC_VERB_SET_AMP_GAIN_MUTE,
657096a8854STakashi Iwai 				    AMP_IN_UNMUTE(0));
658096a8854STakashi Iwai 	}
659096a8854STakashi Iwai 
660096a8854STakashi Iwai 	/* init pins */
6617b315bb4STakashi Iwai 	for (i = 0; i < cfg->num_inputs; i++) {
6627b315bb4STakashi Iwai 		hda_nid_t nid = cfg->inputs[i].pin;
663f4a7828bSTakashi Iwai 		if (spec->smart51_enabled && is_smart51_pins(codec, nid))
66432e0191dSClemens Ladisch 			ctl = PIN_OUT;
6654740860bSTakashi Iwai 		else {
66632e0191dSClemens Ladisch 			ctl = PIN_IN;
6674740860bSTakashi Iwai 			if (cfg->inputs[i].type == AUTO_PIN_MIC)
6684740860bSTakashi Iwai 				ctl |= snd_hda_get_default_vref(codec, nid);
6694740860bSTakashi Iwai 		}
670cdd03cedSTakashi Iwai 		snd_hda_set_pin_ctl(codec, nid, ctl);
671c577b8a1SJoseph Chan 	}
672096a8854STakashi Iwai 
673096a8854STakashi Iwai 	/* init input-src */
674096a8854STakashi Iwai 	for (i = 0; i < spec->num_adc_nids; i++) {
675a86a88eaSTakashi Iwai 		int adc_idx = spec->inputs[spec->cur_mux[i]].adc_idx;
676fc1156c0STakashi Iwai 		/* secondary ADCs must have the unique MUX */
677fc1156c0STakashi Iwai 		if (i > 0 && !spec->mux_nids[i])
678fc1156c0STakashi Iwai 			break;
679a86a88eaSTakashi Iwai 		if (spec->mux_nids[adc_idx]) {
680a86a88eaSTakashi Iwai 			int mux_idx = spec->inputs[spec->cur_mux[i]].mux_idx;
681a86a88eaSTakashi Iwai 			snd_hda_codec_write(codec, spec->mux_nids[adc_idx], 0,
682096a8854STakashi Iwai 					    AC_VERB_SET_CONNECT_SEL,
683a86a88eaSTakashi Iwai 					    mux_idx);
684a86a88eaSTakashi Iwai 		}
685a86a88eaSTakashi Iwai 		if (spec->dyn_adc_switch)
686a86a88eaSTakashi Iwai 			break; /* only one input-src */
687096a8854STakashi Iwai 	}
688096a8854STakashi Iwai 
689096a8854STakashi Iwai 	/* init aa-mixer */
690096a8854STakashi Iwai 	if (!spec->aa_mix_nid)
691096a8854STakashi Iwai 		return;
692096a8854STakashi Iwai 	num_conns = snd_hda_get_connections(codec, spec->aa_mix_nid, conn,
693096a8854STakashi Iwai 					    ARRAY_SIZE(conn));
694096a8854STakashi Iwai 	for (i = 0; i < num_conns; i++) {
695096a8854STakashi Iwai 		unsigned int caps = get_wcaps(codec, conn[i]);
696096a8854STakashi Iwai 		if (get_wcaps_type(caps) == AC_WID_PIN)
697096a8854STakashi Iwai 			snd_hda_codec_write(codec, spec->aa_mix_nid, 0,
698096a8854STakashi Iwai 					    AC_VERB_SET_AMP_GAIN_MUTE,
699096a8854STakashi Iwai 					    AMP_IN_MUTE(i));
700096a8854STakashi Iwai 	}
701c577b8a1SJoseph Chan }
702f5271101SLydia Wang 
703054d867eSTakashi Iwai static void update_power_state(struct hda_codec *codec, hda_nid_t nid,
704054d867eSTakashi Iwai 			       unsigned int parm)
705054d867eSTakashi Iwai {
706054d867eSTakashi Iwai 	if (snd_hda_codec_read(codec, nid, 0,
707054d867eSTakashi Iwai 			       AC_VERB_GET_POWER_STATE, 0) == parm)
708054d867eSTakashi Iwai 		return;
709054d867eSTakashi Iwai 	snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_POWER_STATE, parm);
710054d867eSTakashi Iwai }
711054d867eSTakashi Iwai 
712f5271101SLydia Wang static void set_pin_power_state(struct hda_codec *codec, hda_nid_t nid,
713f5271101SLydia Wang 				unsigned int *affected_parm)
714f5271101SLydia Wang {
715f5271101SLydia Wang 	unsigned parm;
716f5271101SLydia Wang 	unsigned def_conf = snd_hda_codec_get_pincfg(codec, nid);
717f5271101SLydia Wang 	unsigned no_presence = (def_conf & AC_DEFCFG_MISC)
718f5271101SLydia Wang 		>> AC_DEFCFG_MISC_SHIFT
719f5271101SLydia Wang 		& AC_DEFCFG_MISC_NO_PRESENCE; /* do not support pin sense */
7201564b287SLydia Wang 	struct via_spec *spec = codec->spec;
72124088a58STakashi Iwai 	unsigned present = 0;
72224088a58STakashi Iwai 
72324088a58STakashi Iwai 	no_presence |= spec->no_pin_power_ctl;
72424088a58STakashi Iwai 	if (!no_presence)
72524088a58STakashi Iwai 		present = snd_hda_jack_detect(codec, nid);
726f4a7828bSTakashi Iwai 	if ((spec->smart51_enabled && is_smart51_pins(codec, nid))
7271564b287SLydia Wang 	    || ((no_presence || present)
7281564b287SLydia Wang 		&& get_defcfg_connect(def_conf) != AC_JACK_PORT_NONE)) {
729f5271101SLydia Wang 		*affected_parm = AC_PWRST_D0; /* if it's connected */
730f5271101SLydia Wang 		parm = AC_PWRST_D0;
731f5271101SLydia Wang 	} else
732f5271101SLydia Wang 		parm = AC_PWRST_D3;
733f5271101SLydia Wang 
734054d867eSTakashi Iwai 	update_power_state(codec, nid, parm);
735f5271101SLydia Wang }
736f5271101SLydia Wang 
73724088a58STakashi Iwai static int via_pin_power_ctl_info(struct snd_kcontrol *kcontrol,
73824088a58STakashi Iwai 				  struct snd_ctl_elem_info *uinfo)
73924088a58STakashi Iwai {
74024088a58STakashi Iwai 	static const char * const texts[] = {
74124088a58STakashi Iwai 		"Disabled", "Enabled"
74224088a58STakashi Iwai 	};
74324088a58STakashi Iwai 
74424088a58STakashi Iwai 	uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
74524088a58STakashi Iwai 	uinfo->count = 1;
74624088a58STakashi Iwai 	uinfo->value.enumerated.items = 2;
74724088a58STakashi Iwai 	if (uinfo->value.enumerated.item >= uinfo->value.enumerated.items)
74824088a58STakashi Iwai 		uinfo->value.enumerated.item = uinfo->value.enumerated.items - 1;
74924088a58STakashi Iwai 	strcpy(uinfo->value.enumerated.name,
75024088a58STakashi Iwai 	       texts[uinfo->value.enumerated.item]);
75124088a58STakashi Iwai 	return 0;
75224088a58STakashi Iwai }
75324088a58STakashi Iwai 
75424088a58STakashi Iwai static int via_pin_power_ctl_get(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 	ucontrol->value.enumerated.item[0] = !spec->no_pin_power_ctl;
76024088a58STakashi Iwai 	return 0;
76124088a58STakashi Iwai }
76224088a58STakashi Iwai 
76324088a58STakashi Iwai static int via_pin_power_ctl_put(struct snd_kcontrol *kcontrol,
76424088a58STakashi Iwai 				 struct snd_ctl_elem_value *ucontrol)
76524088a58STakashi Iwai {
76624088a58STakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
76724088a58STakashi Iwai 	struct via_spec *spec = codec->spec;
76824088a58STakashi Iwai 	unsigned int val = !ucontrol->value.enumerated.item[0];
76924088a58STakashi Iwai 
77024088a58STakashi Iwai 	if (val == spec->no_pin_power_ctl)
77124088a58STakashi Iwai 		return 0;
77224088a58STakashi Iwai 	spec->no_pin_power_ctl = val;
77324088a58STakashi Iwai 	set_widgets_power_state(codec);
774e9d010c2STakashi Iwai 	analog_low_current_mode(codec);
77524088a58STakashi Iwai 	return 1;
77624088a58STakashi Iwai }
77724088a58STakashi Iwai 
77824088a58STakashi Iwai static const struct snd_kcontrol_new via_pin_power_ctl_enum = {
77924088a58STakashi Iwai 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
78024088a58STakashi Iwai 	.name = "Dynamic Power-Control",
78124088a58STakashi Iwai 	.info = via_pin_power_ctl_info,
78224088a58STakashi Iwai 	.get = via_pin_power_ctl_get,
78324088a58STakashi Iwai 	.put = via_pin_power_ctl_put,
78424088a58STakashi Iwai };
78524088a58STakashi Iwai 
78624088a58STakashi Iwai 
7870aa62aefSHarald Welte static int via_independent_hp_info(struct snd_kcontrol *kcontrol,
7880aa62aefSHarald Welte 				   struct snd_ctl_elem_info *uinfo)
7890aa62aefSHarald Welte {
7908df2a312STakashi Iwai 	static const char * const texts[] = { "OFF", "ON" };
7918df2a312STakashi Iwai 
7928df2a312STakashi Iwai 	uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
7938df2a312STakashi Iwai 	uinfo->count = 1;
7948df2a312STakashi Iwai 	uinfo->value.enumerated.items = 2;
7958df2a312STakashi Iwai 	if (uinfo->value.enumerated.item >= 2)
7968df2a312STakashi Iwai 		uinfo->value.enumerated.item = 1;
7978df2a312STakashi Iwai 	strcpy(uinfo->value.enumerated.name,
7988df2a312STakashi Iwai 	       texts[uinfo->value.enumerated.item]);
7998df2a312STakashi Iwai 	return 0;
8000aa62aefSHarald Welte }
8010aa62aefSHarald Welte 
8020aa62aefSHarald Welte static int via_independent_hp_get(struct snd_kcontrol *kcontrol,
8030aa62aefSHarald Welte 				  struct snd_ctl_elem_value *ucontrol)
8040aa62aefSHarald Welte {
8050aa62aefSHarald Welte 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
806cdc1784dSLydia Wang 	struct via_spec *spec = codec->spec;
807cdc1784dSLydia Wang 
808ece8d043STakashi Iwai 	ucontrol->value.enumerated.item[0] = spec->hp_independent_mode;
809cdc1784dSLydia Wang 	return 0;
810cdc1784dSLydia Wang }
811cdc1784dSLydia Wang 
8123b607e3dSTakashi Iwai /* adjust spec->multiout setup according to the current flags */
8133b607e3dSTakashi Iwai static void setup_playback_multi_pcm(struct via_spec *spec)
8143b607e3dSTakashi Iwai {
8153b607e3dSTakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
8163b607e3dSTakashi Iwai 	spec->multiout.num_dacs = cfg->line_outs + spec->smart51_nums;
8173b607e3dSTakashi Iwai 	spec->multiout.hp_nid = 0;
8183b607e3dSTakashi Iwai 	if (!spec->hp_independent_mode) {
8193b607e3dSTakashi Iwai 		if (!spec->hp_indep_shared)
8203b607e3dSTakashi Iwai 			spec->multiout.hp_nid = spec->hp_dac_nid;
8213b607e3dSTakashi Iwai 	} else {
8223b607e3dSTakashi Iwai 		if (spec->hp_indep_shared)
8233b607e3dSTakashi Iwai 			spec->multiout.num_dacs = cfg->line_outs - 1;
8243b607e3dSTakashi Iwai 	}
8253b607e3dSTakashi Iwai }
8263b607e3dSTakashi Iwai 
8273b607e3dSTakashi Iwai /* update DAC setups according to indep-HP switch;
8283b607e3dSTakashi Iwai  * this function is called only when indep-HP is modified
8293b607e3dSTakashi Iwai  */
8303b607e3dSTakashi Iwai static void switch_indep_hp_dacs(struct hda_codec *codec)
8313b607e3dSTakashi Iwai {
8323b607e3dSTakashi Iwai 	struct via_spec *spec = codec->spec;
8333b607e3dSTakashi Iwai 	int shared = spec->hp_indep_shared;
8343b607e3dSTakashi Iwai 	hda_nid_t shared_dac, hp_dac;
8353b607e3dSTakashi Iwai 
8363b607e3dSTakashi Iwai 	if (!spec->opened_streams)
8373b607e3dSTakashi Iwai 		return;
8383b607e3dSTakashi Iwai 
8393b607e3dSTakashi Iwai 	shared_dac = shared ? spec->multiout.dac_nids[shared] : 0;
8403b607e3dSTakashi Iwai 	hp_dac = spec->hp_dac_nid;
8413b607e3dSTakashi Iwai 	if (spec->hp_independent_mode) {
8423b607e3dSTakashi Iwai 		/* switch to indep-HP mode */
8433b607e3dSTakashi Iwai 		if (spec->active_streams & STREAM_MULTI_OUT) {
8443b607e3dSTakashi Iwai 			__snd_hda_codec_cleanup_stream(codec, hp_dac, 1);
8453b607e3dSTakashi Iwai 			__snd_hda_codec_cleanup_stream(codec, shared_dac, 1);
8463b607e3dSTakashi Iwai 		}
8473b607e3dSTakashi Iwai 		if (spec->active_streams & STREAM_INDEP_HP)
8483b607e3dSTakashi Iwai 			snd_hda_codec_setup_stream(codec, hp_dac,
8493b607e3dSTakashi Iwai 						   spec->cur_hp_stream_tag, 0,
8503b607e3dSTakashi Iwai 						   spec->cur_hp_format);
8513b607e3dSTakashi Iwai 	} else {
8523b607e3dSTakashi Iwai 		/* back to HP or shared-DAC */
8533b607e3dSTakashi Iwai 		if (spec->active_streams & STREAM_INDEP_HP)
8543b607e3dSTakashi Iwai 			__snd_hda_codec_cleanup_stream(codec, hp_dac, 1);
8553b607e3dSTakashi Iwai 		if (spec->active_streams & STREAM_MULTI_OUT) {
8563b607e3dSTakashi Iwai 			hda_nid_t dac;
8573b607e3dSTakashi Iwai 			int ch;
8583b607e3dSTakashi Iwai 			if (shared_dac) { /* reset mutli-ch DAC */
8593b607e3dSTakashi Iwai 				dac = shared_dac;
8603b607e3dSTakashi Iwai 				ch = shared * 2;
8613b607e3dSTakashi Iwai 			} else { /* reset HP DAC */
8623b607e3dSTakashi Iwai 				dac = hp_dac;
8633b607e3dSTakashi Iwai 				ch = 0;
8643b607e3dSTakashi Iwai 			}
8653b607e3dSTakashi Iwai 			snd_hda_codec_setup_stream(codec, dac,
8663b607e3dSTakashi Iwai 						   spec->cur_dac_stream_tag, ch,
8673b607e3dSTakashi Iwai 						   spec->cur_dac_format);
8683b607e3dSTakashi Iwai 		}
8693b607e3dSTakashi Iwai 	}
8703b607e3dSTakashi Iwai 	setup_playback_multi_pcm(spec);
8713b607e3dSTakashi Iwai }
8723b607e3dSTakashi Iwai 
8730aa62aefSHarald Welte static int via_independent_hp_put(struct snd_kcontrol *kcontrol,
8740aa62aefSHarald Welte 				  struct snd_ctl_elem_value *ucontrol)
8750aa62aefSHarald Welte {
8760aa62aefSHarald Welte 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
8770aa62aefSHarald Welte 	struct via_spec *spec = codec->spec;
8783214b966STakashi Iwai 	int cur, shared;
8798df2a312STakashi Iwai 
8803b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
88125250505STakashi Iwai 	cur = !!ucontrol->value.enumerated.item[0];
8823b607e3dSTakashi Iwai 	if (spec->hp_independent_mode == cur) {
8833b607e3dSTakashi Iwai 		mutex_unlock(&spec->config_mutex);
88425250505STakashi Iwai 		return 0;
8853b607e3dSTakashi Iwai 	}
88625250505STakashi Iwai 	spec->hp_independent_mode = cur;
8873214b966STakashi Iwai 	shared = spec->hp_indep_shared;
888020066d1STakashi Iwai 	deactivate_hp_paths(codec);
889020066d1STakashi Iwai 	if (cur)
890020066d1STakashi Iwai 		activate_output_path(codec, &spec->hp_indep_path, true, false);
891020066d1STakashi Iwai 	else {
8923214b966STakashi Iwai 		if (shared)
8933214b966STakashi Iwai 			activate_output_path(codec, &spec->out_path[shared],
89425250505STakashi Iwai 					     true, false);
895020066d1STakashi Iwai 		if (spec->aamix_mode || !spec->hp_path.depth)
896020066d1STakashi Iwai 			activate_output_path(codec, &spec->hp_mix_path,
897020066d1STakashi Iwai 					     true, false);
898020066d1STakashi Iwai 		else
899020066d1STakashi Iwai 			activate_output_path(codec, &spec->hp_path,
900020066d1STakashi Iwai 					     true, false);
9018df2a312STakashi Iwai 	}
9020aa62aefSHarald Welte 
9033b607e3dSTakashi Iwai 	switch_indep_hp_dacs(codec);
9043b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
9053b607e3dSTakashi Iwai 
906ce0e5a9eSLydia Wang 	/* update jack power state */
9073e95b9abSLydia Wang 	set_widgets_power_state(codec);
9086e969d91STakashi Iwai 	via_hp_automute(codec);
90925250505STakashi Iwai 	return 1;
9100aa62aefSHarald Welte }
9110aa62aefSHarald Welte 
912ece8d043STakashi Iwai static const struct snd_kcontrol_new via_hp_mixer = {
9130aa62aefSHarald Welte 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
9140aa62aefSHarald Welte 	.name = "Independent HP",
9150aa62aefSHarald Welte 	.info = via_independent_hp_info,
9160aa62aefSHarald Welte 	.get = via_independent_hp_get,
9170aa62aefSHarald Welte 	.put = via_independent_hp_put,
9180aa62aefSHarald Welte };
9190aa62aefSHarald Welte 
9203d83e577STakashi Iwai static int via_hp_build(struct hda_codec *codec)
9215b0cb1d8SJaroslav Kysela {
9223d83e577STakashi Iwai 	struct via_spec *spec = codec->spec;
9235b0cb1d8SJaroslav Kysela 	struct snd_kcontrol_new *knew;
9245b0cb1d8SJaroslav Kysela 	hda_nid_t nid;
9255b0cb1d8SJaroslav Kysela 
9265b0cb1d8SJaroslav Kysela 	nid = spec->autocfg.hp_pins[0];
927ece8d043STakashi Iwai 	knew = via_clone_control(spec, &via_hp_mixer);
9283d83e577STakashi Iwai 	if (knew == NULL)
9293d83e577STakashi Iwai 		return -ENOMEM;
9303d83e577STakashi Iwai 
9315b0cb1d8SJaroslav Kysela 	knew->subdevice = HDA_SUBDEV_NID_FLAG | nid;
9325b0cb1d8SJaroslav Kysela 
9335b0cb1d8SJaroslav Kysela 	return 0;
9345b0cb1d8SJaroslav Kysela }
9355b0cb1d8SJaroslav Kysela 
9361564b287SLydia Wang static void notify_aa_path_ctls(struct hda_codec *codec)
9371564b287SLydia Wang {
938e3d7a143STakashi Iwai 	struct via_spec *spec = codec->spec;
9391564b287SLydia Wang 	int i;
9401564b287SLydia Wang 
941e3d7a143STakashi Iwai 	for (i = 0; i < spec->smart51_nums; i++) {
942e3d7a143STakashi Iwai 		struct snd_kcontrol *ctl;
943e3d7a143STakashi Iwai 		struct snd_ctl_elem_id id;
9441564b287SLydia Wang 		memset(&id, 0, sizeof(id));
9451564b287SLydia Wang 		id.iface = SNDRV_CTL_ELEM_IFACE_MIXER;
946e3d7a143STakashi Iwai 		sprintf(id.name, "%s Playback Volume", spec->smart51_labels[i]);
947525566cbSLydia Wang 		ctl = snd_hda_find_mixer_ctl(codec, id.name);
948525566cbSLydia Wang 		if (ctl)
949525566cbSLydia Wang 			snd_ctl_notify(codec->bus->card,
950525566cbSLydia Wang 					SNDRV_CTL_EVENT_MASK_VALUE,
951525566cbSLydia Wang 					&ctl->id);
9521564b287SLydia Wang 	}
9531564b287SLydia Wang }
9541564b287SLydia Wang 
9551564b287SLydia Wang static void mute_aa_path(struct hda_codec *codec, int mute)
9561564b287SLydia Wang {
9571564b287SLydia Wang 	struct via_spec *spec = codec->spec;
9581564b287SLydia Wang 	int val = mute ? HDA_AMP_MUTE : HDA_AMP_UNMUTE;
959e3d7a143STakashi Iwai 	int i;
960e3d7a143STakashi Iwai 
961e3d7a143STakashi Iwai 	/* check AA path's mute status */
962e3d7a143STakashi Iwai 	for (i = 0; i < spec->smart51_nums; i++) {
963e3d7a143STakashi Iwai 		if (spec->smart51_idxs[i] < 0)
964e3d7a143STakashi Iwai 			continue;
965e3d7a143STakashi Iwai 		snd_hda_codec_amp_stereo(codec, spec->aa_mix_nid,
966e3d7a143STakashi Iwai 					 HDA_INPUT, spec->smart51_idxs[i],
9671564b287SLydia Wang 					 HDA_AMP_MUTE, val);
9681564b287SLydia Wang 	}
9691564b287SLydia Wang }
970f4a7828bSTakashi Iwai 
971e3d7a143STakashi Iwai static bool is_smart51_pins(struct hda_codec *codec, hda_nid_t pin)
972e3d7a143STakashi Iwai {
973e3d7a143STakashi Iwai 	struct via_spec *spec = codec->spec;
974e3d7a143STakashi Iwai 	int i;
975e3d7a143STakashi Iwai 
976e3d7a143STakashi Iwai 	for (i = 0; i < spec->smart51_nums; i++)
977e3d7a143STakashi Iwai 		if (spec->smart51_pins[i] == pin)
978e3d7a143STakashi Iwai 			return true;
979e3d7a143STakashi Iwai 	return false;
980e3d7a143STakashi Iwai }
981e3d7a143STakashi Iwai 
9821564b287SLydia Wang static int via_smart51_get(struct snd_kcontrol *kcontrol,
9831564b287SLydia Wang 			   struct snd_ctl_elem_value *ucontrol)
9841564b287SLydia Wang {
9851564b287SLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
9861564b287SLydia Wang 	struct via_spec *spec = codec->spec;
9871564b287SLydia Wang 
988f2b1c9f0STakashi Iwai 	*ucontrol->value.integer.value = spec->smart51_enabled;
9891564b287SLydia Wang 	return 0;
9901564b287SLydia Wang }
9911564b287SLydia Wang 
9921564b287SLydia Wang static int via_smart51_put(struct snd_kcontrol *kcontrol,
9931564b287SLydia Wang 			   struct snd_ctl_elem_value *ucontrol)
9941564b287SLydia Wang {
9951564b287SLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
9961564b287SLydia Wang 	struct via_spec *spec = codec->spec;
9971564b287SLydia Wang 	int out_in = *ucontrol->value.integer.value
9981564b287SLydia Wang 		? AC_PINCTL_OUT_EN : AC_PINCTL_IN_EN;
9991564b287SLydia Wang 	int i;
10001564b287SLydia Wang 
1001e3d7a143STakashi Iwai 	for (i = 0; i < spec->smart51_nums; i++) {
1002e3d7a143STakashi Iwai 		hda_nid_t nid = spec->smart51_pins[i];
10037b315bb4STakashi Iwai 		unsigned int parm;
10047b315bb4STakashi Iwai 
10057b315bb4STakashi Iwai 		parm = snd_hda_codec_read(codec, nid, 0,
10061564b287SLydia Wang 					  AC_VERB_GET_PIN_WIDGET_CONTROL, 0);
10071564b287SLydia Wang 		parm &= ~(AC_PINCTL_IN_EN | AC_PINCTL_OUT_EN);
10081564b287SLydia Wang 		parm |= out_in;
1009cdd03cedSTakashi Iwai 		snd_hda_set_pin_ctl(codec, nid, parm);
10101564b287SLydia Wang 		if (out_in == AC_PINCTL_OUT_EN) {
10111564b287SLydia Wang 			mute_aa_path(codec, 1);
10121564b287SLydia Wang 			notify_aa_path_ctls(codec);
10131564b287SLydia Wang 		}
10141564b287SLydia Wang 	}
10151564b287SLydia Wang 	spec->smart51_enabled = *ucontrol->value.integer.value;
10163e95b9abSLydia Wang 	set_widgets_power_state(codec);
10171564b287SLydia Wang 	return 1;
10181564b287SLydia Wang }
10191564b287SLydia Wang 
10205f4b36d6STakashi Iwai static const struct snd_kcontrol_new via_smart51_mixer = {
10211564b287SLydia Wang 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
10221564b287SLydia Wang 	.name = "Smart 5.1",
10231564b287SLydia Wang 	.count = 1,
1024f2b1c9f0STakashi Iwai 	.info = snd_ctl_boolean_mono_info,
10251564b287SLydia Wang 	.get = via_smart51_get,
10261564b287SLydia Wang 	.put = via_smart51_put,
10271564b287SLydia Wang };
10281564b287SLydia Wang 
1029f4a7828bSTakashi Iwai static int via_smart51_build(struct hda_codec *codec)
10305b0cb1d8SJaroslav Kysela {
1031f4a7828bSTakashi Iwai 	struct via_spec *spec = codec->spec;
10325b0cb1d8SJaroslav Kysela 
1033e3d7a143STakashi Iwai 	if (!spec->smart51_nums)
1034cb34c207SLydia Wang 		return 0;
1035e3d7a143STakashi Iwai 	if (!via_clone_control(spec, &via_smart51_mixer))
10365b0cb1d8SJaroslav Kysela 		return -ENOMEM;
10375b0cb1d8SJaroslav Kysela 	return 0;
10385b0cb1d8SJaroslav Kysela }
10395b0cb1d8SJaroslav Kysela 
1040f5271101SLydia Wang /* check AA path's mute status */
1041ada509ecSTakashi Iwai static bool is_aa_path_mute(struct hda_codec *codec)
1042ada509ecSTakashi Iwai {
1043ada509ecSTakashi Iwai 	struct via_spec *spec = codec->spec;
1044ada509ecSTakashi Iwai 	const struct hda_amp_list *p;
1045ada509ecSTakashi Iwai 	int i, ch, v;
1046ada509ecSTakashi Iwai 
1047ada509ecSTakashi Iwai 	for (i = 0; i < spec->num_loopbacks; i++) {
1048ada509ecSTakashi Iwai 		p = &spec->loopback_list[i];
1049ada509ecSTakashi Iwai 		for (ch = 0; ch < 2; ch++) {
1050ada509ecSTakashi Iwai 			v = snd_hda_codec_amp_read(codec, p->nid, ch, p->dir,
1051ada509ecSTakashi Iwai 						   p->idx);
1052ada509ecSTakashi Iwai 			if (!(v & HDA_AMP_MUTE) && v > 0)
1053ada509ecSTakashi Iwai 				return false;
1054f5271101SLydia Wang 		}
1055f5271101SLydia Wang 	}
1056ada509ecSTakashi Iwai 	return true;
1057f5271101SLydia Wang }
1058f5271101SLydia Wang 
1059f5271101SLydia Wang /* enter/exit analog low-current mode */
1060e9d010c2STakashi Iwai static void __analog_low_current_mode(struct hda_codec *codec, bool force)
1061f5271101SLydia Wang {
1062f5271101SLydia Wang 	struct via_spec *spec = codec->spec;
1063ada509ecSTakashi Iwai 	bool enable;
1064ada509ecSTakashi Iwai 	unsigned int verb, parm;
1065f5271101SLydia Wang 
1066e9d010c2STakashi Iwai 	if (spec->no_pin_power_ctl)
1067e9d010c2STakashi Iwai 		enable = false;
1068e9d010c2STakashi Iwai 	else
106992433923STakashi Iwai 		enable = is_aa_path_mute(codec) && !spec->opened_streams;
1070e9d010c2STakashi Iwai 	if (enable == spec->alc_mode && !force)
1071e9d010c2STakashi Iwai 		return;
1072e9d010c2STakashi Iwai 	spec->alc_mode = enable;
1073f5271101SLydia Wang 
1074f5271101SLydia Wang 	/* decide low current mode's verb & parameter */
1075f5271101SLydia Wang 	switch (spec->codec_type) {
1076f5271101SLydia Wang 	case VT1708B_8CH:
1077f5271101SLydia Wang 	case VT1708B_4CH:
1078f5271101SLydia Wang 		verb = 0xf70;
1079f5271101SLydia Wang 		parm = enable ? 0x02 : 0x00; /* 0x02: 2/3x, 0x00: 1x */
1080f5271101SLydia Wang 		break;
1081f5271101SLydia Wang 	case VT1708S:
1082eb7188caSLydia Wang 	case VT1718S:
1083f3db423dSLydia Wang 	case VT1716S:
1084f5271101SLydia Wang 		verb = 0xf73;
1085f5271101SLydia Wang 		parm = enable ? 0x51 : 0xe1; /* 0x51: 4/28x, 0xe1: 1x */
1086f5271101SLydia Wang 		break;
1087f5271101SLydia Wang 	case VT1702:
1088f5271101SLydia Wang 		verb = 0xf73;
1089f5271101SLydia Wang 		parm = enable ? 0x01 : 0x1d; /* 0x01: 4/40x, 0x1d: 1x */
1090f5271101SLydia Wang 		break;
109125eaba2fSLydia Wang 	case VT2002P:
1092ab6734e7SLydia Wang 	case VT1812:
109311890956SLydia Wang 	case VT1802:
109425eaba2fSLydia Wang 		verb = 0xf93;
109525eaba2fSLydia Wang 		parm = enable ? 0x00 : 0xe0; /* 0x00: 4/40x, 0xe0: 1x */
109625eaba2fSLydia Wang 		break;
1097f5271101SLydia Wang 	default:
1098f5271101SLydia Wang 		return;		/* other codecs are not supported */
1099f5271101SLydia Wang 	}
1100f5271101SLydia Wang 	/* send verb */
1101f5271101SLydia Wang 	snd_hda_codec_write(codec, codec->afg, 0, verb, parm);
1102f5271101SLydia Wang }
1103f5271101SLydia Wang 
1104e9d010c2STakashi Iwai static void analog_low_current_mode(struct hda_codec *codec)
1105e9d010c2STakashi Iwai {
1106e9d010c2STakashi Iwai 	return __analog_low_current_mode(codec, false);
1107e9d010c2STakashi Iwai }
1108e9d010c2STakashi Iwai 
1109c577b8a1SJoseph Chan /*
1110c577b8a1SJoseph Chan  * generic initialization of ADC, input mixers and output mixers
1111c577b8a1SJoseph Chan  */
1112096a8854STakashi Iwai static const struct hda_verb vt1708_init_verbs[] = {
1113aa266fccSLydia Wang 	/* power down jack detect function */
1114aa266fccSLydia Wang 	{0x1, 0xf81, 0x1},
1115f7278fd0SJosepch Chan 	{ }
1116c577b8a1SJoseph Chan };
1117c577b8a1SJoseph Chan 
11183b607e3dSTakashi Iwai static void set_stream_open(struct hda_codec *codec, int bit, bool active)
11197eb56e84STakashi Iwai {
1120ada509ecSTakashi Iwai 	struct via_spec *spec = codec->spec;
1121ada509ecSTakashi Iwai 
1122ada509ecSTakashi Iwai 	if (active)
11233b607e3dSTakashi Iwai 		spec->opened_streams |= bit;
1124ada509ecSTakashi Iwai 	else
11253b607e3dSTakashi Iwai 		spec->opened_streams &= ~bit;
1126ada509ecSTakashi Iwai 	analog_low_current_mode(codec);
11277eb56e84STakashi Iwai }
11287eb56e84STakashi Iwai 
1129ece8d043STakashi Iwai static int via_playback_multi_pcm_open(struct hda_pcm_stream *hinfo,
1130c577b8a1SJoseph Chan 				 struct hda_codec *codec,
1131c577b8a1SJoseph Chan 				 struct snd_pcm_substream *substream)
1132c577b8a1SJoseph Chan {
1133c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
113425250505STakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
1135ada509ecSTakashi Iwai 	int err;
1136ece8d043STakashi Iwai 
113725250505STakashi Iwai 	spec->multiout.num_dacs = cfg->line_outs + spec->smart51_nums;
113825250505STakashi Iwai 	spec->multiout.max_channels = spec->multiout.num_dacs * 2;
11393b607e3dSTakashi Iwai 	set_stream_open(codec, STREAM_MULTI_OUT, true);
1140ada509ecSTakashi Iwai 	err = snd_hda_multi_out_analog_open(codec, &spec->multiout, substream,
11419a08160bSTakashi Iwai 					    hinfo);
1142ada509ecSTakashi Iwai 	if (err < 0) {
11433b607e3dSTakashi Iwai 		set_stream_open(codec, STREAM_MULTI_OUT, false);
1144ada509ecSTakashi Iwai 		return err;
1145ada509ecSTakashi Iwai 	}
1146ada509ecSTakashi Iwai 	return 0;
1147c577b8a1SJoseph Chan }
1148c577b8a1SJoseph Chan 
1149ece8d043STakashi Iwai static int via_playback_multi_pcm_close(struct hda_pcm_stream *hinfo,
11509af74210STakashi Iwai 				  struct hda_codec *codec,
11519af74210STakashi Iwai 				  struct snd_pcm_substream *substream)
11529af74210STakashi Iwai {
11533b607e3dSTakashi Iwai 	set_stream_open(codec, STREAM_MULTI_OUT, false);
11549af74210STakashi Iwai 	return 0;
11559af74210STakashi Iwai }
11569af74210STakashi Iwai 
11577eb56e84STakashi Iwai static int via_playback_hp_pcm_open(struct hda_pcm_stream *hinfo,
11587eb56e84STakashi Iwai 				    struct hda_codec *codec,
11597eb56e84STakashi Iwai 				    struct snd_pcm_substream *substream)
11607eb56e84STakashi Iwai {
11617eb56e84STakashi Iwai 	struct via_spec *spec = codec->spec;
11627eb56e84STakashi Iwai 
1163ece8d043STakashi Iwai 	if (snd_BUG_ON(!spec->hp_dac_nid))
11647eb56e84STakashi Iwai 		return -EINVAL;
11653b607e3dSTakashi Iwai 	set_stream_open(codec, STREAM_INDEP_HP, true);
1166ece8d043STakashi Iwai 	return 0;
1167ece8d043STakashi Iwai }
1168ece8d043STakashi Iwai 
1169ece8d043STakashi Iwai static int via_playback_hp_pcm_close(struct hda_pcm_stream *hinfo,
1170ece8d043STakashi Iwai 				     struct hda_codec *codec,
1171ece8d043STakashi Iwai 				     struct snd_pcm_substream *substream)
1172ece8d043STakashi Iwai {
11733b607e3dSTakashi Iwai 	set_stream_open(codec, STREAM_INDEP_HP, false);
11747eb56e84STakashi Iwai 	return 0;
11757eb56e84STakashi Iwai }
11767eb56e84STakashi Iwai 
11777eb56e84STakashi Iwai static int via_playback_multi_pcm_prepare(struct hda_pcm_stream *hinfo,
11787eb56e84STakashi Iwai 					  struct hda_codec *codec,
11790aa62aefSHarald Welte 					  unsigned int stream_tag,
11800aa62aefSHarald Welte 					  unsigned int format,
11810aa62aefSHarald Welte 					  struct snd_pcm_substream *substream)
11820aa62aefSHarald Welte {
11830aa62aefSHarald Welte 	struct via_spec *spec = codec->spec;
11840aa62aefSHarald Welte 
11853b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
11863b607e3dSTakashi Iwai 	setup_playback_multi_pcm(spec);
1187ece8d043STakashi Iwai 	snd_hda_multi_out_analog_prepare(codec, &spec->multiout, stream_tag,
1188ece8d043STakashi Iwai 					 format, substream);
11893b607e3dSTakashi Iwai 	/* remember for dynamic DAC switch with indep-HP */
11903b607e3dSTakashi Iwai 	spec->active_streams |= STREAM_MULTI_OUT;
11913b607e3dSTakashi Iwai 	spec->cur_dac_stream_tag = stream_tag;
11923b607e3dSTakashi Iwai 	spec->cur_dac_format = format;
11933b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
1194187d333eSTakashi Iwai 	vt1708_update_hp_work(spec);
11957eb56e84STakashi Iwai 	return 0;
11960aa62aefSHarald Welte }
11970aa62aefSHarald Welte 
11987eb56e84STakashi Iwai static int via_playback_hp_pcm_prepare(struct hda_pcm_stream *hinfo,
11990aa62aefSHarald Welte 				       struct hda_codec *codec,
12000aa62aefSHarald Welte 				       unsigned int stream_tag,
12010aa62aefSHarald Welte 				       unsigned int format,
12020aa62aefSHarald Welte 				       struct snd_pcm_substream *substream)
12030aa62aefSHarald Welte {
12040aa62aefSHarald Welte 	struct via_spec *spec = codec->spec;
12050aa62aefSHarald Welte 
12063b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
12073b607e3dSTakashi Iwai 	if (spec->hp_independent_mode)
1208ece8d043STakashi Iwai 		snd_hda_codec_setup_stream(codec, spec->hp_dac_nid,
1209ece8d043STakashi Iwai 					   stream_tag, 0, format);
12103b607e3dSTakashi Iwai 	spec->active_streams |= STREAM_INDEP_HP;
12113b607e3dSTakashi Iwai 	spec->cur_hp_stream_tag = stream_tag;
12123b607e3dSTakashi Iwai 	spec->cur_hp_format = format;
12133b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
1214187d333eSTakashi Iwai 	vt1708_update_hp_work(spec);
12150aa62aefSHarald Welte 	return 0;
12160aa62aefSHarald Welte }
12170aa62aefSHarald Welte 
12180aa62aefSHarald Welte static int via_playback_multi_pcm_cleanup(struct hda_pcm_stream *hinfo,
12190aa62aefSHarald Welte 				    struct hda_codec *codec,
12200aa62aefSHarald Welte 				    struct snd_pcm_substream *substream)
12210aa62aefSHarald Welte {
12220aa62aefSHarald Welte 	struct via_spec *spec = codec->spec;
12230aa62aefSHarald Welte 
12243b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
1225ece8d043STakashi Iwai 	snd_hda_multi_out_analog_cleanup(codec, &spec->multiout);
12263b607e3dSTakashi Iwai 	spec->active_streams &= ~STREAM_MULTI_OUT;
12273b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
1228187d333eSTakashi Iwai 	vt1708_update_hp_work(spec);
12297eb56e84STakashi Iwai 	return 0;
12300aa62aefSHarald Welte }
12317eb56e84STakashi Iwai 
12327eb56e84STakashi Iwai static int via_playback_hp_pcm_cleanup(struct hda_pcm_stream *hinfo,
12337eb56e84STakashi Iwai 				       struct hda_codec *codec,
12347eb56e84STakashi Iwai 				       struct snd_pcm_substream *substream)
12357eb56e84STakashi Iwai {
12367eb56e84STakashi Iwai 	struct via_spec *spec = codec->spec;
12377eb56e84STakashi Iwai 
12383b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
12393b607e3dSTakashi Iwai 	if (spec->hp_independent_mode)
1240ece8d043STakashi Iwai 		snd_hda_codec_setup_stream(codec, spec->hp_dac_nid, 0, 0, 0);
12413b607e3dSTakashi Iwai 	spec->active_streams &= ~STREAM_INDEP_HP;
12423b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
1243187d333eSTakashi Iwai 	vt1708_update_hp_work(spec);
12440aa62aefSHarald Welte 	return 0;
12450aa62aefSHarald Welte }
12460aa62aefSHarald Welte 
1247c577b8a1SJoseph Chan /*
1248c577b8a1SJoseph Chan  * Digital out
1249c577b8a1SJoseph Chan  */
1250c577b8a1SJoseph Chan static int via_dig_playback_pcm_open(struct hda_pcm_stream *hinfo,
1251c577b8a1SJoseph Chan 				     struct hda_codec *codec,
1252c577b8a1SJoseph Chan 				     struct snd_pcm_substream *substream)
1253c577b8a1SJoseph Chan {
1254c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
1255c577b8a1SJoseph Chan 	return snd_hda_multi_out_dig_open(codec, &spec->multiout);
1256c577b8a1SJoseph Chan }
1257c577b8a1SJoseph Chan 
1258c577b8a1SJoseph Chan static int via_dig_playback_pcm_close(struct hda_pcm_stream *hinfo,
1259c577b8a1SJoseph Chan 				      struct hda_codec *codec,
1260c577b8a1SJoseph Chan 				      struct snd_pcm_substream *substream)
1261c577b8a1SJoseph Chan {
1262c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
1263c577b8a1SJoseph Chan 	return snd_hda_multi_out_dig_close(codec, &spec->multiout);
1264c577b8a1SJoseph Chan }
1265c577b8a1SJoseph Chan 
12665691ec7fSHarald Welte static int via_dig_playback_pcm_prepare(struct hda_pcm_stream *hinfo,
126798aa34c0SHarald Welte 					struct hda_codec *codec,
126898aa34c0SHarald Welte 					unsigned int stream_tag,
126998aa34c0SHarald Welte 					unsigned int format,
127098aa34c0SHarald Welte 					struct snd_pcm_substream *substream)
127198aa34c0SHarald Welte {
127298aa34c0SHarald Welte 	struct via_spec *spec = codec->spec;
12739da29271STakashi Iwai 	return snd_hda_multi_out_dig_prepare(codec, &spec->multiout,
12749da29271STakashi Iwai 					     stream_tag, format, substream);
12759da29271STakashi Iwai }
12765691ec7fSHarald Welte 
12779da29271STakashi Iwai static int via_dig_playback_pcm_cleanup(struct hda_pcm_stream *hinfo,
12789da29271STakashi Iwai 					struct hda_codec *codec,
12799da29271STakashi Iwai 					struct snd_pcm_substream *substream)
12809da29271STakashi Iwai {
12819da29271STakashi Iwai 	struct via_spec *spec = codec->spec;
12829da29271STakashi Iwai 	snd_hda_multi_out_dig_cleanup(codec, &spec->multiout);
128398aa34c0SHarald Welte 	return 0;
128498aa34c0SHarald Welte }
128598aa34c0SHarald Welte 
1286c577b8a1SJoseph Chan /*
1287c577b8a1SJoseph Chan  * Analog capture
1288c577b8a1SJoseph Chan  */
1289c577b8a1SJoseph Chan static int via_capture_pcm_prepare(struct hda_pcm_stream *hinfo,
1290c577b8a1SJoseph Chan 				   struct hda_codec *codec,
1291c577b8a1SJoseph Chan 				   unsigned int stream_tag,
1292c577b8a1SJoseph Chan 				   unsigned int format,
1293c577b8a1SJoseph Chan 				   struct snd_pcm_substream *substream)
1294c577b8a1SJoseph Chan {
1295c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
1296c577b8a1SJoseph Chan 
1297c577b8a1SJoseph Chan 	snd_hda_codec_setup_stream(codec, spec->adc_nids[substream->number],
1298c577b8a1SJoseph Chan 				   stream_tag, 0, format);
1299c577b8a1SJoseph Chan 	return 0;
1300c577b8a1SJoseph Chan }
1301c577b8a1SJoseph Chan 
1302c577b8a1SJoseph Chan static int via_capture_pcm_cleanup(struct hda_pcm_stream *hinfo,
1303c577b8a1SJoseph Chan 				   struct hda_codec *codec,
1304c577b8a1SJoseph Chan 				   struct snd_pcm_substream *substream)
1305c577b8a1SJoseph Chan {
1306c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
1307888afa15STakashi Iwai 	snd_hda_codec_cleanup_stream(codec, spec->adc_nids[substream->number]);
1308c577b8a1SJoseph Chan 	return 0;
1309c577b8a1SJoseph Chan }
1310c577b8a1SJoseph Chan 
1311a86a88eaSTakashi Iwai /* analog capture with dynamic ADC switching */
1312a86a88eaSTakashi Iwai static int via_dyn_adc_capture_pcm_prepare(struct hda_pcm_stream *hinfo,
1313a86a88eaSTakashi Iwai 					   struct hda_codec *codec,
1314a86a88eaSTakashi Iwai 					   unsigned int stream_tag,
1315a86a88eaSTakashi Iwai 					   unsigned int format,
1316a86a88eaSTakashi Iwai 					   struct snd_pcm_substream *substream)
1317a86a88eaSTakashi Iwai {
1318a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
1319a86a88eaSTakashi Iwai 	int adc_idx = spec->inputs[spec->cur_mux[0]].adc_idx;
1320a86a88eaSTakashi Iwai 
13213b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
1322a86a88eaSTakashi Iwai 	spec->cur_adc = spec->adc_nids[adc_idx];
1323a86a88eaSTakashi Iwai 	spec->cur_adc_stream_tag = stream_tag;
1324a86a88eaSTakashi Iwai 	spec->cur_adc_format = format;
1325a86a88eaSTakashi Iwai 	snd_hda_codec_setup_stream(codec, spec->cur_adc, stream_tag, 0, format);
13263b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
1327a86a88eaSTakashi Iwai 	return 0;
1328a86a88eaSTakashi Iwai }
1329a86a88eaSTakashi Iwai 
1330a86a88eaSTakashi Iwai static int via_dyn_adc_capture_pcm_cleanup(struct hda_pcm_stream *hinfo,
1331a86a88eaSTakashi Iwai 					   struct hda_codec *codec,
1332a86a88eaSTakashi Iwai 					   struct snd_pcm_substream *substream)
1333a86a88eaSTakashi Iwai {
1334a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
1335a86a88eaSTakashi Iwai 
13363b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
1337a86a88eaSTakashi Iwai 	snd_hda_codec_cleanup_stream(codec, spec->cur_adc);
1338a86a88eaSTakashi Iwai 	spec->cur_adc = 0;
13393b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
1340a86a88eaSTakashi Iwai 	return 0;
1341a86a88eaSTakashi Iwai }
1342a86a88eaSTakashi Iwai 
1343a86a88eaSTakashi Iwai /* re-setup the stream if running; called from input-src put */
1344a86a88eaSTakashi Iwai static bool via_dyn_adc_pcm_resetup(struct hda_codec *codec, int cur)
1345a86a88eaSTakashi Iwai {
1346a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
1347a86a88eaSTakashi Iwai 	int adc_idx = spec->inputs[cur].adc_idx;
1348a86a88eaSTakashi Iwai 	hda_nid_t adc = spec->adc_nids[adc_idx];
13493b607e3dSTakashi Iwai 	bool ret = false;
1350a86a88eaSTakashi Iwai 
13513b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
1352a86a88eaSTakashi Iwai 	if (spec->cur_adc && spec->cur_adc != adc) {
1353a86a88eaSTakashi Iwai 		/* stream is running, let's swap the current ADC */
1354a86a88eaSTakashi Iwai 		__snd_hda_codec_cleanup_stream(codec, spec->cur_adc, 1);
1355a86a88eaSTakashi Iwai 		spec->cur_adc = adc;
1356a86a88eaSTakashi Iwai 		snd_hda_codec_setup_stream(codec, adc,
1357a86a88eaSTakashi Iwai 					   spec->cur_adc_stream_tag, 0,
1358a86a88eaSTakashi Iwai 					   spec->cur_adc_format);
13593b607e3dSTakashi Iwai 		ret = true;
1360a86a88eaSTakashi Iwai 	}
13613b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
13623b607e3dSTakashi Iwai 	return ret;
1363a86a88eaSTakashi Iwai }
1364a86a88eaSTakashi Iwai 
13659af74210STakashi Iwai static const struct hda_pcm_stream via_pcm_analog_playback = {
13667eb56e84STakashi Iwai 	.substreams = 1,
1367c577b8a1SJoseph Chan 	.channels_min = 2,
1368c577b8a1SJoseph Chan 	.channels_max = 8,
13699af74210STakashi Iwai 	/* NID is set in via_build_pcms */
1370c577b8a1SJoseph Chan 	.ops = {
1371ece8d043STakashi Iwai 		.open = via_playback_multi_pcm_open,
1372ece8d043STakashi Iwai 		.close = via_playback_multi_pcm_close,
13730aa62aefSHarald Welte 		.prepare = via_playback_multi_pcm_prepare,
13740aa62aefSHarald Welte 		.cleanup = via_playback_multi_pcm_cleanup
1375c577b8a1SJoseph Chan 	},
1376c577b8a1SJoseph Chan };
1377c577b8a1SJoseph Chan 
13787eb56e84STakashi Iwai static const struct hda_pcm_stream via_pcm_hp_playback = {
13797eb56e84STakashi Iwai 	.substreams = 1,
13807eb56e84STakashi Iwai 	.channels_min = 2,
13817eb56e84STakashi Iwai 	.channels_max = 2,
13827eb56e84STakashi Iwai 	/* NID is set in via_build_pcms */
13837eb56e84STakashi Iwai 	.ops = {
13847eb56e84STakashi Iwai 		.open = via_playback_hp_pcm_open,
1385ece8d043STakashi Iwai 		.close = via_playback_hp_pcm_close,
13867eb56e84STakashi Iwai 		.prepare = via_playback_hp_pcm_prepare,
13877eb56e84STakashi Iwai 		.cleanup = via_playback_hp_pcm_cleanup
13887eb56e84STakashi Iwai 	},
13897eb56e84STakashi Iwai };
13907eb56e84STakashi Iwai 
139190dd48a1STakashi Iwai static const struct hda_pcm_stream vt1708_pcm_analog_s16_playback = {
13927eb56e84STakashi Iwai 	.substreams = 1,
1393bc9b5623STakashi Iwai 	.channels_min = 2,
1394bc9b5623STakashi Iwai 	.channels_max = 8,
13959af74210STakashi Iwai 	/* NID is set in via_build_pcms */
1396bc9b5623STakashi Iwai 	/* We got noisy outputs on the right channel on VT1708 when
1397bc9b5623STakashi Iwai 	 * 24bit samples are used.  Until any workaround is found,
1398bc9b5623STakashi Iwai 	 * disable the 24bit format, so far.
1399bc9b5623STakashi Iwai 	 */
1400bc9b5623STakashi Iwai 	.formats = SNDRV_PCM_FMTBIT_S16_LE,
1401bc9b5623STakashi Iwai 	.ops = {
1402ece8d043STakashi Iwai 		.open = via_playback_multi_pcm_open,
1403ece8d043STakashi Iwai 		.close = via_playback_multi_pcm_close,
1404c873cc25SLydia Wang 		.prepare = via_playback_multi_pcm_prepare,
1405c873cc25SLydia Wang 		.cleanup = via_playback_multi_pcm_cleanup
1406bc9b5623STakashi Iwai 	},
1407bc9b5623STakashi Iwai };
1408bc9b5623STakashi Iwai 
14099af74210STakashi Iwai static const struct hda_pcm_stream via_pcm_analog_capture = {
14107eb56e84STakashi Iwai 	.substreams = 1, /* will be changed in via_build_pcms() */
1411c577b8a1SJoseph Chan 	.channels_min = 2,
1412c577b8a1SJoseph Chan 	.channels_max = 2,
14139af74210STakashi Iwai 	/* NID is set in via_build_pcms */
1414c577b8a1SJoseph Chan 	.ops = {
1415c577b8a1SJoseph Chan 		.prepare = via_capture_pcm_prepare,
1416c577b8a1SJoseph Chan 		.cleanup = via_capture_pcm_cleanup
1417c577b8a1SJoseph Chan 	},
1418c577b8a1SJoseph Chan };
1419c577b8a1SJoseph Chan 
1420a86a88eaSTakashi Iwai static const struct hda_pcm_stream via_pcm_dyn_adc_analog_capture = {
1421a86a88eaSTakashi Iwai 	.substreams = 1,
1422a86a88eaSTakashi Iwai 	.channels_min = 2,
1423a86a88eaSTakashi Iwai 	.channels_max = 2,
1424a86a88eaSTakashi Iwai 	/* NID is set in via_build_pcms */
1425a86a88eaSTakashi Iwai 	.ops = {
1426a86a88eaSTakashi Iwai 		.prepare = via_dyn_adc_capture_pcm_prepare,
1427a86a88eaSTakashi Iwai 		.cleanup = via_dyn_adc_capture_pcm_cleanup,
1428a86a88eaSTakashi Iwai 	},
1429a86a88eaSTakashi Iwai };
1430a86a88eaSTakashi Iwai 
14319af74210STakashi Iwai static const struct hda_pcm_stream via_pcm_digital_playback = {
1432c577b8a1SJoseph Chan 	.substreams = 1,
1433c577b8a1SJoseph Chan 	.channels_min = 2,
1434c577b8a1SJoseph Chan 	.channels_max = 2,
1435c577b8a1SJoseph Chan 	/* NID is set in via_build_pcms */
1436c577b8a1SJoseph Chan 	.ops = {
1437c577b8a1SJoseph Chan 		.open = via_dig_playback_pcm_open,
14386b97eb45STakashi Iwai 		.close = via_dig_playback_pcm_close,
14399da29271STakashi Iwai 		.prepare = via_dig_playback_pcm_prepare,
14409da29271STakashi Iwai 		.cleanup = via_dig_playback_pcm_cleanup
1441c577b8a1SJoseph Chan 	},
1442c577b8a1SJoseph Chan };
1443c577b8a1SJoseph Chan 
14449af74210STakashi Iwai static const struct hda_pcm_stream via_pcm_digital_capture = {
1445c577b8a1SJoseph Chan 	.substreams = 1,
1446c577b8a1SJoseph Chan 	.channels_min = 2,
1447c577b8a1SJoseph Chan 	.channels_max = 2,
1448c577b8a1SJoseph Chan };
1449c577b8a1SJoseph Chan 
1450370bafbdSTakashi Iwai /*
1451370bafbdSTakashi Iwai  * slave controls for virtual master
1452370bafbdSTakashi Iwai  */
14539322ca54STakashi Iwai static const char * const via_slave_pfxs[] = {
14549322ca54STakashi Iwai 	"Front", "Surround", "Center", "LFE", "Side",
14559322ca54STakashi Iwai 	"Headphone", "Speaker",
1456370bafbdSTakashi Iwai 	NULL,
1457370bafbdSTakashi Iwai };
1458370bafbdSTakashi Iwai 
1459c577b8a1SJoseph Chan static int via_build_controls(struct hda_codec *codec)
1460c577b8a1SJoseph Chan {
1461c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
14625b0cb1d8SJaroslav Kysela 	struct snd_kcontrol *kctl;
14635b0cb1d8SJaroslav Kysela 	int err, i;
1464c577b8a1SJoseph Chan 
1465b5bcc189STakashi Iwai 	spec->no_pin_power_ctl = 1;
146624088a58STakashi Iwai 	if (spec->set_widgets_power_state)
146724088a58STakashi Iwai 		if (!via_clone_control(spec, &via_pin_power_ctl_enum))
146824088a58STakashi Iwai 			return -ENOMEM;
146924088a58STakashi Iwai 
1470c577b8a1SJoseph Chan 	for (i = 0; i < spec->num_mixers; i++) {
1471c577b8a1SJoseph Chan 		err = snd_hda_add_new_ctls(codec, spec->mixers[i]);
1472c577b8a1SJoseph Chan 		if (err < 0)
1473c577b8a1SJoseph Chan 			return err;
1474c577b8a1SJoseph Chan 	}
1475c577b8a1SJoseph Chan 
1476c577b8a1SJoseph Chan 	if (spec->multiout.dig_out_nid) {
1477c577b8a1SJoseph Chan 		err = snd_hda_create_spdif_out_ctls(codec,
147874b654c9SStephen Warren 						    spec->multiout.dig_out_nid,
1479c577b8a1SJoseph Chan 						    spec->multiout.dig_out_nid);
1480c577b8a1SJoseph Chan 		if (err < 0)
1481c577b8a1SJoseph Chan 			return err;
14829a08160bSTakashi Iwai 		err = snd_hda_create_spdif_share_sw(codec,
14839a08160bSTakashi Iwai 						    &spec->multiout);
14849a08160bSTakashi Iwai 		if (err < 0)
14859a08160bSTakashi Iwai 			return err;
14869a08160bSTakashi Iwai 		spec->multiout.share_spdif = 1;
1487c577b8a1SJoseph Chan 	}
1488c577b8a1SJoseph Chan 	if (spec->dig_in_nid) {
1489c577b8a1SJoseph Chan 		err = snd_hda_create_spdif_in_ctls(codec, spec->dig_in_nid);
1490c577b8a1SJoseph Chan 		if (err < 0)
1491c577b8a1SJoseph Chan 			return err;
1492c577b8a1SJoseph Chan 	}
149317314379SLydia Wang 
1494370bafbdSTakashi Iwai 	/* if we have no master control, let's create it */
1495370bafbdSTakashi Iwai 	if (!snd_hda_find_mixer_ctl(codec, "Master Playback Volume")) {
1496370bafbdSTakashi Iwai 		unsigned int vmaster_tlv[4];
1497370bafbdSTakashi Iwai 		snd_hda_set_vmaster_tlv(codec, spec->multiout.dac_nids[0],
1498370bafbdSTakashi Iwai 					HDA_OUTPUT, vmaster_tlv);
1499370bafbdSTakashi Iwai 		err = snd_hda_add_vmaster(codec, "Master Playback Volume",
15009322ca54STakashi Iwai 					  vmaster_tlv, via_slave_pfxs,
15019322ca54STakashi Iwai 					  "Playback Volume");
1502370bafbdSTakashi Iwai 		if (err < 0)
1503370bafbdSTakashi Iwai 			return err;
1504370bafbdSTakashi Iwai 	}
1505370bafbdSTakashi Iwai 	if (!snd_hda_find_mixer_ctl(codec, "Master Playback Switch")) {
1506370bafbdSTakashi Iwai 		err = snd_hda_add_vmaster(codec, "Master Playback Switch",
15079322ca54STakashi Iwai 					  NULL, via_slave_pfxs,
15089322ca54STakashi Iwai 					  "Playback Switch");
1509370bafbdSTakashi Iwai 		if (err < 0)
1510370bafbdSTakashi Iwai 			return err;
1511370bafbdSTakashi Iwai 	}
1512370bafbdSTakashi Iwai 
15135b0cb1d8SJaroslav Kysela 	/* assign Capture Source enums to NID */
15145b0cb1d8SJaroslav Kysela 	kctl = snd_hda_find_mixer_ctl(codec, "Input Source");
15155b0cb1d8SJaroslav Kysela 	for (i = 0; kctl && i < kctl->count; i++) {
151677e314f7STakashi Iwai 		if (!spec->mux_nids[i])
151777e314f7STakashi Iwai 			continue;
151821949f00STakashi Iwai 		err = snd_hda_add_nid(codec, kctl, i, spec->mux_nids[i]);
15195b0cb1d8SJaroslav Kysela 		if (err < 0)
15205b0cb1d8SJaroslav Kysela 			return err;
15215b0cb1d8SJaroslav Kysela 	}
15225b0cb1d8SJaroslav Kysela 
1523603c4019STakashi Iwai 	via_free_kctls(codec); /* no longer needed */
152401a61e12STakashi Iwai 
152501a61e12STakashi Iwai 	err = snd_hda_jack_add_kctls(codec, &spec->autocfg);
152601a61e12STakashi Iwai 	if (err < 0)
152701a61e12STakashi Iwai 		return err;
152801a61e12STakashi Iwai 
1529c577b8a1SJoseph Chan 	return 0;
1530c577b8a1SJoseph Chan }
1531c577b8a1SJoseph Chan 
1532c577b8a1SJoseph Chan static int via_build_pcms(struct hda_codec *codec)
1533c577b8a1SJoseph Chan {
1534c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
1535c577b8a1SJoseph Chan 	struct hda_pcm *info = spec->pcm_rec;
1536c577b8a1SJoseph Chan 
1537a5973103STakashi Iwai 	codec->num_pcms = 0;
1538c577b8a1SJoseph Chan 	codec->pcm_info = info;
1539c577b8a1SJoseph Chan 
1540a5973103STakashi Iwai 	if (spec->multiout.num_dacs || spec->num_adc_nids) {
1541a5973103STakashi Iwai 		snprintf(spec->stream_name_analog,
1542a5973103STakashi Iwai 			 sizeof(spec->stream_name_analog),
154382673bc8STakashi Iwai 			 "%s Analog", codec->chip_name);
1544c577b8a1SJoseph Chan 		info->name = spec->stream_name_analog;
15459af74210STakashi Iwai 
1546a5973103STakashi Iwai 		if (spec->multiout.num_dacs) {
15479af74210STakashi Iwai 			if (!spec->stream_analog_playback)
1548a5973103STakashi Iwai 				spec->stream_analog_playback =
1549a5973103STakashi Iwai 					&via_pcm_analog_playback;
1550377ff31aSLydia Wang 			info->stream[SNDRV_PCM_STREAM_PLAYBACK] =
15519af74210STakashi Iwai 				*spec->stream_analog_playback;
1552377ff31aSLydia Wang 			info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid =
1553377ff31aSLydia Wang 				spec->multiout.dac_nids[0];
1554c577b8a1SJoseph Chan 			info->stream[SNDRV_PCM_STREAM_PLAYBACK].channels_max =
1555c577b8a1SJoseph Chan 				spec->multiout.max_channels;
1556a5973103STakashi Iwai 		}
15579af74210STakashi Iwai 
1558a86a88eaSTakashi Iwai 		if (!spec->stream_analog_capture) {
1559a86a88eaSTakashi Iwai 			if (spec->dyn_adc_switch)
1560a86a88eaSTakashi Iwai 				spec->stream_analog_capture =
1561a86a88eaSTakashi Iwai 					&via_pcm_dyn_adc_analog_capture;
1562a86a88eaSTakashi Iwai 			else
1563a5973103STakashi Iwai 				spec->stream_analog_capture =
1564a5973103STakashi Iwai 					&via_pcm_analog_capture;
1565a86a88eaSTakashi Iwai 		}
1566a5973103STakashi Iwai 		if (spec->num_adc_nids) {
15679af74210STakashi Iwai 			info->stream[SNDRV_PCM_STREAM_CAPTURE] =
15689af74210STakashi Iwai 				*spec->stream_analog_capture;
1569a5973103STakashi Iwai 			info->stream[SNDRV_PCM_STREAM_CAPTURE].nid =
1570a5973103STakashi Iwai 				spec->adc_nids[0];
1571a86a88eaSTakashi Iwai 			if (!spec->dyn_adc_switch)
15729af74210STakashi Iwai 				info->stream[SNDRV_PCM_STREAM_CAPTURE].substreams =
15739af74210STakashi Iwai 					spec->num_adc_nids;
1574a5973103STakashi Iwai 		}
1575c577b8a1SJoseph Chan 		codec->num_pcms++;
1576c577b8a1SJoseph Chan 		info++;
1577a5973103STakashi Iwai 	}
1578a5973103STakashi Iwai 
1579a5973103STakashi Iwai 	if (spec->multiout.dig_out_nid || spec->dig_in_nid) {
158082673bc8STakashi Iwai 		snprintf(spec->stream_name_digital,
158182673bc8STakashi Iwai 			 sizeof(spec->stream_name_digital),
158282673bc8STakashi Iwai 			 "%s Digital", codec->chip_name);
1583c577b8a1SJoseph Chan 		info->name = spec->stream_name_digital;
15847ba72ba1STakashi Iwai 		info->pcm_type = HDA_PCM_TYPE_SPDIF;
1585c577b8a1SJoseph Chan 		if (spec->multiout.dig_out_nid) {
15869af74210STakashi Iwai 			if (!spec->stream_digital_playback)
15879af74210STakashi Iwai 				spec->stream_digital_playback =
15889af74210STakashi Iwai 					&via_pcm_digital_playback;
1589c577b8a1SJoseph Chan 			info->stream[SNDRV_PCM_STREAM_PLAYBACK] =
15909af74210STakashi Iwai 				*spec->stream_digital_playback;
1591c577b8a1SJoseph Chan 			info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid =
1592c577b8a1SJoseph Chan 				spec->multiout.dig_out_nid;
1593c577b8a1SJoseph Chan 		}
1594c577b8a1SJoseph Chan 		if (spec->dig_in_nid) {
15959af74210STakashi Iwai 			if (!spec->stream_digital_capture)
15969af74210STakashi Iwai 				spec->stream_digital_capture =
15979af74210STakashi Iwai 					&via_pcm_digital_capture;
1598c577b8a1SJoseph Chan 			info->stream[SNDRV_PCM_STREAM_CAPTURE] =
15999af74210STakashi Iwai 				*spec->stream_digital_capture;
1600c577b8a1SJoseph Chan 			info->stream[SNDRV_PCM_STREAM_CAPTURE].nid =
1601c577b8a1SJoseph Chan 				spec->dig_in_nid;
1602c577b8a1SJoseph Chan 		}
1603a5973103STakashi Iwai 		codec->num_pcms++;
1604a5973103STakashi Iwai 		info++;
1605c577b8a1SJoseph Chan 	}
1606c577b8a1SJoseph Chan 
1607ece8d043STakashi Iwai 	if (spec->hp_dac_nid) {
16087eb56e84STakashi Iwai 		snprintf(spec->stream_name_hp, sizeof(spec->stream_name_hp),
16097eb56e84STakashi Iwai 			 "%s HP", codec->chip_name);
16107eb56e84STakashi Iwai 		info->name = spec->stream_name_hp;
16117eb56e84STakashi Iwai 		info->stream[SNDRV_PCM_STREAM_PLAYBACK] = via_pcm_hp_playback;
16127eb56e84STakashi Iwai 		info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid =
1613ece8d043STakashi Iwai 			spec->hp_dac_nid;
1614a5973103STakashi Iwai 		codec->num_pcms++;
1615a5973103STakashi Iwai 		info++;
16167eb56e84STakashi Iwai 	}
1617c577b8a1SJoseph Chan 	return 0;
1618c577b8a1SJoseph Chan }
1619c577b8a1SJoseph Chan 
1620c577b8a1SJoseph Chan static void via_free(struct hda_codec *codec)
1621c577b8a1SJoseph Chan {
1622c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
1623c577b8a1SJoseph Chan 
1624c577b8a1SJoseph Chan 	if (!spec)
1625c577b8a1SJoseph Chan 		return;
1626c577b8a1SJoseph Chan 
1627603c4019STakashi Iwai 	via_free_kctls(codec);
16281f2e99feSLydia Wang 	vt1708_stop_hp_work(spec);
1629a86a88eaSTakashi Iwai 	kfree(spec->bind_cap_vol);
1630a86a88eaSTakashi Iwai 	kfree(spec->bind_cap_sw);
1631a86a88eaSTakashi Iwai 	kfree(spec);
1632c577b8a1SJoseph Chan }
1633c577b8a1SJoseph Chan 
163464be285bSTakashi Iwai /* mute/unmute outputs */
163564be285bSTakashi Iwai static void toggle_output_mutes(struct hda_codec *codec, int num_pins,
163664be285bSTakashi Iwai 				hda_nid_t *pins, bool mute)
163764be285bSTakashi Iwai {
163864be285bSTakashi Iwai 	int i;
163994994734STakashi Iwai 	for (i = 0; i < num_pins; i++) {
164094994734STakashi Iwai 		unsigned int parm = snd_hda_codec_read(codec, pins[i], 0,
164194994734STakashi Iwai 					  AC_VERB_GET_PIN_WIDGET_CONTROL, 0);
164294994734STakashi Iwai 		if (parm & AC_PINCTL_IN_EN)
164394994734STakashi Iwai 			continue;
164494994734STakashi Iwai 		if (mute)
164594994734STakashi Iwai 			parm &= ~AC_PINCTL_OUT_EN;
164694994734STakashi Iwai 		else
164794994734STakashi Iwai 			parm |= AC_PINCTL_OUT_EN;
1648cdd03cedSTakashi Iwai 		snd_hda_set_pin_ctl(codec, pins[i], parm);
164994994734STakashi Iwai 	}
165064be285bSTakashi Iwai }
165164be285bSTakashi Iwai 
16524a918ffeSTakashi Iwai /* mute internal speaker if line-out is plugged */
16534a918ffeSTakashi Iwai static void via_line_automute(struct hda_codec *codec, int present)
16544a918ffeSTakashi Iwai {
16554a918ffeSTakashi Iwai 	struct via_spec *spec = codec->spec;
16564a918ffeSTakashi Iwai 
16574a918ffeSTakashi Iwai 	if (!spec->autocfg.speaker_outs)
16584a918ffeSTakashi Iwai 		return;
16594a918ffeSTakashi Iwai 	if (!present)
16604a918ffeSTakashi Iwai 		present = snd_hda_jack_detect(codec,
16614a918ffeSTakashi Iwai 					      spec->autocfg.line_out_pins[0]);
16624a918ffeSTakashi Iwai 	toggle_output_mutes(codec, spec->autocfg.speaker_outs,
16634a918ffeSTakashi Iwai 			    spec->autocfg.speaker_pins,
16644a918ffeSTakashi Iwai 			    present);
16654a918ffeSTakashi Iwai }
16664a918ffeSTakashi Iwai 
166769e52a80SHarald Welte /* mute internal speaker if HP is plugged */
166869e52a80SHarald Welte static void via_hp_automute(struct hda_codec *codec)
166969e52a80SHarald Welte {
16704a918ffeSTakashi Iwai 	int present = 0;
16716e969d91STakashi Iwai 	int nums;
167269e52a80SHarald Welte 	struct via_spec *spec = codec->spec;
167369e52a80SHarald Welte 
1674187d333eSTakashi Iwai 	if (!spec->hp_independent_mode && spec->autocfg.hp_pins[0] &&
1675*cf55e904SHerton Ronaldo Krzesinski 	    (spec->codec_type != VT1708 || spec->vt1708_jack_detect) &&
1676*cf55e904SHerton Ronaldo Krzesinski 	    is_jack_detectable(codec, spec->autocfg.hp_pins[0]))
1677d56757abSTakashi Iwai 		present = snd_hda_jack_detect(codec, spec->autocfg.hp_pins[0]);
16786e969d91STakashi Iwai 
1679f2b1c9f0STakashi Iwai 	if (spec->smart51_enabled)
1680f2b1c9f0STakashi Iwai 		nums = spec->autocfg.line_outs + spec->smart51_nums;
1681f2b1c9f0STakashi Iwai 	else
1682f2b1c9f0STakashi Iwai 		nums = spec->autocfg.line_outs;
16836e969d91STakashi Iwai 	toggle_output_mutes(codec, nums, spec->autocfg.line_out_pins, present);
16846e969d91STakashi Iwai 
16854a918ffeSTakashi Iwai 	via_line_automute(codec, present);
1686f3db423dSLydia Wang }
1687f3db423dSLydia Wang 
168869e52a80SHarald Welte static void via_gpio_control(struct hda_codec *codec)
168969e52a80SHarald Welte {
169069e52a80SHarald Welte 	unsigned int gpio_data;
169169e52a80SHarald Welte 	unsigned int vol_counter;
169269e52a80SHarald Welte 	unsigned int vol;
169369e52a80SHarald Welte 	unsigned int master_vol;
169469e52a80SHarald Welte 
169569e52a80SHarald Welte 	struct via_spec *spec = codec->spec;
169669e52a80SHarald Welte 
169769e52a80SHarald Welte 	gpio_data = snd_hda_codec_read(codec, codec->afg, 0,
169869e52a80SHarald Welte 				       AC_VERB_GET_GPIO_DATA, 0) & 0x03;
169969e52a80SHarald Welte 
170069e52a80SHarald Welte 	vol_counter = (snd_hda_codec_read(codec, codec->afg, 0,
170169e52a80SHarald Welte 					  0xF84, 0) & 0x3F0000) >> 16;
170269e52a80SHarald Welte 
170369e52a80SHarald Welte 	vol = vol_counter & 0x1F;
170469e52a80SHarald Welte 	master_vol = snd_hda_codec_read(codec, 0x1A, 0,
170569e52a80SHarald Welte 					AC_VERB_GET_AMP_GAIN_MUTE,
170669e52a80SHarald Welte 					AC_AMP_GET_INPUT);
170769e52a80SHarald Welte 
170869e52a80SHarald Welte 	if (gpio_data == 0x02) {
170969e52a80SHarald Welte 		/* unmute line out */
1710cdd03cedSTakashi Iwai 		snd_hda_set_pin_ctl(codec, spec->autocfg.line_out_pins[0],
17113e0693e2STakashi Iwai 				    PIN_OUT);
171269e52a80SHarald Welte 		if (vol_counter & 0x20) {
171369e52a80SHarald Welte 			/* decrease volume */
171469e52a80SHarald Welte 			if (vol > master_vol)
171569e52a80SHarald Welte 				vol = master_vol;
171669e52a80SHarald Welte 			snd_hda_codec_amp_stereo(codec, 0x1A, HDA_INPUT,
171769e52a80SHarald Welte 						 0, HDA_AMP_VOLMASK,
171869e52a80SHarald Welte 						 master_vol-vol);
171969e52a80SHarald Welte 		} else {
172069e52a80SHarald Welte 			/* increase volume */
172169e52a80SHarald Welte 			snd_hda_codec_amp_stereo(codec, 0x1A, HDA_INPUT, 0,
172269e52a80SHarald Welte 					 HDA_AMP_VOLMASK,
172369e52a80SHarald Welte 					 ((master_vol+vol) > 0x2A) ? 0x2A :
172469e52a80SHarald Welte 					  (master_vol+vol));
172569e52a80SHarald Welte 		}
172669e52a80SHarald Welte 	} else if (!(gpio_data & 0x02)) {
172769e52a80SHarald Welte 		/* mute line out */
1728cdd03cedSTakashi Iwai 		snd_hda_set_pin_ctl(codec, spec->autocfg.line_out_pins[0], 0);
172969e52a80SHarald Welte 	}
173069e52a80SHarald Welte }
173169e52a80SHarald Welte 
173269e52a80SHarald Welte /* unsolicited event for jack sensing */
173369e52a80SHarald Welte static void via_unsol_event(struct hda_codec *codec,
173469e52a80SHarald Welte 				  unsigned int res)
173569e52a80SHarald Welte {
173669e52a80SHarald Welte 	res >>= 26;
17373a93897eSTakashi Iwai 	res = snd_hda_jack_get_action(codec, res);
1738ec7e7e42SLydia Wang 
1739a34df19aSLydia Wang 	if (res & VIA_JACK_EVENT)
17403e95b9abSLydia Wang 		set_widgets_power_state(codec);
1741ec7e7e42SLydia Wang 
1742ec7e7e42SLydia Wang 	res &= ~VIA_JACK_EVENT;
1743ec7e7e42SLydia Wang 
174421ce0b65STakashi Iwai 	if (res == VIA_HP_EVENT || res == VIA_LINE_EVENT)
1745ec7e7e42SLydia Wang 		via_hp_automute(codec);
1746ec7e7e42SLydia Wang 	else if (res == VIA_GPIO_EVENT)
1747ec7e7e42SLydia Wang 		via_gpio_control(codec);
174801a61e12STakashi Iwai 	snd_hda_jack_report_sync(codec);
174969e52a80SHarald Welte }
175069e52a80SHarald Welte 
17512a43952aSTakashi Iwai #ifdef CONFIG_PM
175268cb2b55STakashi Iwai static int via_suspend(struct hda_codec *codec)
17531f2e99feSLydia Wang {
17541f2e99feSLydia Wang 	struct via_spec *spec = codec->spec;
17551f2e99feSLydia Wang 	vt1708_stop_hp_work(spec);
175694c142a1SDavid Henningsson 
175794c142a1SDavid Henningsson 	if (spec->codec_type == VT1802) {
175894c142a1SDavid Henningsson 		/* Fix pop noise on headphones */
175994c142a1SDavid Henningsson 		int i;
176094c142a1SDavid Henningsson 		for (i = 0; i < spec->autocfg.hp_outs; i++)
176194c142a1SDavid Henningsson 			snd_hda_set_pin_ctl(codec, spec->autocfg.hp_pins[i], 0);
176294c142a1SDavid Henningsson 	}
176394c142a1SDavid Henningsson 
17641f2e99feSLydia Wang 	return 0;
17651f2e99feSLydia Wang }
17661f2e99feSLydia Wang #endif
17671f2e99feSLydia Wang 
176883012a7cSTakashi Iwai #ifdef CONFIG_PM
1769cb53c626STakashi Iwai static int via_check_power_status(struct hda_codec *codec, hda_nid_t nid)
1770cb53c626STakashi Iwai {
1771cb53c626STakashi Iwai 	struct via_spec *spec = codec->spec;
1772cb53c626STakashi Iwai 	return snd_hda_check_amp_list_power(codec, &spec->loopback, nid);
1773cb53c626STakashi Iwai }
1774cb53c626STakashi Iwai #endif
1775cb53c626STakashi Iwai 
1776c577b8a1SJoseph Chan /*
1777c577b8a1SJoseph Chan  */
17785d41762aSTakashi Iwai 
17795d41762aSTakashi Iwai static int via_init(struct hda_codec *codec);
17805d41762aSTakashi Iwai 
178190dd48a1STakashi Iwai static const struct hda_codec_ops via_patch_ops = {
1782c577b8a1SJoseph Chan 	.build_controls = via_build_controls,
1783c577b8a1SJoseph Chan 	.build_pcms = via_build_pcms,
1784c577b8a1SJoseph Chan 	.init = via_init,
1785c577b8a1SJoseph Chan 	.free = via_free,
17864a918ffeSTakashi Iwai 	.unsol_event = via_unsol_event,
17872a43952aSTakashi Iwai #ifdef CONFIG_PM
17881f2e99feSLydia Wang 	.suspend = via_suspend,
1789cb53c626STakashi Iwai 	.check_power_status = via_check_power_status,
1790cb53c626STakashi Iwai #endif
1791c577b8a1SJoseph Chan };
1792c577b8a1SJoseph Chan 
17934a79616dSTakashi Iwai static bool is_empty_dac(struct hda_codec *codec, hda_nid_t dac)
1794c577b8a1SJoseph Chan {
17954a79616dSTakashi Iwai 	struct via_spec *spec = codec->spec;
17964a79616dSTakashi Iwai 	int i;
17974a79616dSTakashi Iwai 
17984a79616dSTakashi Iwai 	for (i = 0; i < spec->multiout.num_dacs; i++) {
17994a79616dSTakashi Iwai 		if (spec->multiout.dac_nids[i] == dac)
18004a79616dSTakashi Iwai 			return false;
18014a79616dSTakashi Iwai 	}
1802ece8d043STakashi Iwai 	if (spec->hp_dac_nid == dac)
18034a79616dSTakashi Iwai 		return false;
18044a79616dSTakashi Iwai 	return true;
18054a79616dSTakashi Iwai }
18064a79616dSTakashi Iwai 
18078e3679dcSTakashi Iwai static bool __parse_output_path(struct hda_codec *codec, hda_nid_t nid,
18083214b966STakashi Iwai 				hda_nid_t target_dac, int with_aa_mix,
18093214b966STakashi Iwai 				struct nid_path *path, int depth)
18104a79616dSTakashi Iwai {
18113214b966STakashi Iwai 	struct via_spec *spec = codec->spec;
18124a79616dSTakashi Iwai 	hda_nid_t conn[8];
18134a79616dSTakashi Iwai 	int i, nums;
18144a79616dSTakashi Iwai 
18153214b966STakashi Iwai 	if (nid == spec->aa_mix_nid) {
18163214b966STakashi Iwai 		if (!with_aa_mix)
18173214b966STakashi Iwai 			return false;
18183214b966STakashi Iwai 		with_aa_mix = 2; /* mark aa-mix is included */
18193214b966STakashi Iwai 	}
18203214b966STakashi Iwai 
18214a79616dSTakashi Iwai 	nums = snd_hda_get_connections(codec, nid, conn, ARRAY_SIZE(conn));
18224a79616dSTakashi Iwai 	for (i = 0; i < nums; i++) {
18234a79616dSTakashi Iwai 		if (get_wcaps_type(get_wcaps(codec, conn[i])) != AC_WID_AUD_OUT)
18244a79616dSTakashi Iwai 			continue;
18253214b966STakashi Iwai 		if (conn[i] == target_dac || is_empty_dac(codec, conn[i])) {
18263214b966STakashi Iwai 			/* aa-mix is requested but not included? */
18273214b966STakashi Iwai 			if (!(spec->aa_mix_nid && with_aa_mix == 1))
182809a9ad69STakashi Iwai 				goto found;
18294a79616dSTakashi Iwai 		}
18303214b966STakashi Iwai 	}
18318e3679dcSTakashi Iwai 	if (depth >= MAX_NID_PATH_DEPTH)
18324a79616dSTakashi Iwai 		return false;
18334a79616dSTakashi Iwai 	for (i = 0; i < nums; i++) {
18344a79616dSTakashi Iwai 		unsigned int type;
18354a79616dSTakashi Iwai 		type = get_wcaps_type(get_wcaps(codec, conn[i]));
18363214b966STakashi Iwai 		if (type == AC_WID_AUD_OUT)
18374a79616dSTakashi Iwai 			continue;
18388e3679dcSTakashi Iwai 		if (__parse_output_path(codec, conn[i], target_dac,
18393214b966STakashi Iwai 					with_aa_mix, path, depth + 1))
184009a9ad69STakashi Iwai 			goto found;
18414a79616dSTakashi Iwai 	}
18424a79616dSTakashi Iwai 	return false;
184309a9ad69STakashi Iwai 
184409a9ad69STakashi Iwai  found:
184509a9ad69STakashi Iwai 	path->path[path->depth] = conn[i];
184609a9ad69STakashi Iwai 	path->idx[path->depth] = i;
184709a9ad69STakashi Iwai 	if (nums > 1 && get_wcaps_type(get_wcaps(codec, nid)) != AC_WID_AUD_MIX)
184809a9ad69STakashi Iwai 		path->multi[path->depth] = 1;
184909a9ad69STakashi Iwai 	path->depth++;
185009a9ad69STakashi Iwai 	return true;
18514a79616dSTakashi Iwai }
18524a79616dSTakashi Iwai 
18538e3679dcSTakashi Iwai static bool parse_output_path(struct hda_codec *codec, hda_nid_t nid,
18543214b966STakashi Iwai 			      hda_nid_t target_dac, int with_aa_mix,
18553214b966STakashi Iwai 			      struct nid_path *path)
18568e3679dcSTakashi Iwai {
18573214b966STakashi Iwai 	if (__parse_output_path(codec, nid, target_dac, with_aa_mix, path, 1)) {
18588e3679dcSTakashi Iwai 		path->path[path->depth] = nid;
18598e3679dcSTakashi Iwai 		path->depth++;
18603214b966STakashi Iwai 		snd_printdd("output-path: depth=%d, %02x/%02x/%02x/%02x/%02x\n",
18613214b966STakashi Iwai 			    path->depth, path->path[0], path->path[1],
18623214b966STakashi Iwai 			    path->path[2], path->path[3], path->path[4]);
18638e3679dcSTakashi Iwai 		return true;
18648e3679dcSTakashi Iwai 	}
18658e3679dcSTakashi Iwai 	return false;
18668e3679dcSTakashi Iwai }
18678e3679dcSTakashi Iwai 
18684a79616dSTakashi Iwai static int via_auto_fill_dac_nids(struct hda_codec *codec)
18694a79616dSTakashi Iwai {
18704a79616dSTakashi Iwai 	struct via_spec *spec = codec->spec;
18714a79616dSTakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
18725c9a5615SLydia Wang 	int i, dac_num;
1873c577b8a1SJoseph Chan 	hda_nid_t nid;
1874c577b8a1SJoseph Chan 
1875c577b8a1SJoseph Chan 	spec->multiout.dac_nids = spec->private_dac_nids;
18765c9a5615SLydia Wang 	dac_num = 0;
18774a79616dSTakashi Iwai 	for (i = 0; i < cfg->line_outs; i++) {
18783214b966STakashi Iwai 		hda_nid_t dac = 0;
1879c577b8a1SJoseph Chan 		nid = cfg->line_out_pins[i];
18804a79616dSTakashi Iwai 		if (!nid)
18814a79616dSTakashi Iwai 			continue;
18823214b966STakashi Iwai 		if (parse_output_path(codec, nid, 0, 0, &spec->out_path[i]))
18833214b966STakashi Iwai 			dac = spec->out_path[i].path[0];
18843214b966STakashi Iwai 		if (!i && parse_output_path(codec, nid, dac, 1,
18853214b966STakashi Iwai 					    &spec->out_mix_path))
18863214b966STakashi Iwai 			dac = spec->out_mix_path.path[0];
18873214b966STakashi Iwai 		if (dac) {
18883214b966STakashi Iwai 			spec->private_dac_nids[i] = dac;
18895c9a5615SLydia Wang 			dac_num++;
1890c577b8a1SJoseph Chan 		}
18915c9a5615SLydia Wang 	}
18923214b966STakashi Iwai 	if (!spec->out_path[0].depth && spec->out_mix_path.depth) {
18933214b966STakashi Iwai 		spec->out_path[0] = spec->out_mix_path;
18943214b966STakashi Iwai 		spec->out_mix_path.depth = 0;
18953214b966STakashi Iwai 	}
18965c9a5615SLydia Wang 	spec->multiout.num_dacs = dac_num;
1897c577b8a1SJoseph Chan 	return 0;
1898c577b8a1SJoseph Chan }
1899c577b8a1SJoseph Chan 
19004a79616dSTakashi Iwai static int create_ch_ctls(struct hda_codec *codec, const char *pfx,
190109a9ad69STakashi Iwai 			  int chs, bool check_dac, struct nid_path *path)
1902c577b8a1SJoseph Chan {
19034a79616dSTakashi Iwai 	struct via_spec *spec = codec->spec;
1904c577b8a1SJoseph Chan 	char name[32];
190509a9ad69STakashi Iwai 	hda_nid_t dac, pin, sel, nid;
190609a9ad69STakashi Iwai 	int err;
1907a934d5a9STakashi Iwai 
190809a9ad69STakashi Iwai 	dac = check_dac ? path->path[0] : 0;
190909a9ad69STakashi Iwai 	pin = path->path[path->depth - 1];
191009a9ad69STakashi Iwai 	sel = path->depth > 1 ? path->path[1] : 0;
1911c577b8a1SJoseph Chan 
19128df2a312STakashi Iwai 	if (dac && check_amp_caps(codec, dac, HDA_OUTPUT, AC_AMPCAP_NUM_STEPS))
19134a79616dSTakashi Iwai 		nid = dac;
19148df2a312STakashi Iwai 	else if (check_amp_caps(codec, pin, HDA_OUTPUT, AC_AMPCAP_NUM_STEPS))
19154a79616dSTakashi Iwai 		nid = pin;
1916a934d5a9STakashi Iwai 	else if (check_amp_caps(codec, sel, HDA_OUTPUT, AC_AMPCAP_NUM_STEPS))
1917a934d5a9STakashi Iwai 		nid = sel;
19184a79616dSTakashi Iwai 	else
19194a79616dSTakashi Iwai 		nid = 0;
19204a79616dSTakashi Iwai 	if (nid) {
19214a79616dSTakashi Iwai 		sprintf(name, "%s Playback Volume", pfx);
1922c577b8a1SJoseph Chan 		err = via_add_control(spec, VIA_CTL_WIDGET_VOL, name,
1923a00a5fadSLydia Wang 			      HDA_COMPOSE_AMP_VAL(nid, chs, 0, HDA_OUTPUT));
1924c577b8a1SJoseph Chan 		if (err < 0)
1925c577b8a1SJoseph Chan 			return err;
192609a9ad69STakashi Iwai 		path->vol_ctl = nid;
1927c577b8a1SJoseph Chan 	}
19284a79616dSTakashi Iwai 
19298df2a312STakashi Iwai 	if (dac && check_amp_caps(codec, dac, HDA_OUTPUT, AC_AMPCAP_MUTE))
19304a79616dSTakashi Iwai 		nid = dac;
19318df2a312STakashi Iwai 	else if (check_amp_caps(codec, pin, HDA_OUTPUT, AC_AMPCAP_MUTE))
19324a79616dSTakashi Iwai 		nid = pin;
1933a934d5a9STakashi Iwai 	else if (check_amp_caps(codec, sel, HDA_OUTPUT, AC_AMPCAP_MUTE))
1934a934d5a9STakashi Iwai 		nid = sel;
19354a79616dSTakashi Iwai 	else
19364a79616dSTakashi Iwai 		nid = 0;
19374a79616dSTakashi Iwai 	if (nid) {
19384a79616dSTakashi Iwai 		sprintf(name, "%s Playback Switch", pfx);
19394a79616dSTakashi Iwai 		err = via_add_control(spec, VIA_CTL_WIDGET_MUTE, name,
19404a79616dSTakashi Iwai 			      HDA_COMPOSE_AMP_VAL(nid, chs, 0, HDA_OUTPUT));
19414a79616dSTakashi Iwai 		if (err < 0)
19424a79616dSTakashi Iwai 			return err;
194309a9ad69STakashi Iwai 		path->mute_ctl = nid;
19444a79616dSTakashi Iwai 	}
19454a79616dSTakashi Iwai 	return 0;
19464a79616dSTakashi Iwai }
19474a79616dSTakashi Iwai 
1948f4a7828bSTakashi Iwai static void mangle_smart51(struct hda_codec *codec)
1949f4a7828bSTakashi Iwai {
1950f4a7828bSTakashi Iwai 	struct via_spec *spec = codec->spec;
1951f4a7828bSTakashi Iwai 	struct auto_pin_cfg *cfg = &spec->autocfg;
19520f98c24bSTakashi Iwai 	struct auto_pin_cfg_item *ins = cfg->inputs;
19530f98c24bSTakashi Iwai 	int i, j, nums, attr;
19540f98c24bSTakashi Iwai 	int pins[AUTO_CFG_MAX_INS];
1955f4a7828bSTakashi Iwai 
19560f98c24bSTakashi Iwai 	for (attr = INPUT_PIN_ATTR_REAR; attr >= INPUT_PIN_ATTR_NORMAL; attr--) {
19570f98c24bSTakashi Iwai 		nums = 0;
1958f4a7828bSTakashi Iwai 		for (i = 0; i < cfg->num_inputs; i++) {
19590f98c24bSTakashi Iwai 			unsigned int def;
19600f98c24bSTakashi Iwai 			if (ins[i].type > AUTO_PIN_LINE_IN)
19610f98c24bSTakashi Iwai 				continue;
19620f98c24bSTakashi Iwai 			def = snd_hda_codec_get_pincfg(codec, ins[i].pin);
19630f98c24bSTakashi Iwai 			if (snd_hda_get_input_pin_attr(def) != attr)
19640f98c24bSTakashi Iwai 				continue;
19650f98c24bSTakashi Iwai 			for (j = 0; j < nums; j++)
19660f98c24bSTakashi Iwai 				if (ins[pins[j]].type < ins[i].type) {
19670f98c24bSTakashi Iwai 					memmove(pins + j + 1, pins + j,
196821d45d2bSTakashi Iwai 						(nums - j) * sizeof(int));
19690f98c24bSTakashi Iwai 					break;
19700f98c24bSTakashi Iwai 				}
19710f98c24bSTakashi Iwai 			pins[j] = i;
1972e3d7a143STakashi Iwai 			nums++;
1973e3d7a143STakashi Iwai 		}
1974e3d7a143STakashi Iwai 		if (cfg->line_outs + nums < 3)
1975f4a7828bSTakashi Iwai 			continue;
19760f98c24bSTakashi Iwai 		for (i = 0; i < nums; i++) {
19770f98c24bSTakashi Iwai 			hda_nid_t pin = ins[pins[i]].pin;
19780f98c24bSTakashi Iwai 			spec->smart51_pins[spec->smart51_nums++] = pin;
19790f98c24bSTakashi Iwai 			cfg->line_out_pins[cfg->line_outs++] = pin;
1980f4a7828bSTakashi Iwai 			if (cfg->line_outs == 3)
1981f4a7828bSTakashi Iwai 				break;
1982f4a7828bSTakashi Iwai 		}
19830f98c24bSTakashi Iwai 		return;
19840f98c24bSTakashi Iwai 	}
1985f4a7828bSTakashi Iwai }
1986f4a7828bSTakashi Iwai 
1987020066d1STakashi Iwai static void copy_path_mixer_ctls(struct nid_path *dst, struct nid_path *src)
1988020066d1STakashi Iwai {
1989020066d1STakashi Iwai 	dst->vol_ctl = src->vol_ctl;
1990020066d1STakashi Iwai 	dst->mute_ctl = src->mute_ctl;
1991020066d1STakashi Iwai }
1992020066d1STakashi Iwai 
19934a79616dSTakashi Iwai /* add playback controls from the parsed DAC table */
19944a79616dSTakashi Iwai static int via_auto_create_multi_out_ctls(struct hda_codec *codec)
19954a79616dSTakashi Iwai {
19964a79616dSTakashi Iwai 	struct via_spec *spec = codec->spec;
1997f4a7828bSTakashi Iwai 	struct auto_pin_cfg *cfg = &spec->autocfg;
19983214b966STakashi Iwai 	struct nid_path *path;
19994a79616dSTakashi Iwai 	static const char * const chname[4] = {
20004a79616dSTakashi Iwai 		"Front", "Surround", "C/LFE", "Side"
20014a79616dSTakashi Iwai 	};
20024a79616dSTakashi Iwai 	int i, idx, err;
2003f4a7828bSTakashi Iwai 	int old_line_outs;
2004f4a7828bSTakashi Iwai 
2005f4a7828bSTakashi Iwai 	/* check smart51 */
2006f4a7828bSTakashi Iwai 	old_line_outs = cfg->line_outs;
2007f4a7828bSTakashi Iwai 	if (cfg->line_outs == 1)
2008f4a7828bSTakashi Iwai 		mangle_smart51(codec);
20094a79616dSTakashi Iwai 
2010e3d7a143STakashi Iwai 	err = via_auto_fill_dac_nids(codec);
2011e3d7a143STakashi Iwai 	if (err < 0)
2012e3d7a143STakashi Iwai 		return err;
2013e3d7a143STakashi Iwai 
20145c9a5615SLydia Wang 	if (spec->multiout.num_dacs < 3) {
20155c9a5615SLydia Wang 		spec->smart51_nums = 0;
20165c9a5615SLydia Wang 		cfg->line_outs = old_line_outs;
20175c9a5615SLydia Wang 	}
20184a79616dSTakashi Iwai 	for (i = 0; i < cfg->line_outs; i++) {
20194a79616dSTakashi Iwai 		hda_nid_t pin, dac;
20204a79616dSTakashi Iwai 		pin = cfg->line_out_pins[i];
20214a79616dSTakashi Iwai 		dac = spec->multiout.dac_nids[i];
20224a79616dSTakashi Iwai 		if (!pin || !dac)
20234a79616dSTakashi Iwai 			continue;
20243214b966STakashi Iwai 		path = spec->out_path + i;
20250fe0adf8STakashi Iwai 		if (i == HDA_CLFE) {
20263214b966STakashi Iwai 			err = create_ch_ctls(codec, "Center", 1, true, path);
20274a79616dSTakashi Iwai 			if (err < 0)
20284a79616dSTakashi Iwai 				return err;
20293214b966STakashi Iwai 			err = create_ch_ctls(codec, "LFE", 2, true, path);
20304a79616dSTakashi Iwai 			if (err < 0)
20314a79616dSTakashi Iwai 				return err;
20324a79616dSTakashi Iwai 		} else {
20336aadf41dSTakashi Iwai 			const char *pfx = chname[i];
20346aadf41dSTakashi Iwai 			if (cfg->line_out_type == AUTO_PIN_SPEAKER_OUT &&
20356aadf41dSTakashi Iwai 			    cfg->line_outs == 1)
20366aadf41dSTakashi Iwai 				pfx = "Speaker";
20373214b966STakashi Iwai 			err = create_ch_ctls(codec, pfx, 3, true, path);
20384a79616dSTakashi Iwai 			if (err < 0)
20394a79616dSTakashi Iwai 				return err;
20404a79616dSTakashi Iwai 		}
2041020066d1STakashi Iwai 		if (path != spec->out_path + i)
2042020066d1STakashi Iwai 			copy_path_mixer_ctls(&spec->out_path[i], path);
2043020066d1STakashi Iwai 		if (path == spec->out_path && spec->out_mix_path.depth)
2044020066d1STakashi Iwai 			copy_path_mixer_ctls(&spec->out_mix_path, path);
20454a79616dSTakashi Iwai 	}
20464a79616dSTakashi Iwai 
20474a79616dSTakashi Iwai 	idx = get_connection_index(codec, spec->aa_mix_nid,
20484a79616dSTakashi Iwai 				   spec->multiout.dac_nids[0]);
20494a79616dSTakashi Iwai 	if (idx >= 0) {
20504a79616dSTakashi Iwai 		/* add control to mixer */
20513214b966STakashi Iwai 		const char *name;
20523214b966STakashi Iwai 		name = spec->out_mix_path.depth ?
20533214b966STakashi Iwai 			"PCM Loopback Playback Volume" : "PCM Playback Volume";
20543214b966STakashi Iwai 		err = via_add_control(spec, VIA_CTL_WIDGET_VOL, name,
20554a79616dSTakashi Iwai 				      HDA_COMPOSE_AMP_VAL(spec->aa_mix_nid, 3,
20564a79616dSTakashi Iwai 							  idx, HDA_INPUT));
20574a79616dSTakashi Iwai 		if (err < 0)
20584a79616dSTakashi Iwai 			return err;
20593214b966STakashi Iwai 		name = spec->out_mix_path.depth ?
20603214b966STakashi Iwai 			"PCM Loopback Playback Switch" : "PCM Playback Switch";
20613214b966STakashi Iwai 		err = via_add_control(spec, VIA_CTL_WIDGET_MUTE, name,
20624a79616dSTakashi Iwai 				      HDA_COMPOSE_AMP_VAL(spec->aa_mix_nid, 3,
20634a79616dSTakashi Iwai 							  idx, HDA_INPUT));
20644a79616dSTakashi Iwai 		if (err < 0)
20654a79616dSTakashi Iwai 			return err;
2066c577b8a1SJoseph Chan 	}
2067c577b8a1SJoseph Chan 
2068f4a7828bSTakashi Iwai 	cfg->line_outs = old_line_outs;
2069f4a7828bSTakashi Iwai 
2070c577b8a1SJoseph Chan 	return 0;
2071c577b8a1SJoseph Chan }
2072c577b8a1SJoseph Chan 
20734a79616dSTakashi Iwai static int via_auto_create_hp_ctls(struct hda_codec *codec, hda_nid_t pin)
2074c577b8a1SJoseph Chan {
20754a79616dSTakashi Iwai 	struct via_spec *spec = codec->spec;
207609a9ad69STakashi Iwai 	struct nid_path *path;
207718bd2c44STakashi Iwai 	bool check_dac;
20783214b966STakashi Iwai 	int i, err;
2079c577b8a1SJoseph Chan 
2080c577b8a1SJoseph Chan 	if (!pin)
2081c577b8a1SJoseph Chan 		return 0;
2082c577b8a1SJoseph Chan 
20833214b966STakashi Iwai 	if (!parse_output_path(codec, pin, 0, 0, &spec->hp_indep_path)) {
20843214b966STakashi Iwai 		for (i = HDA_SIDE; i >= HDA_CLFE; i--) {
20853214b966STakashi Iwai 			if (i < spec->multiout.num_dacs &&
208625250505STakashi Iwai 			    parse_output_path(codec, pin,
20873214b966STakashi Iwai 					      spec->multiout.dac_nids[i], 0,
20883214b966STakashi Iwai 					      &spec->hp_indep_path)) {
20893214b966STakashi Iwai 				spec->hp_indep_shared = i;
20903214b966STakashi Iwai 				break;
209125250505STakashi Iwai 			}
20923214b966STakashi Iwai 		}
20933214b966STakashi Iwai 	}
20943214b966STakashi Iwai 	if (spec->hp_indep_path.depth) {
20953214b966STakashi Iwai 		spec->hp_dac_nid = spec->hp_indep_path.path[0];
20963214b966STakashi Iwai 		if (!spec->hp_indep_shared)
20973214b966STakashi Iwai 			spec->hp_path = spec->hp_indep_path;
20983214b966STakashi Iwai 	}
20993214b966STakashi Iwai 	/* optionally check front-path w/o AA-mix */
21003214b966STakashi Iwai 	if (!spec->hp_path.depth)
21013214b966STakashi Iwai 		parse_output_path(codec, pin,
21023214b966STakashi Iwai 				  spec->multiout.dac_nids[HDA_FRONT], 0,
21033214b966STakashi Iwai 				  &spec->hp_path);
21044a79616dSTakashi Iwai 
2105ece8d043STakashi Iwai 	if (!parse_output_path(codec, pin, spec->multiout.dac_nids[HDA_FRONT],
21063214b966STakashi Iwai 			       1, &spec->hp_mix_path) && !spec->hp_path.depth)
2107ece8d043STakashi Iwai 		return 0;
2108ece8d043STakashi Iwai 
21093214b966STakashi Iwai 	if (spec->hp_path.depth) {
211009a9ad69STakashi Iwai 		path = &spec->hp_path;
211118bd2c44STakashi Iwai 		check_dac = true;
211218bd2c44STakashi Iwai 	} else {
21133214b966STakashi Iwai 		path = &spec->hp_mix_path;
211418bd2c44STakashi Iwai 		check_dac = false;
211518bd2c44STakashi Iwai 	}
211618bd2c44STakashi Iwai 	err = create_ch_ctls(codec, "Headphone", 3, check_dac, path);
21174a79616dSTakashi Iwai 	if (err < 0)
21184a79616dSTakashi Iwai 		return err;
2119020066d1STakashi Iwai 	if (check_dac)
2120020066d1STakashi Iwai 		copy_path_mixer_ctls(&spec->hp_mix_path, path);
2121020066d1STakashi Iwai 	else
2122020066d1STakashi Iwai 		copy_path_mixer_ctls(&spec->hp_path, path);
2123020066d1STakashi Iwai 	if (spec->hp_indep_path.depth)
2124020066d1STakashi Iwai 		copy_path_mixer_ctls(&spec->hp_indep_path, path);
2125c577b8a1SJoseph Chan 	return 0;
2126c577b8a1SJoseph Chan }
2127c577b8a1SJoseph Chan 
21284a918ffeSTakashi Iwai static int via_auto_create_speaker_ctls(struct hda_codec *codec)
21294a918ffeSTakashi Iwai {
21304a918ffeSTakashi Iwai 	struct via_spec *spec = codec->spec;
21313214b966STakashi Iwai 	struct nid_path *path;
21323214b966STakashi Iwai 	bool check_dac;
213381c0a78bSWang Shaoyan 	hda_nid_t pin, dac = 0;
21343214b966STakashi Iwai 	int err;
21354a918ffeSTakashi Iwai 
21364a918ffeSTakashi Iwai 	pin = spec->autocfg.speaker_pins[0];
21374a918ffeSTakashi Iwai 	if (!spec->autocfg.speaker_outs || !pin)
21384a918ffeSTakashi Iwai 		return 0;
21394a918ffeSTakashi Iwai 
21403214b966STakashi Iwai 	if (parse_output_path(codec, pin, 0, 0, &spec->speaker_path))
21418e3679dcSTakashi Iwai 		dac = spec->speaker_path.path[0];
21423214b966STakashi Iwai 	if (!dac)
21433214b966STakashi Iwai 		parse_output_path(codec, pin,
21443214b966STakashi Iwai 				  spec->multiout.dac_nids[HDA_FRONT], 0,
214509a9ad69STakashi Iwai 				  &spec->speaker_path);
21463214b966STakashi Iwai 	if (!parse_output_path(codec, pin, spec->multiout.dac_nids[HDA_FRONT],
21473214b966STakashi Iwai 			       1, &spec->speaker_mix_path) && !dac)
21483214b966STakashi Iwai 		return 0;
21494a918ffeSTakashi Iwai 
21503214b966STakashi Iwai 	/* no AA-path for front? */
21513214b966STakashi Iwai 	if (!spec->out_mix_path.depth && spec->speaker_mix_path.depth)
21523214b966STakashi Iwai 		dac = 0;
21533214b966STakashi Iwai 
21543214b966STakashi Iwai 	spec->speaker_dac_nid = dac;
21553214b966STakashi Iwai 	spec->multiout.extra_out_nid[0] = dac;
21563214b966STakashi Iwai 	if (dac) {
21573214b966STakashi Iwai 		path = &spec->speaker_path;
21583214b966STakashi Iwai 		check_dac = true;
21593214b966STakashi Iwai 	} else {
21603214b966STakashi Iwai 		path = &spec->speaker_mix_path;
21613214b966STakashi Iwai 		check_dac = false;
21623214b966STakashi Iwai 	}
21633214b966STakashi Iwai 	err = create_ch_ctls(codec, "Speaker", 3, check_dac, path);
21643214b966STakashi Iwai 	if (err < 0)
21653214b966STakashi Iwai 		return err;
2166020066d1STakashi Iwai 	if (check_dac)
2167020066d1STakashi Iwai 		copy_path_mixer_ctls(&spec->speaker_mix_path, path);
2168020066d1STakashi Iwai 	else
2169020066d1STakashi Iwai 		copy_path_mixer_ctls(&spec->speaker_path, path);
21703214b966STakashi Iwai 	return 0;
21713214b966STakashi Iwai }
21723214b966STakashi Iwai 
21733214b966STakashi Iwai #define via_aamix_ctl_info	via_pin_power_ctl_info
21743214b966STakashi Iwai 
21753214b966STakashi Iwai static int via_aamix_ctl_get(struct snd_kcontrol *kcontrol,
21763214b966STakashi Iwai 			     struct snd_ctl_elem_value *ucontrol)
21773214b966STakashi Iwai {
21783214b966STakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
21793214b966STakashi Iwai 	struct via_spec *spec = codec->spec;
21803214b966STakashi Iwai 	ucontrol->value.enumerated.item[0] = spec->aamix_mode;
21813214b966STakashi Iwai 	return 0;
21823214b966STakashi Iwai }
21833214b966STakashi Iwai 
21843214b966STakashi Iwai static void update_aamix_paths(struct hda_codec *codec, int do_mix,
21853214b966STakashi Iwai 			       struct nid_path *nomix, struct nid_path *mix)
21863214b966STakashi Iwai {
21873214b966STakashi Iwai 	if (do_mix) {
21883214b966STakashi Iwai 		activate_output_path(codec, nomix, false, false);
21893214b966STakashi Iwai 		activate_output_path(codec, mix, true, false);
21903214b966STakashi Iwai 	} else {
21913214b966STakashi Iwai 		activate_output_path(codec, mix, false, false);
21923214b966STakashi Iwai 		activate_output_path(codec, nomix, true, false);
21933214b966STakashi Iwai 	}
21943214b966STakashi Iwai }
21953214b966STakashi Iwai 
21963214b966STakashi Iwai static int via_aamix_ctl_put(struct snd_kcontrol *kcontrol,
21973214b966STakashi Iwai 			     struct snd_ctl_elem_value *ucontrol)
21983214b966STakashi Iwai {
21993214b966STakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
22003214b966STakashi Iwai 	struct via_spec *spec = codec->spec;
22013214b966STakashi Iwai 	unsigned int val = ucontrol->value.enumerated.item[0];
22023214b966STakashi Iwai 
22033214b966STakashi Iwai 	if (val == spec->aamix_mode)
22043214b966STakashi Iwai 		return 0;
22053214b966STakashi Iwai 	spec->aamix_mode = val;
22063214b966STakashi Iwai 	/* update front path */
22073214b966STakashi Iwai 	update_aamix_paths(codec, val, &spec->out_path[0], &spec->out_mix_path);
22083214b966STakashi Iwai 	/* update HP path */
22093214b966STakashi Iwai 	if (!spec->hp_independent_mode) {
22103214b966STakashi Iwai 		update_aamix_paths(codec, val, &spec->hp_path,
22113214b966STakashi Iwai 				   &spec->hp_mix_path);
22123214b966STakashi Iwai 	}
22133214b966STakashi Iwai 	/* update speaker path */
22143214b966STakashi Iwai 	update_aamix_paths(codec, val, &spec->speaker_path,
22153214b966STakashi Iwai 			   &spec->speaker_mix_path);
22163214b966STakashi Iwai 	return 1;
22173214b966STakashi Iwai }
22183214b966STakashi Iwai 
22193214b966STakashi Iwai static const struct snd_kcontrol_new via_aamix_ctl_enum = {
22203214b966STakashi Iwai 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
22213214b966STakashi Iwai 	.name = "Loopback Mixing",
22223214b966STakashi Iwai 	.info = via_aamix_ctl_info,
22233214b966STakashi Iwai 	.get = via_aamix_ctl_get,
22243214b966STakashi Iwai 	.put = via_aamix_ctl_put,
22253214b966STakashi Iwai };
22263214b966STakashi Iwai 
22273214b966STakashi Iwai static int via_auto_create_loopback_switch(struct hda_codec *codec)
22283214b966STakashi Iwai {
22293214b966STakashi Iwai 	struct via_spec *spec = codec->spec;
22303214b966STakashi Iwai 
22314808d12dSTakashi Iwai 	if (!spec->aa_mix_nid)
22324808d12dSTakashi Iwai 		return 0; /* no loopback switching available */
22334808d12dSTakashi Iwai 	if (!(spec->out_mix_path.depth || spec->hp_mix_path.depth ||
22344808d12dSTakashi Iwai 	      spec->speaker_path.depth))
22353214b966STakashi Iwai 		return 0; /* no loopback switching available */
22363214b966STakashi Iwai 	if (!via_clone_control(spec, &via_aamix_ctl_enum))
22373214b966STakashi Iwai 		return -ENOMEM;
22384a918ffeSTakashi Iwai 	return 0;
22394a918ffeSTakashi Iwai }
22404a918ffeSTakashi Iwai 
2241a766d0d7STakashi Iwai /* look for ADCs */
2242a766d0d7STakashi Iwai static int via_fill_adcs(struct hda_codec *codec)
2243a766d0d7STakashi Iwai {
2244a766d0d7STakashi Iwai 	struct via_spec *spec = codec->spec;
2245a766d0d7STakashi Iwai 	hda_nid_t nid = codec->start_nid;
2246a766d0d7STakashi Iwai 	int i;
2247a766d0d7STakashi Iwai 
2248a766d0d7STakashi Iwai 	for (i = 0; i < codec->num_nodes; i++, nid++) {
2249a766d0d7STakashi Iwai 		unsigned int wcaps = get_wcaps(codec, nid);
2250a766d0d7STakashi Iwai 		if (get_wcaps_type(wcaps) != AC_WID_AUD_IN)
2251a766d0d7STakashi Iwai 			continue;
2252a766d0d7STakashi Iwai 		if (wcaps & AC_WCAP_DIGITAL)
2253a766d0d7STakashi Iwai 			continue;
2254a766d0d7STakashi Iwai 		if (!(wcaps & AC_WCAP_CONN_LIST))
2255a766d0d7STakashi Iwai 			continue;
2256a766d0d7STakashi Iwai 		if (spec->num_adc_nids >= ARRAY_SIZE(spec->adc_nids))
2257a766d0d7STakashi Iwai 			return -ENOMEM;
2258a766d0d7STakashi Iwai 		spec->adc_nids[spec->num_adc_nids++] = nid;
2259a766d0d7STakashi Iwai 	}
2260a766d0d7STakashi Iwai 	return 0;
2261a766d0d7STakashi Iwai }
2262a766d0d7STakashi Iwai 
2263a86a88eaSTakashi Iwai /* input-src control */
2264a86a88eaSTakashi Iwai static int via_mux_enum_info(struct snd_kcontrol *kcontrol,
2265a86a88eaSTakashi Iwai 			     struct snd_ctl_elem_info *uinfo)
2266a86a88eaSTakashi Iwai {
2267a86a88eaSTakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2268a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2269a86a88eaSTakashi Iwai 
2270a86a88eaSTakashi Iwai 	uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
2271a86a88eaSTakashi Iwai 	uinfo->count = 1;
2272a86a88eaSTakashi Iwai 	uinfo->value.enumerated.items = spec->num_inputs;
2273a86a88eaSTakashi Iwai 	if (uinfo->value.enumerated.item >= spec->num_inputs)
2274a86a88eaSTakashi Iwai 		uinfo->value.enumerated.item = spec->num_inputs - 1;
2275a86a88eaSTakashi Iwai 	strcpy(uinfo->value.enumerated.name,
2276a86a88eaSTakashi Iwai 	       spec->inputs[uinfo->value.enumerated.item].label);
2277a86a88eaSTakashi Iwai 	return 0;
2278a86a88eaSTakashi Iwai }
2279a86a88eaSTakashi Iwai 
2280a86a88eaSTakashi Iwai static int via_mux_enum_get(struct snd_kcontrol *kcontrol,
2281a86a88eaSTakashi Iwai 			    struct snd_ctl_elem_value *ucontrol)
2282a86a88eaSTakashi Iwai {
2283a86a88eaSTakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2284a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2285a86a88eaSTakashi Iwai 	unsigned int idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
2286a86a88eaSTakashi Iwai 
2287a86a88eaSTakashi Iwai 	ucontrol->value.enumerated.item[0] = spec->cur_mux[idx];
2288a86a88eaSTakashi Iwai 	return 0;
2289a86a88eaSTakashi Iwai }
2290a86a88eaSTakashi Iwai 
2291a86a88eaSTakashi Iwai static int via_mux_enum_put(struct snd_kcontrol *kcontrol,
2292a86a88eaSTakashi Iwai 			    struct snd_ctl_elem_value *ucontrol)
2293a86a88eaSTakashi Iwai {
2294a86a88eaSTakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2295a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2296a86a88eaSTakashi Iwai 	unsigned int idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
2297a86a88eaSTakashi Iwai 	hda_nid_t mux;
2298a86a88eaSTakashi Iwai 	int cur;
2299a86a88eaSTakashi Iwai 
2300a86a88eaSTakashi Iwai 	cur = ucontrol->value.enumerated.item[0];
2301a86a88eaSTakashi Iwai 	if (cur < 0 || cur >= spec->num_inputs)
2302a86a88eaSTakashi Iwai 		return -EINVAL;
2303a86a88eaSTakashi Iwai 	if (spec->cur_mux[idx] == cur)
2304a86a88eaSTakashi Iwai 		return 0;
2305a86a88eaSTakashi Iwai 	spec->cur_mux[idx] = cur;
2306a86a88eaSTakashi Iwai 	if (spec->dyn_adc_switch) {
2307a86a88eaSTakashi Iwai 		int adc_idx = spec->inputs[cur].adc_idx;
2308a86a88eaSTakashi Iwai 		mux = spec->mux_nids[adc_idx];
2309a86a88eaSTakashi Iwai 		via_dyn_adc_pcm_resetup(codec, cur);
2310a86a88eaSTakashi Iwai 	} else {
2311a86a88eaSTakashi Iwai 		mux = spec->mux_nids[idx];
2312a86a88eaSTakashi Iwai 		if (snd_BUG_ON(!mux))
2313a86a88eaSTakashi Iwai 			return -EINVAL;
2314a86a88eaSTakashi Iwai 	}
2315a86a88eaSTakashi Iwai 
2316a86a88eaSTakashi Iwai 	if (mux) {
2317a86a88eaSTakashi Iwai 		/* switch to D0 beofre change index */
2318054d867eSTakashi Iwai 		update_power_state(codec, mux, AC_PWRST_D0);
2319a86a88eaSTakashi Iwai 		snd_hda_codec_write(codec, mux, 0,
2320a86a88eaSTakashi Iwai 				    AC_VERB_SET_CONNECT_SEL,
2321a86a88eaSTakashi Iwai 				    spec->inputs[cur].mux_idx);
2322a86a88eaSTakashi Iwai 	}
2323a86a88eaSTakashi Iwai 
2324a86a88eaSTakashi Iwai 	/* update jack power state */
2325a86a88eaSTakashi Iwai 	set_widgets_power_state(codec);
2326a86a88eaSTakashi Iwai 	return 0;
2327a86a88eaSTakashi Iwai }
2328a766d0d7STakashi Iwai 
2329d7a99cceSTakashi Iwai static const struct snd_kcontrol_new via_input_src_ctl = {
2330d7a99cceSTakashi Iwai 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2331d7a99cceSTakashi Iwai 	/* The multiple "Capture Source" controls confuse alsamixer
2332d7a99cceSTakashi Iwai 	 * So call somewhat different..
2333d7a99cceSTakashi Iwai 	 */
2334d7a99cceSTakashi Iwai 	/* .name = "Capture Source", */
2335d7a99cceSTakashi Iwai 	.name = "Input Source",
2336d7a99cceSTakashi Iwai 	.info = via_mux_enum_info,
2337d7a99cceSTakashi Iwai 	.get = via_mux_enum_get,
2338d7a99cceSTakashi Iwai 	.put = via_mux_enum_put,
2339d7a99cceSTakashi Iwai };
2340d7a99cceSTakashi Iwai 
2341a86a88eaSTakashi Iwai static int create_input_src_ctls(struct hda_codec *codec, int count)
2342a86a88eaSTakashi Iwai {
2343a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2344a86a88eaSTakashi Iwai 	struct snd_kcontrol_new *knew;
2345a86a88eaSTakashi Iwai 
2346a86a88eaSTakashi Iwai 	if (spec->num_inputs <= 1 || !count)
2347a86a88eaSTakashi Iwai 		return 0; /* no need for single src */
2348a86a88eaSTakashi Iwai 
2349a86a88eaSTakashi Iwai 	knew = via_clone_control(spec, &via_input_src_ctl);
2350a86a88eaSTakashi Iwai 	if (!knew)
2351a86a88eaSTakashi Iwai 		return -ENOMEM;
2352a86a88eaSTakashi Iwai 	knew->count = count;
2353a86a88eaSTakashi Iwai 	return 0;
2354a86a88eaSTakashi Iwai }
2355a86a88eaSTakashi Iwai 
2356a86a88eaSTakashi Iwai /* add the powersave loopback-list entry */
235713af8e77STakashi Iwai static void add_loopback_list(struct via_spec *spec, hda_nid_t mix, int idx)
235813af8e77STakashi Iwai {
235913af8e77STakashi Iwai 	struct hda_amp_list *list;
236013af8e77STakashi Iwai 
236113af8e77STakashi Iwai 	if (spec->num_loopbacks >= ARRAY_SIZE(spec->loopback_list) - 1)
236213af8e77STakashi Iwai 		return;
236313af8e77STakashi Iwai 	list = spec->loopback_list + spec->num_loopbacks;
236413af8e77STakashi Iwai 	list->nid = mix;
236513af8e77STakashi Iwai 	list->dir = HDA_INPUT;
236613af8e77STakashi Iwai 	list->idx = idx;
236713af8e77STakashi Iwai 	spec->num_loopbacks++;
236813af8e77STakashi Iwai 	spec->loopback.amplist = spec->loopback_list;
236913af8e77STakashi Iwai }
237013af8e77STakashi Iwai 
2371a86a88eaSTakashi Iwai static bool is_reachable_nid(struct hda_codec *codec, hda_nid_t src,
23728d087c76STakashi Iwai 			     hda_nid_t dst)
2373a86a88eaSTakashi Iwai {
23748d087c76STakashi Iwai 	return snd_hda_get_conn_index(codec, src, dst, 1) >= 0;
2375a86a88eaSTakashi Iwai }
2376a86a88eaSTakashi Iwai 
2377a86a88eaSTakashi Iwai /* add the input-route to the given pin */
2378a86a88eaSTakashi Iwai static bool add_input_route(struct hda_codec *codec, hda_nid_t pin)
2379c577b8a1SJoseph Chan {
238010a20af7STakashi Iwai 	struct via_spec *spec = codec->spec;
2381a86a88eaSTakashi Iwai 	int c, idx;
2382a86a88eaSTakashi Iwai 
2383a86a88eaSTakashi Iwai 	spec->inputs[spec->num_inputs].adc_idx = -1;
2384a86a88eaSTakashi Iwai 	spec->inputs[spec->num_inputs].pin = pin;
2385a86a88eaSTakashi Iwai 	for (c = 0; c < spec->num_adc_nids; c++) {
2386a86a88eaSTakashi Iwai 		if (spec->mux_nids[c]) {
2387a86a88eaSTakashi Iwai 			idx = get_connection_index(codec, spec->mux_nids[c],
2388a86a88eaSTakashi Iwai 						   pin);
2389a86a88eaSTakashi Iwai 			if (idx < 0)
2390a86a88eaSTakashi Iwai 				continue;
2391a86a88eaSTakashi Iwai 			spec->inputs[spec->num_inputs].mux_idx = idx;
2392a86a88eaSTakashi Iwai 		} else {
23938d087c76STakashi Iwai 			if (!is_reachable_nid(codec, spec->adc_nids[c], pin))
2394a86a88eaSTakashi Iwai 				continue;
2395a86a88eaSTakashi Iwai 		}
2396a86a88eaSTakashi Iwai 		spec->inputs[spec->num_inputs].adc_idx = c;
2397a86a88eaSTakashi Iwai 		/* Can primary ADC satisfy all inputs? */
2398a86a88eaSTakashi Iwai 		if (!spec->dyn_adc_switch &&
2399a86a88eaSTakashi Iwai 		    spec->num_inputs > 0 && spec->inputs[0].adc_idx != c) {
2400a86a88eaSTakashi Iwai 			snd_printd(KERN_INFO
2401a86a88eaSTakashi Iwai 				   "via: dynamic ADC switching enabled\n");
2402a86a88eaSTakashi Iwai 			spec->dyn_adc_switch = 1;
2403a86a88eaSTakashi Iwai 		}
2404a86a88eaSTakashi Iwai 		return true;
2405a86a88eaSTakashi Iwai 	}
2406a86a88eaSTakashi Iwai 	return false;
2407a86a88eaSTakashi Iwai }
2408a86a88eaSTakashi Iwai 
2409a86a88eaSTakashi Iwai static int get_mux_nids(struct hda_codec *codec);
2410a86a88eaSTakashi Iwai 
2411a86a88eaSTakashi Iwai /* parse input-routes; fill ADCs, MUXs and input-src entries */
2412a86a88eaSTakashi Iwai static int parse_analog_inputs(struct hda_codec *codec)
2413a86a88eaSTakashi Iwai {
2414a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2415a86a88eaSTakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
2416a86a88eaSTakashi Iwai 	int i, err;
2417a766d0d7STakashi Iwai 
2418a766d0d7STakashi Iwai 	err = via_fill_adcs(codec);
2419a766d0d7STakashi Iwai 	if (err < 0)
2420a766d0d7STakashi Iwai 		return err;
2421a766d0d7STakashi Iwai 	err = get_mux_nids(codec);
2422a766d0d7STakashi Iwai 	if (err < 0)
2423a766d0d7STakashi Iwai 		return err;
2424a766d0d7STakashi Iwai 
2425a86a88eaSTakashi Iwai 	/* fill all input-routes */
2426a86a88eaSTakashi Iwai 	for (i = 0; i < cfg->num_inputs; i++) {
2427a86a88eaSTakashi Iwai 		if (add_input_route(codec, cfg->inputs[i].pin))
2428a86a88eaSTakashi Iwai 			spec->inputs[spec->num_inputs++].label =
2429a86a88eaSTakashi Iwai 				hda_get_autocfg_input_label(codec, cfg, i);
2430a86a88eaSTakashi Iwai 	}
2431a86a88eaSTakashi Iwai 
2432a86a88eaSTakashi Iwai 	/* check for internal loopback recording */
2433a86a88eaSTakashi Iwai 	if (spec->aa_mix_nid &&
2434a86a88eaSTakashi Iwai 	    add_input_route(codec, spec->aa_mix_nid))
2435a86a88eaSTakashi Iwai 		spec->inputs[spec->num_inputs++].label = "Stereo Mixer";
2436a86a88eaSTakashi Iwai 
2437a86a88eaSTakashi Iwai 	return 0;
2438a86a88eaSTakashi Iwai }
2439a86a88eaSTakashi Iwai 
2440a86a88eaSTakashi Iwai /* create analog-loopback volume/switch controls */
2441a86a88eaSTakashi Iwai static int create_loopback_ctls(struct hda_codec *codec)
2442a86a88eaSTakashi Iwai {
2443a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2444a86a88eaSTakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
2445a86a88eaSTakashi Iwai 	const char *prev_label = NULL;
2446a86a88eaSTakashi Iwai 	int type_idx = 0;
2447a86a88eaSTakashi Iwai 	int i, j, err, idx;
2448a86a88eaSTakashi Iwai 
2449a86a88eaSTakashi Iwai 	if (!spec->aa_mix_nid)
2450a766d0d7STakashi Iwai 		return 0;
2451c577b8a1SJoseph Chan 
24527b315bb4STakashi Iwai 	for (i = 0; i < cfg->num_inputs; i++) {
2453a86a88eaSTakashi Iwai 		hda_nid_t pin = cfg->inputs[i].pin;
2454a86a88eaSTakashi Iwai 		const char *label = hda_get_autocfg_input_label(codec, cfg, i);
2455a86a88eaSTakashi Iwai 
24561e11cae1STakashi Iwai 		if (prev_label && !strcmp(label, prev_label))
24577b315bb4STakashi Iwai 			type_idx++;
24587b315bb4STakashi Iwai 		else
24597b315bb4STakashi Iwai 			type_idx = 0;
24601e11cae1STakashi Iwai 		prev_label = label;
2461a86a88eaSTakashi Iwai 		idx = get_connection_index(codec, spec->aa_mix_nid, pin);
2462a86a88eaSTakashi Iwai 		if (idx >= 0) {
246316922281SLydia Wang 			err = via_new_analog_input(spec, label, type_idx,
2464a86a88eaSTakashi Iwai 						   idx, spec->aa_mix_nid);
2465c577b8a1SJoseph Chan 			if (err < 0)
2466c577b8a1SJoseph Chan 				return err;
2467a86a88eaSTakashi Iwai 			add_loopback_list(spec, spec->aa_mix_nid, idx);
246813af8e77STakashi Iwai 		}
2469e3d7a143STakashi Iwai 
2470e3d7a143STakashi Iwai 		/* remember the label for smart51 control */
2471e3d7a143STakashi Iwai 		for (j = 0; j < spec->smart51_nums; j++) {
2472a86a88eaSTakashi Iwai 			if (spec->smart51_pins[j] == pin) {
2473e3d7a143STakashi Iwai 				spec->smart51_idxs[j] = idx;
2474e3d7a143STakashi Iwai 				spec->smart51_labels[j] = label;
2475e3d7a143STakashi Iwai 				break;
2476e3d7a143STakashi Iwai 			}
2477e3d7a143STakashi Iwai 		}
2478c577b8a1SJoseph Chan 	}
2479a86a88eaSTakashi Iwai 	return 0;
2480a86a88eaSTakashi Iwai }
2481a86a88eaSTakashi Iwai 
2482a86a88eaSTakashi Iwai /* create mic-boost controls (if present) */
2483a86a88eaSTakashi Iwai static int create_mic_boost_ctls(struct hda_codec *codec)
2484a86a88eaSTakashi Iwai {
2485a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2486a86a88eaSTakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
24878d8bbc6fSTakashi Iwai 	const char *prev_label = NULL;
24888d8bbc6fSTakashi Iwai 	int type_idx = 0;
2489a86a88eaSTakashi Iwai 	int i, err;
2490a86a88eaSTakashi Iwai 
2491a86a88eaSTakashi Iwai 	for (i = 0; i < cfg->num_inputs; i++) {
2492a86a88eaSTakashi Iwai 		hda_nid_t pin = cfg->inputs[i].pin;
2493a86a88eaSTakashi Iwai 		unsigned int caps;
2494a86a88eaSTakashi Iwai 		const char *label;
2495a86a88eaSTakashi Iwai 		char name[32];
2496a86a88eaSTakashi Iwai 
2497a86a88eaSTakashi Iwai 		if (cfg->inputs[i].type != AUTO_PIN_MIC)
2498a86a88eaSTakashi Iwai 			continue;
2499a86a88eaSTakashi Iwai 		caps = query_amp_caps(codec, pin, HDA_INPUT);
2500a86a88eaSTakashi Iwai 		if (caps == -1 || !(caps & AC_AMPCAP_NUM_STEPS))
2501a86a88eaSTakashi Iwai 			continue;
2502a86a88eaSTakashi Iwai 		label = hda_get_autocfg_input_label(codec, cfg, i);
25038d8bbc6fSTakashi Iwai 		if (prev_label && !strcmp(label, prev_label))
25048d8bbc6fSTakashi Iwai 			type_idx++;
25058d8bbc6fSTakashi Iwai 		else
25068d8bbc6fSTakashi Iwai 			type_idx = 0;
25078d8bbc6fSTakashi Iwai 		prev_label = label;
2508a86a88eaSTakashi Iwai 		snprintf(name, sizeof(name), "%s Boost Volume", label);
25098d8bbc6fSTakashi Iwai 		err = __via_add_control(spec, VIA_CTL_WIDGET_VOL, name, type_idx,
2510a86a88eaSTakashi Iwai 			      HDA_COMPOSE_AMP_VAL(pin, 3, 0, HDA_INPUT));
2511a86a88eaSTakashi Iwai 		if (err < 0)
2512a86a88eaSTakashi Iwai 			return err;
2513a86a88eaSTakashi Iwai 	}
2514a86a88eaSTakashi Iwai 	return 0;
2515a86a88eaSTakashi Iwai }
2516a86a88eaSTakashi Iwai 
2517a86a88eaSTakashi Iwai /* create capture and input-src controls for multiple streams */
2518a86a88eaSTakashi Iwai static int create_multi_adc_ctls(struct hda_codec *codec)
2519a86a88eaSTakashi Iwai {
2520a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2521a86a88eaSTakashi Iwai 	int i, err;
2522d7a99cceSTakashi Iwai 
2523d7a99cceSTakashi Iwai 	/* create capture mixer elements */
2524d7a99cceSTakashi Iwai 	for (i = 0; i < spec->num_adc_nids; i++) {
2525d7a99cceSTakashi Iwai 		hda_nid_t adc = spec->adc_nids[i];
2526d7a99cceSTakashi Iwai 		err = __via_add_control(spec, VIA_CTL_WIDGET_VOL,
2527d7a99cceSTakashi Iwai 					"Capture Volume", i,
2528d7a99cceSTakashi Iwai 					HDA_COMPOSE_AMP_VAL(adc, 3, 0,
2529d7a99cceSTakashi Iwai 							    HDA_INPUT));
2530d7a99cceSTakashi Iwai 		if (err < 0)
2531d7a99cceSTakashi Iwai 			return err;
2532d7a99cceSTakashi Iwai 		err = __via_add_control(spec, VIA_CTL_WIDGET_MUTE,
2533d7a99cceSTakashi Iwai 					"Capture Switch", i,
2534d7a99cceSTakashi Iwai 					HDA_COMPOSE_AMP_VAL(adc, 3, 0,
2535d7a99cceSTakashi Iwai 							    HDA_INPUT));
2536d7a99cceSTakashi Iwai 		if (err < 0)
2537d7a99cceSTakashi Iwai 			return err;
2538d7a99cceSTakashi Iwai 	}
2539d7a99cceSTakashi Iwai 
2540d7a99cceSTakashi Iwai 	/* input-source control */
2541d7a99cceSTakashi Iwai 	for (i = 0; i < spec->num_adc_nids; i++)
2542d7a99cceSTakashi Iwai 		if (!spec->mux_nids[i])
2543d7a99cceSTakashi Iwai 			break;
2544a86a88eaSTakashi Iwai 	err = create_input_src_ctls(codec, i);
2545d7a99cceSTakashi Iwai 	if (err < 0)
2546d7a99cceSTakashi Iwai 		return err;
2547a86a88eaSTakashi Iwai 	return 0;
2548d7a99cceSTakashi Iwai }
2549d7a99cceSTakashi Iwai 
2550a86a88eaSTakashi Iwai /* bind capture volume/switch */
2551a86a88eaSTakashi Iwai static struct snd_kcontrol_new via_bind_cap_vol_ctl =
2552a86a88eaSTakashi Iwai 	HDA_BIND_VOL("Capture Volume", 0);
2553a86a88eaSTakashi Iwai static struct snd_kcontrol_new via_bind_cap_sw_ctl =
2554a86a88eaSTakashi Iwai 	HDA_BIND_SW("Capture Switch", 0);
2555a86a88eaSTakashi Iwai 
2556a86a88eaSTakashi Iwai static int init_bind_ctl(struct via_spec *spec, struct hda_bind_ctls **ctl_ret,
2557a86a88eaSTakashi Iwai 			 struct hda_ctl_ops *ops)
2558a86a88eaSTakashi Iwai {
2559a86a88eaSTakashi Iwai 	struct hda_bind_ctls *ctl;
2560a86a88eaSTakashi Iwai 	int i;
2561a86a88eaSTakashi Iwai 
2562a86a88eaSTakashi Iwai 	ctl = kzalloc(sizeof(*ctl) + sizeof(long) * 4, GFP_KERNEL);
2563a86a88eaSTakashi Iwai 	if (!ctl)
2564a86a88eaSTakashi Iwai 		return -ENOMEM;
2565a86a88eaSTakashi Iwai 	ctl->ops = ops;
2566a86a88eaSTakashi Iwai 	for (i = 0; i < spec->num_adc_nids; i++)
2567a86a88eaSTakashi Iwai 		ctl->values[i] =
2568a86a88eaSTakashi Iwai 			HDA_COMPOSE_AMP_VAL(spec->adc_nids[i], 3, 0, HDA_INPUT);
2569a86a88eaSTakashi Iwai 	*ctl_ret = ctl;
2570a86a88eaSTakashi Iwai 	return 0;
2571a86a88eaSTakashi Iwai }
2572a86a88eaSTakashi Iwai 
2573a86a88eaSTakashi Iwai /* create capture and input-src controls for dynamic ADC-switch case */
2574a86a88eaSTakashi Iwai static int create_dyn_adc_ctls(struct hda_codec *codec)
2575a86a88eaSTakashi Iwai {
2576a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2577a86a88eaSTakashi Iwai 	struct snd_kcontrol_new *knew;
2578a86a88eaSTakashi Iwai 	int err;
2579a86a88eaSTakashi Iwai 
2580a86a88eaSTakashi Iwai 	/* set up the bind capture ctls */
2581a86a88eaSTakashi Iwai 	err = init_bind_ctl(spec, &spec->bind_cap_vol, &snd_hda_bind_vol);
2582a86a88eaSTakashi Iwai 	if (err < 0)
2583a86a88eaSTakashi Iwai 		return err;
2584a86a88eaSTakashi Iwai 	err = init_bind_ctl(spec, &spec->bind_cap_sw, &snd_hda_bind_sw);
2585a86a88eaSTakashi Iwai 	if (err < 0)
2586a86a88eaSTakashi Iwai 		return err;
2587a86a88eaSTakashi Iwai 
2588a86a88eaSTakashi Iwai 	/* create capture mixer elements */
2589a86a88eaSTakashi Iwai 	knew = via_clone_control(spec, &via_bind_cap_vol_ctl);
2590a86a88eaSTakashi Iwai 	if (!knew)
2591a86a88eaSTakashi Iwai 		return -ENOMEM;
2592a86a88eaSTakashi Iwai 	knew->private_value = (long)spec->bind_cap_vol;
2593a86a88eaSTakashi Iwai 
2594a86a88eaSTakashi Iwai 	knew = via_clone_control(spec, &via_bind_cap_sw_ctl);
2595a86a88eaSTakashi Iwai 	if (!knew)
2596a86a88eaSTakashi Iwai 		return -ENOMEM;
2597a86a88eaSTakashi Iwai 	knew->private_value = (long)spec->bind_cap_sw;
2598a86a88eaSTakashi Iwai 
2599a86a88eaSTakashi Iwai 	/* input-source control */
2600a86a88eaSTakashi Iwai 	err = create_input_src_ctls(codec, 1);
2601a86a88eaSTakashi Iwai 	if (err < 0)
2602a86a88eaSTakashi Iwai 		return err;
2603a86a88eaSTakashi Iwai 	return 0;
2604a86a88eaSTakashi Iwai }
2605a86a88eaSTakashi Iwai 
2606a86a88eaSTakashi Iwai /* parse and create capture-related stuff */
2607a86a88eaSTakashi Iwai static int via_auto_create_analog_input_ctls(struct hda_codec *codec)
2608a86a88eaSTakashi Iwai {
2609a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2610a86a88eaSTakashi Iwai 	int err;
2611a86a88eaSTakashi Iwai 
2612a86a88eaSTakashi Iwai 	err = parse_analog_inputs(codec);
2613a86a88eaSTakashi Iwai 	if (err < 0)
2614a86a88eaSTakashi Iwai 		return err;
2615a86a88eaSTakashi Iwai 	if (spec->dyn_adc_switch)
2616a86a88eaSTakashi Iwai 		err = create_dyn_adc_ctls(codec);
2617a86a88eaSTakashi Iwai 	else
2618a86a88eaSTakashi Iwai 		err = create_multi_adc_ctls(codec);
2619a86a88eaSTakashi Iwai 	if (err < 0)
2620a86a88eaSTakashi Iwai 		return err;
2621a86a88eaSTakashi Iwai 	err = create_loopback_ctls(codec);
2622a86a88eaSTakashi Iwai 	if (err < 0)
2623a86a88eaSTakashi Iwai 		return err;
2624a86a88eaSTakashi Iwai 	err = create_mic_boost_ctls(codec);
2625a86a88eaSTakashi Iwai 	if (err < 0)
2626a86a88eaSTakashi Iwai 		return err;
2627c577b8a1SJoseph Chan 	return 0;
2628c577b8a1SJoseph Chan }
2629c577b8a1SJoseph Chan 
263076d9b0ddSHarald Welte static void vt1708_set_pinconfig_connect(struct hda_codec *codec, hda_nid_t nid)
263176d9b0ddSHarald Welte {
263276d9b0ddSHarald Welte 	unsigned int def_conf;
263376d9b0ddSHarald Welte 	unsigned char seqassoc;
263476d9b0ddSHarald Welte 
26352f334f92STakashi Iwai 	def_conf = snd_hda_codec_get_pincfg(codec, nid);
263676d9b0ddSHarald Welte 	seqassoc = (unsigned char) get_defcfg_association(def_conf);
263776d9b0ddSHarald Welte 	seqassoc = (seqassoc << 4) | get_defcfg_sequence(def_conf);
263882ef9e45SLydia Wang 	if (get_defcfg_connect(def_conf) == AC_JACK_PORT_NONE
263982ef9e45SLydia Wang 	    && (seqassoc == 0xf0 || seqassoc == 0xff)) {
264076d9b0ddSHarald Welte 		def_conf = def_conf & (~(AC_JACK_PORT_BOTH << 30));
26412f334f92STakashi Iwai 		snd_hda_codec_set_pincfg(codec, nid, def_conf);
264276d9b0ddSHarald Welte 	}
264376d9b0ddSHarald Welte 
264476d9b0ddSHarald Welte 	return;
264576d9b0ddSHarald Welte }
264676d9b0ddSHarald Welte 
2647e06e5a29STakashi Iwai static int vt1708_jack_detect_get(struct snd_kcontrol *kcontrol,
26481f2e99feSLydia Wang 				     struct snd_ctl_elem_value *ucontrol)
26491f2e99feSLydia Wang {
26501f2e99feSLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
26511f2e99feSLydia Wang 	struct via_spec *spec = codec->spec;
26521f2e99feSLydia Wang 
26531f2e99feSLydia Wang 	if (spec->codec_type != VT1708)
26541f2e99feSLydia Wang 		return 0;
2655e06e5a29STakashi Iwai 	ucontrol->value.integer.value[0] = spec->vt1708_jack_detect;
26561f2e99feSLydia Wang 	return 0;
26571f2e99feSLydia Wang }
26581f2e99feSLydia Wang 
2659e06e5a29STakashi Iwai static int vt1708_jack_detect_put(struct snd_kcontrol *kcontrol,
26601f2e99feSLydia Wang 				     struct snd_ctl_elem_value *ucontrol)
26611f2e99feSLydia Wang {
26621f2e99feSLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
26631f2e99feSLydia Wang 	struct via_spec *spec = codec->spec;
2664187d333eSTakashi Iwai 	int val;
26651f2e99feSLydia Wang 
26661f2e99feSLydia Wang 	if (spec->codec_type != VT1708)
26671f2e99feSLydia Wang 		return 0;
2668187d333eSTakashi Iwai 	val = !!ucontrol->value.integer.value[0];
2669187d333eSTakashi Iwai 	if (spec->vt1708_jack_detect == val)
2670187d333eSTakashi Iwai 		return 0;
2671187d333eSTakashi Iwai 	spec->vt1708_jack_detect = val;
2672187d333eSTakashi Iwai 	if (spec->vt1708_jack_detect &&
2673187d333eSTakashi Iwai 	    snd_hda_get_bool_hint(codec, "analog_loopback_hp_detect") != 1) {
26741f2e99feSLydia Wang 		mute_aa_path(codec, 1);
26751f2e99feSLydia Wang 		notify_aa_path_ctls(codec);
26761f2e99feSLydia Wang 	}
2677187d333eSTakashi Iwai 	via_hp_automute(codec);
2678187d333eSTakashi Iwai 	vt1708_update_hp_work(spec);
2679187d333eSTakashi Iwai 	return 1;
26801f2e99feSLydia Wang }
26811f2e99feSLydia Wang 
2682e06e5a29STakashi Iwai static const struct snd_kcontrol_new vt1708_jack_detect_ctl = {
26831f2e99feSLydia Wang 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
26841f2e99feSLydia Wang 	.name = "Jack Detect",
26851f2e99feSLydia Wang 	.count = 1,
26861f2e99feSLydia Wang 	.info = snd_ctl_boolean_mono_info,
2687e06e5a29STakashi Iwai 	.get = vt1708_jack_detect_get,
2688e06e5a29STakashi Iwai 	.put = vt1708_jack_detect_put,
26891f2e99feSLydia Wang };
26901f2e99feSLydia Wang 
269112daef65STakashi Iwai static void fill_dig_outs(struct hda_codec *codec);
269212daef65STakashi Iwai static void fill_dig_in(struct hda_codec *codec);
269312daef65STakashi Iwai 
269412daef65STakashi Iwai static int via_parse_auto_config(struct hda_codec *codec)
2695c577b8a1SJoseph Chan {
2696c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
2697c577b8a1SJoseph Chan 	int err;
2698c577b8a1SJoseph Chan 
2699c577b8a1SJoseph Chan 	err = snd_hda_parse_pin_def_config(codec, &spec->autocfg, NULL);
2700c577b8a1SJoseph Chan 	if (err < 0)
2701c577b8a1SJoseph Chan 		return err;
2702c577b8a1SJoseph Chan 	if (!spec->autocfg.line_outs && !spec->autocfg.hp_pins[0])
27037f0df88cSTakashi Iwai 		return -EINVAL;
2704c577b8a1SJoseph Chan 
27054a79616dSTakashi Iwai 	err = via_auto_create_multi_out_ctls(codec);
2706c577b8a1SJoseph Chan 	if (err < 0)
2707c577b8a1SJoseph Chan 		return err;
27084a79616dSTakashi Iwai 	err = via_auto_create_hp_ctls(codec, spec->autocfg.hp_pins[0]);
2709c577b8a1SJoseph Chan 	if (err < 0)
2710c577b8a1SJoseph Chan 		return err;
27114a918ffeSTakashi Iwai 	err = via_auto_create_speaker_ctls(codec);
27124a918ffeSTakashi Iwai 	if (err < 0)
27134a918ffeSTakashi Iwai 		return err;
27143214b966STakashi Iwai 	err = via_auto_create_loopback_switch(codec);
27153214b966STakashi Iwai 	if (err < 0)
27163214b966STakashi Iwai 		return err;
2717a86a88eaSTakashi Iwai 	err = via_auto_create_analog_input_ctls(codec);
2718c577b8a1SJoseph Chan 	if (err < 0)
2719c577b8a1SJoseph Chan 		return err;
2720c577b8a1SJoseph Chan 
2721c577b8a1SJoseph Chan 	spec->multiout.max_channels = spec->multiout.num_dacs * 2;
2722c577b8a1SJoseph Chan 
272312daef65STakashi Iwai 	fill_dig_outs(codec);
272412daef65STakashi Iwai 	fill_dig_in(codec);
2725c577b8a1SJoseph Chan 
2726603c4019STakashi Iwai 	if (spec->kctls.list)
2727603c4019STakashi Iwai 		spec->mixers[spec->num_mixers++] = spec->kctls.list;
2728c577b8a1SJoseph Chan 
2729c577b8a1SJoseph Chan 
27303214b966STakashi Iwai 	if (spec->hp_dac_nid && spec->hp_mix_path.depth) {
2731ece8d043STakashi Iwai 		err = via_hp_build(codec);
2732ece8d043STakashi Iwai 		if (err < 0)
2733ece8d043STakashi Iwai 			return err;
2734ece8d043STakashi Iwai 	}
2735c577b8a1SJoseph Chan 
2736f4a7828bSTakashi Iwai 	err = via_smart51_build(codec);
2737f4a7828bSTakashi Iwai 	if (err < 0)
2738f4a7828bSTakashi Iwai 		return err;
2739f4a7828bSTakashi Iwai 
27405d41762aSTakashi Iwai 	/* assign slave outs */
27415d41762aSTakashi Iwai 	if (spec->slave_dig_outs[0])
27425d41762aSTakashi Iwai 		codec->slave_dig_outs = spec->slave_dig_outs;
27435d41762aSTakashi Iwai 
2744c577b8a1SJoseph Chan 	return 1;
2745c577b8a1SJoseph Chan }
2746c577b8a1SJoseph Chan 
27475d41762aSTakashi Iwai static void via_auto_init_dig_outs(struct hda_codec *codec)
2748c577b8a1SJoseph Chan {
274925eaba2fSLydia Wang 	struct via_spec *spec = codec->spec;
27505d41762aSTakashi Iwai 	if (spec->multiout.dig_out_nid)
27515d41762aSTakashi Iwai 		init_output_pin(codec, spec->autocfg.dig_out_pins[0], PIN_OUT);
27525d41762aSTakashi Iwai 	if (spec->slave_dig_outs[0])
27535d41762aSTakashi Iwai 		init_output_pin(codec, spec->autocfg.dig_out_pins[1], PIN_OUT);
27545d41762aSTakashi Iwai }
275525eaba2fSLydia Wang 
27565d41762aSTakashi Iwai static void via_auto_init_dig_in(struct hda_codec *codec)
27575d41762aSTakashi Iwai {
27585d41762aSTakashi Iwai 	struct via_spec *spec = codec->spec;
27595d41762aSTakashi Iwai 	if (!spec->dig_in_nid)
27605d41762aSTakashi Iwai 		return;
2761cdd03cedSTakashi Iwai 	snd_hda_set_pin_ctl(codec, spec->autocfg.dig_in_pin, PIN_IN);
27625d41762aSTakashi Iwai }
27635d41762aSTakashi Iwai 
27644a918ffeSTakashi Iwai /* initialize the unsolicited events */
27654a918ffeSTakashi Iwai static void via_auto_init_unsol_event(struct hda_codec *codec)
27664a918ffeSTakashi Iwai {
27674a918ffeSTakashi Iwai 	struct via_spec *spec = codec->spec;
27684a918ffeSTakashi Iwai 	struct auto_pin_cfg *cfg = &spec->autocfg;
27694a918ffeSTakashi Iwai 	unsigned int ev;
27704a918ffeSTakashi Iwai 	int i;
27714a918ffeSTakashi Iwai 
27724a918ffeSTakashi Iwai 	if (cfg->hp_pins[0] && is_jack_detectable(codec, cfg->hp_pins[0]))
27731835a0f9STakashi Iwai 		snd_hda_jack_detect_enable(codec, cfg->hp_pins[0],
27741835a0f9STakashi Iwai 					   VIA_HP_EVENT | VIA_JACK_EVENT);
27754a918ffeSTakashi Iwai 
27764a918ffeSTakashi Iwai 	if (cfg->speaker_pins[0])
27774a918ffeSTakashi Iwai 		ev = VIA_LINE_EVENT;
27784a918ffeSTakashi Iwai 	else
27794a918ffeSTakashi Iwai 		ev = 0;
27804a918ffeSTakashi Iwai 	for (i = 0; i < cfg->line_outs; i++) {
27814a918ffeSTakashi Iwai 		if (cfg->line_out_pins[i] &&
27824a918ffeSTakashi Iwai 		    is_jack_detectable(codec, cfg->line_out_pins[i]))
27831835a0f9STakashi Iwai 			snd_hda_jack_detect_enable(codec, cfg->line_out_pins[i],
27841835a0f9STakashi Iwai 						   ev | VIA_JACK_EVENT);
27854a918ffeSTakashi Iwai 	}
27864a918ffeSTakashi Iwai 
27874a918ffeSTakashi Iwai 	for (i = 0; i < cfg->num_inputs; i++) {
27884a918ffeSTakashi Iwai 		if (is_jack_detectable(codec, cfg->inputs[i].pin))
27891835a0f9STakashi Iwai 			snd_hda_jack_detect_enable(codec, cfg->inputs[i].pin,
27901835a0f9STakashi Iwai 						   VIA_JACK_EVENT);
27914a918ffeSTakashi Iwai 	}
27924a918ffeSTakashi Iwai }
27934a918ffeSTakashi Iwai 
27945d41762aSTakashi Iwai static int via_init(struct hda_codec *codec)
27955d41762aSTakashi Iwai {
27965d41762aSTakashi Iwai 	struct via_spec *spec = codec->spec;
27975d41762aSTakashi Iwai 	int i;
27985d41762aSTakashi Iwai 
27995d41762aSTakashi Iwai 	for (i = 0; i < spec->num_iverbs; i++)
28005d41762aSTakashi Iwai 		snd_hda_sequence_write(codec, spec->init_verbs[i]);
28015d41762aSTakashi Iwai 
2802e9d010c2STakashi Iwai 	/* init power states */
2803e9d010c2STakashi Iwai 	set_widgets_power_state(codec);
2804e9d010c2STakashi Iwai 	__analog_low_current_mode(codec, true);
2805e9d010c2STakashi Iwai 
2806c577b8a1SJoseph Chan 	via_auto_init_multi_out(codec);
2807c577b8a1SJoseph Chan 	via_auto_init_hp_out(codec);
28084a918ffeSTakashi Iwai 	via_auto_init_speaker_out(codec);
2809c577b8a1SJoseph Chan 	via_auto_init_analog_input(codec);
28105d41762aSTakashi Iwai 	via_auto_init_dig_outs(codec);
28115d41762aSTakashi Iwai 	via_auto_init_dig_in(codec);
281211890956SLydia Wang 
28134a918ffeSTakashi Iwai 	via_auto_init_unsol_event(codec);
28144a918ffeSTakashi Iwai 
281525eaba2fSLydia Wang 	via_hp_automute(codec);
2816187d333eSTakashi Iwai 	vt1708_update_hp_work(spec);
281725eaba2fSLydia Wang 
2818c577b8a1SJoseph Chan 	return 0;
2819c577b8a1SJoseph Chan }
2820c577b8a1SJoseph Chan 
28211f2e99feSLydia Wang static void vt1708_update_hp_jack_state(struct work_struct *work)
28221f2e99feSLydia Wang {
28231f2e99feSLydia Wang 	struct via_spec *spec = container_of(work, struct via_spec,
28241f2e99feSLydia Wang 					     vt1708_hp_work.work);
28251f2e99feSLydia Wang 	if (spec->codec_type != VT1708)
28261f2e99feSLydia Wang 		return;
28271835a0f9STakashi Iwai 	snd_hda_jack_set_dirty_all(spec->codec);
28281f2e99feSLydia Wang 	/* if jack state toggled */
28291f2e99feSLydia Wang 	if (spec->vt1708_hp_present
2830d56757abSTakashi Iwai 	    != snd_hda_jack_detect(spec->codec, spec->autocfg.hp_pins[0])) {
28311f2e99feSLydia Wang 		spec->vt1708_hp_present ^= 1;
28321f2e99feSLydia Wang 		via_hp_automute(spec->codec);
28331f2e99feSLydia Wang 	}
2834187d333eSTakashi Iwai 	if (spec->vt1708_jack_detect)
2835187d333eSTakashi Iwai 		schedule_delayed_work(&spec->vt1708_hp_work,
2836187d333eSTakashi Iwai 				      msecs_to_jiffies(100));
28371f2e99feSLydia Wang }
28381f2e99feSLydia Wang 
2839337b9d02STakashi Iwai static int get_mux_nids(struct hda_codec *codec)
2840337b9d02STakashi Iwai {
2841337b9d02STakashi Iwai 	struct via_spec *spec = codec->spec;
2842337b9d02STakashi Iwai 	hda_nid_t nid, conn[8];
2843337b9d02STakashi Iwai 	unsigned int type;
2844337b9d02STakashi Iwai 	int i, n;
2845337b9d02STakashi Iwai 
2846337b9d02STakashi Iwai 	for (i = 0; i < spec->num_adc_nids; i++) {
2847337b9d02STakashi Iwai 		nid = spec->adc_nids[i];
2848337b9d02STakashi Iwai 		while (nid) {
2849a22d543aSTakashi Iwai 			type = get_wcaps_type(get_wcaps(codec, nid));
28501c55d521STakashi Iwai 			if (type == AC_WID_PIN)
28511c55d521STakashi Iwai 				break;
2852337b9d02STakashi Iwai 			n = snd_hda_get_connections(codec, nid, conn,
2853337b9d02STakashi Iwai 						    ARRAY_SIZE(conn));
2854337b9d02STakashi Iwai 			if (n <= 0)
2855337b9d02STakashi Iwai 				break;
2856337b9d02STakashi Iwai 			if (n > 1) {
2857337b9d02STakashi Iwai 				spec->mux_nids[i] = nid;
2858337b9d02STakashi Iwai 				break;
2859337b9d02STakashi Iwai 			}
2860337b9d02STakashi Iwai 			nid = conn[0];
2861337b9d02STakashi Iwai 		}
2862337b9d02STakashi Iwai 	}
28631c55d521STakashi Iwai 	return 0;
2864337b9d02STakashi Iwai }
2865337b9d02STakashi Iwai 
2866c577b8a1SJoseph Chan static int patch_vt1708(struct hda_codec *codec)
2867c577b8a1SJoseph Chan {
2868c577b8a1SJoseph Chan 	struct via_spec *spec;
2869c577b8a1SJoseph Chan 	int err;
2870c577b8a1SJoseph Chan 
2871c577b8a1SJoseph Chan 	/* create a codec specific record */
28725b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
2873c577b8a1SJoseph Chan 	if (spec == NULL)
2874c577b8a1SJoseph Chan 		return -ENOMEM;
2875c577b8a1SJoseph Chan 
2876620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x17;
2877620e2b28STakashi Iwai 
287812daef65STakashi Iwai 	/* Add HP and CD pin config connect bit re-config action */
287912daef65STakashi Iwai 	vt1708_set_pinconfig_connect(codec, VT1708_HP_PIN_NID);
288012daef65STakashi Iwai 	vt1708_set_pinconfig_connect(codec, VT1708_CD_PIN_NID);
288112daef65STakashi Iwai 
2882c577b8a1SJoseph Chan 	/* automatic parse from the BIOS config */
288312daef65STakashi Iwai 	err = via_parse_auto_config(codec);
2884c577b8a1SJoseph Chan 	if (err < 0) {
2885c577b8a1SJoseph Chan 		via_free(codec);
2886c577b8a1SJoseph Chan 		return err;
2887c577b8a1SJoseph Chan 	}
2888c577b8a1SJoseph Chan 
288912daef65STakashi Iwai 	/* add jack detect on/off control */
289012daef65STakashi Iwai 	if (!via_clone_control(spec, &vt1708_jack_detect_ctl))
289112daef65STakashi Iwai 		return -ENOMEM;
289212daef65STakashi Iwai 
2893bc9b5623STakashi Iwai 	/* disable 32bit format on VT1708 */
2894bc9b5623STakashi Iwai 	if (codec->vendor_id == 0x11061708)
2895bc9b5623STakashi Iwai 		spec->stream_analog_playback = &vt1708_pcm_analog_s16_playback;
2896c577b8a1SJoseph Chan 
2897e322a36dSLydia Wang 	spec->init_verbs[spec->num_iverbs++] = vt1708_init_verbs;
2898e322a36dSLydia Wang 
2899c577b8a1SJoseph Chan 	codec->patch_ops = via_patch_ops;
2900c577b8a1SJoseph Chan 
29011f2e99feSLydia Wang 	INIT_DELAYED_WORK(&spec->vt1708_hp_work, vt1708_update_hp_jack_state);
2902c577b8a1SJoseph Chan 	return 0;
2903c577b8a1SJoseph Chan }
2904c577b8a1SJoseph Chan 
2905ddd304d8STakashi Iwai static int patch_vt1709(struct hda_codec *codec)
2906c577b8a1SJoseph Chan {
2907c577b8a1SJoseph Chan 	struct via_spec *spec;
2908c577b8a1SJoseph Chan 	int err;
2909c577b8a1SJoseph Chan 
2910c577b8a1SJoseph Chan 	/* create a codec specific record */
29115b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
2912c577b8a1SJoseph Chan 	if (spec == NULL)
2913c577b8a1SJoseph Chan 		return -ENOMEM;
2914c577b8a1SJoseph Chan 
2915620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x18;
2916620e2b28STakashi Iwai 
291712daef65STakashi Iwai 	err = via_parse_auto_config(codec);
2918c577b8a1SJoseph Chan 	if (err < 0) {
2919c577b8a1SJoseph Chan 		via_free(codec);
2920c577b8a1SJoseph Chan 		return err;
2921c577b8a1SJoseph Chan 	}
2922c577b8a1SJoseph Chan 
2923c577b8a1SJoseph Chan 	codec->patch_ops = via_patch_ops;
2924c577b8a1SJoseph Chan 
2925f7278fd0SJosepch Chan 	return 0;
2926f7278fd0SJosepch Chan }
2927f7278fd0SJosepch Chan 
29283e95b9abSLydia Wang static void set_widgets_power_state_vt1708B(struct hda_codec *codec)
29293e95b9abSLydia Wang {
29303e95b9abSLydia Wang 	struct via_spec *spec = codec->spec;
29313e95b9abSLydia Wang 	int imux_is_smixer;
29323e95b9abSLydia Wang 	unsigned int parm;
29333e95b9abSLydia Wang 	int is_8ch = 0;
2934bc92df7fSLydia Wang 	if ((spec->codec_type != VT1708B_4CH) &&
2935bc92df7fSLydia Wang 	    (codec->vendor_id != 0x11064397))
29363e95b9abSLydia Wang 		is_8ch = 1;
29373e95b9abSLydia Wang 
29383e95b9abSLydia Wang 	/* SW0 (17h) = stereo mixer */
29393e95b9abSLydia Wang 	imux_is_smixer =
29403e95b9abSLydia Wang 	(snd_hda_codec_read(codec, 0x17, 0, AC_VERB_GET_CONNECT_SEL, 0x00)
29413e95b9abSLydia Wang 	 == ((spec->codec_type == VT1708S) ? 5 : 0));
29423e95b9abSLydia Wang 	/* inputs */
29433e95b9abSLydia Wang 	/* PW 1/2/5 (1ah/1bh/1eh) */
29443e95b9abSLydia Wang 	parm = AC_PWRST_D3;
29453e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1a, &parm);
29463e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1b, &parm);
29473e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1e, &parm);
29483e95b9abSLydia Wang 	if (imux_is_smixer)
29493e95b9abSLydia Wang 		parm = AC_PWRST_D0;
29503e95b9abSLydia Wang 	/* SW0 (17h), AIW 0/1 (13h/14h) */
2951054d867eSTakashi Iwai 	update_power_state(codec, 0x17, parm);
2952054d867eSTakashi Iwai 	update_power_state(codec, 0x13, parm);
2953054d867eSTakashi Iwai 	update_power_state(codec, 0x14, parm);
29543e95b9abSLydia Wang 
29553e95b9abSLydia Wang 	/* outputs */
29563e95b9abSLydia Wang 	/* PW0 (19h), SW1 (18h), AOW1 (11h) */
29573e95b9abSLydia Wang 	parm = AC_PWRST_D3;
29583e95b9abSLydia Wang 	set_pin_power_state(codec, 0x19, &parm);
29593e95b9abSLydia Wang 	if (spec->smart51_enabled)
29603e95b9abSLydia Wang 		set_pin_power_state(codec, 0x1b, &parm);
2961054d867eSTakashi Iwai 	update_power_state(codec, 0x18, parm);
2962054d867eSTakashi Iwai 	update_power_state(codec, 0x11, parm);
29633e95b9abSLydia Wang 
29643e95b9abSLydia Wang 	/* PW6 (22h), SW2 (26h), AOW2 (24h) */
29653e95b9abSLydia Wang 	if (is_8ch) {
29663e95b9abSLydia Wang 		parm = AC_PWRST_D3;
29673e95b9abSLydia Wang 		set_pin_power_state(codec, 0x22, &parm);
29683e95b9abSLydia Wang 		if (spec->smart51_enabled)
29693e95b9abSLydia Wang 			set_pin_power_state(codec, 0x1a, &parm);
2970054d867eSTakashi Iwai 		update_power_state(codec, 0x26, parm);
2971054d867eSTakashi Iwai 		update_power_state(codec, 0x24, parm);
2972bc92df7fSLydia Wang 	} else if (codec->vendor_id == 0x11064397) {
2973bc92df7fSLydia Wang 		/* PW7(23h), SW2(27h), AOW2(25h) */
2974bc92df7fSLydia Wang 		parm = AC_PWRST_D3;
2975bc92df7fSLydia Wang 		set_pin_power_state(codec, 0x23, &parm);
2976bc92df7fSLydia Wang 		if (spec->smart51_enabled)
2977bc92df7fSLydia Wang 			set_pin_power_state(codec, 0x1a, &parm);
2978054d867eSTakashi Iwai 		update_power_state(codec, 0x27, parm);
2979054d867eSTakashi Iwai 		update_power_state(codec, 0x25, parm);
29803e95b9abSLydia Wang 	}
29813e95b9abSLydia Wang 
29823e95b9abSLydia Wang 	/* PW 3/4/7 (1ch/1dh/23h) */
29833e95b9abSLydia Wang 	parm = AC_PWRST_D3;
29843e95b9abSLydia Wang 	/* force to D0 for internal Speaker */
29853e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1c, &parm);
29863e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1d, &parm);
29873e95b9abSLydia Wang 	if (is_8ch)
29883e95b9abSLydia Wang 		set_pin_power_state(codec, 0x23, &parm);
29893e95b9abSLydia Wang 
29903e95b9abSLydia Wang 	/* MW0 (16h), Sw3 (27h), AOW 0/3 (10h/25h) */
2991054d867eSTakashi Iwai 	update_power_state(codec, 0x16, imux_is_smixer ? AC_PWRST_D0 : parm);
2992054d867eSTakashi Iwai 	update_power_state(codec, 0x10, parm);
29933e95b9abSLydia Wang 	if (is_8ch) {
2994054d867eSTakashi Iwai 		update_power_state(codec, 0x25, parm);
2995054d867eSTakashi Iwai 		update_power_state(codec, 0x27, parm);
2996bc92df7fSLydia Wang 	} else if (codec->vendor_id == 0x11064397 && spec->hp_independent_mode)
2997054d867eSTakashi Iwai 		update_power_state(codec, 0x25, parm);
29983e95b9abSLydia Wang }
29993e95b9abSLydia Wang 
3000518bf3baSLydia Wang static int patch_vt1708S(struct hda_codec *codec);
3001ddd304d8STakashi Iwai static int patch_vt1708B(struct hda_codec *codec)
3002f7278fd0SJosepch Chan {
3003f7278fd0SJosepch Chan 	struct via_spec *spec;
3004f7278fd0SJosepch Chan 	int err;
3005f7278fd0SJosepch Chan 
3006518bf3baSLydia Wang 	if (get_codec_type(codec) == VT1708BCE)
3007518bf3baSLydia Wang 		return patch_vt1708S(codec);
3008ddd304d8STakashi Iwai 
3009f7278fd0SJosepch Chan 	/* create a codec specific record */
30105b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
3011f7278fd0SJosepch Chan 	if (spec == NULL)
3012f7278fd0SJosepch Chan 		return -ENOMEM;
3013f7278fd0SJosepch Chan 
3014620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x16;
3015620e2b28STakashi Iwai 
3016f7278fd0SJosepch Chan 	/* automatic parse from the BIOS config */
301712daef65STakashi Iwai 	err = via_parse_auto_config(codec);
3018f7278fd0SJosepch Chan 	if (err < 0) {
3019f7278fd0SJosepch Chan 		via_free(codec);
3020f7278fd0SJosepch Chan 		return err;
3021f7278fd0SJosepch Chan 	}
3022f7278fd0SJosepch Chan 
3023f7278fd0SJosepch Chan 	codec->patch_ops = via_patch_ops;
3024f7278fd0SJosepch Chan 
30253e95b9abSLydia Wang 	spec->set_widgets_power_state =  set_widgets_power_state_vt1708B;
30263e95b9abSLydia Wang 
3027f7278fd0SJosepch Chan 	return 0;
3028f7278fd0SJosepch Chan }
3029f7278fd0SJosepch Chan 
3030d949cac1SHarald Welte /* Patch for VT1708S */
3031096a8854STakashi Iwai static const struct hda_verb vt1708S_init_verbs[] = {
3032d7426329SHarald Welte 	/* Enable Mic Boost Volume backdoor */
3033d7426329SHarald Welte 	{0x1, 0xf98, 0x1},
3034bc7e7e5cSLydia Wang 	/* don't bybass mixer */
3035bc7e7e5cSLydia Wang 	{0x1, 0xf88, 0xc0},
3036d949cac1SHarald Welte 	{ }
3037d949cac1SHarald Welte };
3038d949cac1SHarald Welte 
30399da29271STakashi Iwai /* fill out digital output widgets; one for master and one for slave outputs */
30409da29271STakashi Iwai static void fill_dig_outs(struct hda_codec *codec)
30419da29271STakashi Iwai {
30429da29271STakashi Iwai 	struct via_spec *spec = codec->spec;
30439da29271STakashi Iwai 	int i;
30449da29271STakashi Iwai 
30459da29271STakashi Iwai 	for (i = 0; i < spec->autocfg.dig_outs; i++) {
30469da29271STakashi Iwai 		hda_nid_t nid;
30479da29271STakashi Iwai 		int conn;
30489da29271STakashi Iwai 
30499da29271STakashi Iwai 		nid = spec->autocfg.dig_out_pins[i];
30509da29271STakashi Iwai 		if (!nid)
30519da29271STakashi Iwai 			continue;
30529da29271STakashi Iwai 		conn = snd_hda_get_connections(codec, nid, &nid, 1);
30539da29271STakashi Iwai 		if (conn < 1)
30549da29271STakashi Iwai 			continue;
30559da29271STakashi Iwai 		if (!spec->multiout.dig_out_nid)
30569da29271STakashi Iwai 			spec->multiout.dig_out_nid = nid;
30579da29271STakashi Iwai 		else {
30589da29271STakashi Iwai 			spec->slave_dig_outs[0] = nid;
30599da29271STakashi Iwai 			break; /* at most two dig outs */
30609da29271STakashi Iwai 		}
30619da29271STakashi Iwai 	}
30629da29271STakashi Iwai }
30639da29271STakashi Iwai 
306412daef65STakashi Iwai static void fill_dig_in(struct hda_codec *codec)
3065d949cac1SHarald Welte {
3066d949cac1SHarald Welte 	struct via_spec *spec = codec->spec;
306712daef65STakashi Iwai 	hda_nid_t dig_nid;
306812daef65STakashi Iwai 	int i, err;
3069d949cac1SHarald Welte 
307012daef65STakashi Iwai 	if (!spec->autocfg.dig_in_pin)
307112daef65STakashi Iwai 		return;
3072d949cac1SHarald Welte 
307312daef65STakashi Iwai 	dig_nid = codec->start_nid;
307412daef65STakashi Iwai 	for (i = 0; i < codec->num_nodes; i++, dig_nid++) {
307512daef65STakashi Iwai 		unsigned int wcaps = get_wcaps(codec, dig_nid);
307612daef65STakashi Iwai 		if (get_wcaps_type(wcaps) != AC_WID_AUD_IN)
307712daef65STakashi Iwai 			continue;
307812daef65STakashi Iwai 		if (!(wcaps & AC_WCAP_DIGITAL))
307912daef65STakashi Iwai 			continue;
308012daef65STakashi Iwai 		if (!(wcaps & AC_WCAP_CONN_LIST))
308112daef65STakashi Iwai 			continue;
308212daef65STakashi Iwai 		err = get_connection_index(codec, dig_nid,
308312daef65STakashi Iwai 					   spec->autocfg.dig_in_pin);
308412daef65STakashi Iwai 		if (err >= 0) {
308512daef65STakashi Iwai 			spec->dig_in_nid = dig_nid;
308612daef65STakashi Iwai 			break;
308712daef65STakashi Iwai 		}
308812daef65STakashi Iwai 	}
3089d949cac1SHarald Welte }
3090d949cac1SHarald Welte 
30916369bcfcSLydia Wang static void override_mic_boost(struct hda_codec *codec, hda_nid_t pin,
30926369bcfcSLydia Wang 			       int offset, int num_steps, int step_size)
30936369bcfcSLydia Wang {
30946369bcfcSLydia Wang 	snd_hda_override_amp_caps(codec, pin, HDA_INPUT,
30956369bcfcSLydia Wang 				  (offset << AC_AMPCAP_OFFSET_SHIFT) |
30966369bcfcSLydia Wang 				  (num_steps << AC_AMPCAP_NUM_STEPS_SHIFT) |
30976369bcfcSLydia Wang 				  (step_size << AC_AMPCAP_STEP_SIZE_SHIFT) |
30986369bcfcSLydia Wang 				  (0 << AC_AMPCAP_MUTE_SHIFT));
30996369bcfcSLydia Wang }
31006369bcfcSLydia Wang 
3101d949cac1SHarald Welte static int patch_vt1708S(struct hda_codec *codec)
3102d949cac1SHarald Welte {
3103d949cac1SHarald Welte 	struct via_spec *spec;
3104d949cac1SHarald Welte 	int err;
3105d949cac1SHarald Welte 
3106d949cac1SHarald Welte 	/* create a codec specific record */
31075b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
3108d949cac1SHarald Welte 	if (spec == NULL)
3109d949cac1SHarald Welte 		return -ENOMEM;
3110d949cac1SHarald Welte 
3111620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x16;
3112d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x1a, 0, 3, 40);
3113d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x1e, 0, 3, 40);
3114620e2b28STakashi Iwai 
3115d949cac1SHarald Welte 	/* automatic parse from the BIOS config */
311612daef65STakashi Iwai 	err = via_parse_auto_config(codec);
3117d949cac1SHarald Welte 	if (err < 0) {
3118d949cac1SHarald Welte 		via_free(codec);
3119d949cac1SHarald Welte 		return err;
3120d949cac1SHarald Welte 	}
3121d949cac1SHarald Welte 
3122096a8854STakashi Iwai 	spec->init_verbs[spec->num_iverbs++] = vt1708S_init_verbs;
3123d949cac1SHarald Welte 
3124d949cac1SHarald Welte 	codec->patch_ops = via_patch_ops;
3125d949cac1SHarald Welte 
3126518bf3baSLydia Wang 	/* correct names for VT1708BCE */
3127518bf3baSLydia Wang 	if (get_codec_type(codec) == VT1708BCE)	{
3128518bf3baSLydia Wang 		kfree(codec->chip_name);
3129518bf3baSLydia Wang 		codec->chip_name = kstrdup("VT1708BCE", GFP_KERNEL);
3130518bf3baSLydia Wang 		snprintf(codec->bus->card->mixername,
3131518bf3baSLydia Wang 			 sizeof(codec->bus->card->mixername),
3132518bf3baSLydia Wang 			 "%s %s", codec->vendor_name, codec->chip_name);
3133970f630fSLydia Wang 	}
3134bc92df7fSLydia Wang 	/* correct names for VT1705 */
3135bc92df7fSLydia Wang 	if (codec->vendor_id == 0x11064397)	{
3136bc92df7fSLydia Wang 		kfree(codec->chip_name);
3137bc92df7fSLydia Wang 		codec->chip_name = kstrdup("VT1705", GFP_KERNEL);
3138bc92df7fSLydia Wang 		snprintf(codec->bus->card->mixername,
3139bc92df7fSLydia Wang 			 sizeof(codec->bus->card->mixername),
3140bc92df7fSLydia Wang 			 "%s %s", codec->vendor_name, codec->chip_name);
3141bc92df7fSLydia Wang 	}
31423e95b9abSLydia Wang 	spec->set_widgets_power_state =  set_widgets_power_state_vt1708B;
3143d949cac1SHarald Welte 	return 0;
3144d949cac1SHarald Welte }
3145d949cac1SHarald Welte 
3146d949cac1SHarald Welte /* Patch for VT1702 */
3147d949cac1SHarald Welte 
3148096a8854STakashi Iwai static const struct hda_verb vt1702_init_verbs[] = {
3149bc7e7e5cSLydia Wang 	/* mixer enable */
3150bc7e7e5cSLydia Wang 	{0x1, 0xF88, 0x3},
3151bc7e7e5cSLydia Wang 	/* GPIO 0~2 */
3152bc7e7e5cSLydia Wang 	{0x1, 0xF82, 0x3F},
3153d949cac1SHarald Welte 	{ }
3154d949cac1SHarald Welte };
3155d949cac1SHarald Welte 
31563e95b9abSLydia Wang static void set_widgets_power_state_vt1702(struct hda_codec *codec)
31573e95b9abSLydia Wang {
31583e95b9abSLydia Wang 	int imux_is_smixer =
31593e95b9abSLydia Wang 	snd_hda_codec_read(codec, 0x13, 0, AC_VERB_GET_CONNECT_SEL, 0x00) == 3;
31603e95b9abSLydia Wang 	unsigned int parm;
31613e95b9abSLydia Wang 	/* inputs */
31623e95b9abSLydia Wang 	/* PW 1/2/5 (14h/15h/18h) */
31633e95b9abSLydia Wang 	parm = AC_PWRST_D3;
31643e95b9abSLydia Wang 	set_pin_power_state(codec, 0x14, &parm);
31653e95b9abSLydia Wang 	set_pin_power_state(codec, 0x15, &parm);
31663e95b9abSLydia Wang 	set_pin_power_state(codec, 0x18, &parm);
31673e95b9abSLydia Wang 	if (imux_is_smixer)
31683e95b9abSLydia Wang 		parm = AC_PWRST_D0; /* SW0 (13h) = stereo mixer (idx 3) */
31693e95b9abSLydia Wang 	/* SW0 (13h), AIW 0/1/2 (12h/1fh/20h) */
3170054d867eSTakashi Iwai 	update_power_state(codec, 0x13, parm);
3171054d867eSTakashi Iwai 	update_power_state(codec, 0x12, parm);
3172054d867eSTakashi Iwai 	update_power_state(codec, 0x1f, parm);
3173054d867eSTakashi Iwai 	update_power_state(codec, 0x20, parm);
31743e95b9abSLydia Wang 
31753e95b9abSLydia Wang 	/* outputs */
31763e95b9abSLydia Wang 	/* PW 3/4 (16h/17h) */
31773e95b9abSLydia Wang 	parm = AC_PWRST_D3;
31783e95b9abSLydia Wang 	set_pin_power_state(codec, 0x17, &parm);
31793e95b9abSLydia Wang 	set_pin_power_state(codec, 0x16, &parm);
31803e95b9abSLydia Wang 	/* MW0 (1ah), AOW 0/1 (10h/1dh) */
3181054d867eSTakashi Iwai 	update_power_state(codec, 0x1a, imux_is_smixer ? AC_PWRST_D0 : parm);
3182054d867eSTakashi Iwai 	update_power_state(codec, 0x10, parm);
3183054d867eSTakashi Iwai 	update_power_state(codec, 0x1d, parm);
31843e95b9abSLydia Wang }
31853e95b9abSLydia Wang 
3186d949cac1SHarald Welte static int patch_vt1702(struct hda_codec *codec)
3187d949cac1SHarald Welte {
3188d949cac1SHarald Welte 	struct via_spec *spec;
3189d949cac1SHarald Welte 	int err;
3190d949cac1SHarald Welte 
3191d949cac1SHarald Welte 	/* create a codec specific record */
31925b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
3193d949cac1SHarald Welte 	if (spec == NULL)
3194d949cac1SHarald Welte 		return -ENOMEM;
3195d949cac1SHarald Welte 
3196620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x1a;
3197620e2b28STakashi Iwai 
319812daef65STakashi Iwai 	/* limit AA path volume to 0 dB */
319912daef65STakashi Iwai 	snd_hda_override_amp_caps(codec, 0x1A, HDA_INPUT,
320012daef65STakashi Iwai 				  (0x17 << AC_AMPCAP_OFFSET_SHIFT) |
320112daef65STakashi Iwai 				  (0x17 << AC_AMPCAP_NUM_STEPS_SHIFT) |
320212daef65STakashi Iwai 				  (0x5 << AC_AMPCAP_STEP_SIZE_SHIFT) |
320312daef65STakashi Iwai 				  (1 << AC_AMPCAP_MUTE_SHIFT));
320412daef65STakashi Iwai 
3205d949cac1SHarald Welte 	/* automatic parse from the BIOS config */
320612daef65STakashi Iwai 	err = via_parse_auto_config(codec);
3207d949cac1SHarald Welte 	if (err < 0) {
3208d949cac1SHarald Welte 		via_free(codec);
3209d949cac1SHarald Welte 		return err;
3210d949cac1SHarald Welte 	}
3211d949cac1SHarald Welte 
3212096a8854STakashi Iwai 	spec->init_verbs[spec->num_iverbs++] = vt1702_init_verbs;
3213d949cac1SHarald Welte 
3214d949cac1SHarald Welte 	codec->patch_ops = via_patch_ops;
3215d949cac1SHarald Welte 
32163e95b9abSLydia Wang 	spec->set_widgets_power_state =  set_widgets_power_state_vt1702;
3217d949cac1SHarald Welte 	return 0;
3218d949cac1SHarald Welte }
3219d949cac1SHarald Welte 
3220eb7188caSLydia Wang /* Patch for VT1718S */
3221eb7188caSLydia Wang 
3222096a8854STakashi Iwai static const struct hda_verb vt1718S_init_verbs[] = {
32234ab2d53aSLydia Wang 	/* Enable MW0 adjust Gain 5 */
32244ab2d53aSLydia Wang 	{0x1, 0xfb2, 0x10},
3225eb7188caSLydia Wang 	/* Enable Boost Volume backdoor */
3226eb7188caSLydia Wang 	{0x1, 0xf88, 0x8},
32275d41762aSTakashi Iwai 
3228eb7188caSLydia Wang 	{ }
3229eb7188caSLydia Wang };
3230eb7188caSLydia Wang 
32313e95b9abSLydia Wang static void set_widgets_power_state_vt1718S(struct hda_codec *codec)
32323e95b9abSLydia Wang {
32333e95b9abSLydia Wang 	struct via_spec *spec = codec->spec;
32343e95b9abSLydia Wang 	int imux_is_smixer;
32356162552bSTakashi Iwai 	unsigned int parm, parm2;
32363e95b9abSLydia Wang 	/* MUX6 (1eh) = stereo mixer */
32373e95b9abSLydia Wang 	imux_is_smixer =
32383e95b9abSLydia Wang 	snd_hda_codec_read(codec, 0x1e, 0, AC_VERB_GET_CONNECT_SEL, 0x00) == 5;
32393e95b9abSLydia Wang 	/* inputs */
32403e95b9abSLydia Wang 	/* PW 5/6/7 (29h/2ah/2bh) */
32413e95b9abSLydia Wang 	parm = AC_PWRST_D3;
32423e95b9abSLydia Wang 	set_pin_power_state(codec, 0x29, &parm);
32433e95b9abSLydia Wang 	set_pin_power_state(codec, 0x2a, &parm);
32443e95b9abSLydia Wang 	set_pin_power_state(codec, 0x2b, &parm);
32453e95b9abSLydia Wang 	if (imux_is_smixer)
32463e95b9abSLydia Wang 		parm = AC_PWRST_D0;
32473e95b9abSLydia Wang 	/* MUX6/7 (1eh/1fh), AIW 0/1 (10h/11h) */
3248054d867eSTakashi Iwai 	update_power_state(codec, 0x1e, parm);
3249054d867eSTakashi Iwai 	update_power_state(codec, 0x1f, parm);
3250054d867eSTakashi Iwai 	update_power_state(codec, 0x10, parm);
3251054d867eSTakashi Iwai 	update_power_state(codec, 0x11, parm);
32523e95b9abSLydia Wang 
32533e95b9abSLydia Wang 	/* outputs */
32543e95b9abSLydia Wang 	/* PW3 (27h), MW2 (1ah), AOW3 (bh) */
32553e95b9abSLydia Wang 	parm = AC_PWRST_D3;
32563e95b9abSLydia Wang 	set_pin_power_state(codec, 0x27, &parm);
3257054d867eSTakashi Iwai 	update_power_state(codec, 0x1a, parm);
32586162552bSTakashi Iwai 	parm2 = parm; /* for pin 0x0b */
32593e95b9abSLydia Wang 
32603e95b9abSLydia Wang 	/* PW2 (26h), AOW2 (ah) */
32613e95b9abSLydia Wang 	parm = AC_PWRST_D3;
32623e95b9abSLydia Wang 	set_pin_power_state(codec, 0x26, &parm);
32633e95b9abSLydia Wang 	if (spec->smart51_enabled)
32643e95b9abSLydia Wang 		set_pin_power_state(codec, 0x2b, &parm);
3265054d867eSTakashi Iwai 	update_power_state(codec, 0xa, parm);
32663e95b9abSLydia Wang 
32673e95b9abSLydia Wang 	/* PW0 (24h), AOW0 (8h) */
32683e95b9abSLydia Wang 	parm = AC_PWRST_D3;
32693e95b9abSLydia Wang 	set_pin_power_state(codec, 0x24, &parm);
32703e95b9abSLydia Wang 	if (!spec->hp_independent_mode) /* check for redirected HP */
32713e95b9abSLydia Wang 		set_pin_power_state(codec, 0x28, &parm);
3272054d867eSTakashi Iwai 	update_power_state(codec, 0x8, parm);
32736162552bSTakashi Iwai 	if (!spec->hp_independent_mode && parm2 != AC_PWRST_D3)
32746162552bSTakashi Iwai 		parm = parm2;
32756162552bSTakashi Iwai 	update_power_state(codec, 0xb, parm);
32763e95b9abSLydia Wang 	/* MW9 (21h), Mw2 (1ah), AOW0 (8h) */
3277054d867eSTakashi Iwai 	update_power_state(codec, 0x21, imux_is_smixer ? AC_PWRST_D0 : parm);
32783e95b9abSLydia Wang 
32793e95b9abSLydia Wang 	/* PW1 (25h), AOW1 (9h) */
32803e95b9abSLydia Wang 	parm = AC_PWRST_D3;
32813e95b9abSLydia Wang 	set_pin_power_state(codec, 0x25, &parm);
32823e95b9abSLydia Wang 	if (spec->smart51_enabled)
32833e95b9abSLydia Wang 		set_pin_power_state(codec, 0x2a, &parm);
3284054d867eSTakashi Iwai 	update_power_state(codec, 0x9, parm);
32853e95b9abSLydia Wang 
32863e95b9abSLydia Wang 	if (spec->hp_independent_mode) {
32873e95b9abSLydia Wang 		/* PW4 (28h), MW3 (1bh), MUX1(34h), AOW4 (ch) */
32883e95b9abSLydia Wang 		parm = AC_PWRST_D3;
32893e95b9abSLydia Wang 		set_pin_power_state(codec, 0x28, &parm);
3290054d867eSTakashi Iwai 		update_power_state(codec, 0x1b, parm);
3291054d867eSTakashi Iwai 		update_power_state(codec, 0x34, parm);
3292054d867eSTakashi Iwai 		update_power_state(codec, 0xc, parm);
32933e95b9abSLydia Wang 	}
32943e95b9abSLydia Wang }
32953e95b9abSLydia Wang 
329630b45033STakashi Iwai /* Add a connection to the primary DAC from AA-mixer for some codecs
329730b45033STakashi Iwai  * This isn't listed from the raw info, but the chip has a secret connection.
329830b45033STakashi Iwai  */
329930b45033STakashi Iwai static int add_secret_dac_path(struct hda_codec *codec)
330030b45033STakashi Iwai {
330130b45033STakashi Iwai 	struct via_spec *spec = codec->spec;
330230b45033STakashi Iwai 	int i, nums;
330330b45033STakashi Iwai 	hda_nid_t conn[8];
330430b45033STakashi Iwai 	hda_nid_t nid;
330530b45033STakashi Iwai 
330630b45033STakashi Iwai 	if (!spec->aa_mix_nid)
330730b45033STakashi Iwai 		return 0;
330830b45033STakashi Iwai 	nums = snd_hda_get_connections(codec, spec->aa_mix_nid, conn,
330930b45033STakashi Iwai 				       ARRAY_SIZE(conn) - 1);
331030b45033STakashi Iwai 	for (i = 0; i < nums; i++) {
331130b45033STakashi Iwai 		if (get_wcaps_type(get_wcaps(codec, conn[i])) == AC_WID_AUD_OUT)
331230b45033STakashi Iwai 			return 0;
331330b45033STakashi Iwai 	}
331430b45033STakashi Iwai 
331530b45033STakashi Iwai 	/* find the primary DAC and add to the connection list */
331630b45033STakashi Iwai 	nid = codec->start_nid;
331730b45033STakashi Iwai 	for (i = 0; i < codec->num_nodes; i++, nid++) {
331830b45033STakashi Iwai 		unsigned int caps = get_wcaps(codec, nid);
331930b45033STakashi Iwai 		if (get_wcaps_type(caps) == AC_WID_AUD_OUT &&
332030b45033STakashi Iwai 		    !(caps & AC_WCAP_DIGITAL)) {
332130b45033STakashi Iwai 			conn[nums++] = nid;
332230b45033STakashi Iwai 			return snd_hda_override_conn_list(codec,
332330b45033STakashi Iwai 							  spec->aa_mix_nid,
332430b45033STakashi Iwai 							  nums, conn);
332530b45033STakashi Iwai 		}
332630b45033STakashi Iwai 	}
332730b45033STakashi Iwai 	return 0;
332830b45033STakashi Iwai }
332930b45033STakashi Iwai 
333030b45033STakashi Iwai 
3331eb7188caSLydia Wang static int patch_vt1718S(struct hda_codec *codec)
3332eb7188caSLydia Wang {
3333eb7188caSLydia Wang 	struct via_spec *spec;
3334eb7188caSLydia Wang 	int err;
3335eb7188caSLydia Wang 
3336eb7188caSLydia Wang 	/* create a codec specific record */
33375b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
3338eb7188caSLydia Wang 	if (spec == NULL)
3339eb7188caSLydia Wang 		return -ENOMEM;
3340eb7188caSLydia Wang 
3341620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x21;
3342d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x2b, 0, 3, 40);
3343d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x29, 0, 3, 40);
334430b45033STakashi Iwai 	add_secret_dac_path(codec);
3345620e2b28STakashi Iwai 
3346eb7188caSLydia Wang 	/* automatic parse from the BIOS config */
334712daef65STakashi Iwai 	err = via_parse_auto_config(codec);
3348eb7188caSLydia Wang 	if (err < 0) {
3349eb7188caSLydia Wang 		via_free(codec);
3350eb7188caSLydia Wang 		return err;
3351eb7188caSLydia Wang 	}
3352eb7188caSLydia Wang 
3353096a8854STakashi Iwai 	spec->init_verbs[spec->num_iverbs++] = vt1718S_init_verbs;
3354eb7188caSLydia Wang 
3355eb7188caSLydia Wang 	codec->patch_ops = via_patch_ops;
3356eb7188caSLydia Wang 
33573e95b9abSLydia Wang 	spec->set_widgets_power_state =  set_widgets_power_state_vt1718S;
33583e95b9abSLydia Wang 
3359eb7188caSLydia Wang 	return 0;
3360eb7188caSLydia Wang }
3361f3db423dSLydia Wang 
3362f3db423dSLydia Wang /* Patch for VT1716S */
3363f3db423dSLydia Wang 
3364f3db423dSLydia Wang static int vt1716s_dmic_info(struct snd_kcontrol *kcontrol,
3365f3db423dSLydia Wang 			    struct snd_ctl_elem_info *uinfo)
3366f3db423dSLydia Wang {
3367f3db423dSLydia Wang 	uinfo->type = SNDRV_CTL_ELEM_TYPE_BOOLEAN;
3368f3db423dSLydia Wang 	uinfo->count = 1;
3369f3db423dSLydia Wang 	uinfo->value.integer.min = 0;
3370f3db423dSLydia Wang 	uinfo->value.integer.max = 1;
3371f3db423dSLydia Wang 	return 0;
3372f3db423dSLydia Wang }
3373f3db423dSLydia Wang 
3374f3db423dSLydia Wang static int vt1716s_dmic_get(struct snd_kcontrol *kcontrol,
3375f3db423dSLydia Wang 			   struct snd_ctl_elem_value *ucontrol)
3376f3db423dSLydia Wang {
3377f3db423dSLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
3378f3db423dSLydia Wang 	int index = 0;
3379f3db423dSLydia Wang 
3380f3db423dSLydia Wang 	index = snd_hda_codec_read(codec, 0x26, 0,
3381f3db423dSLydia Wang 					       AC_VERB_GET_CONNECT_SEL, 0);
3382f3db423dSLydia Wang 	if (index != -1)
3383f3db423dSLydia Wang 		*ucontrol->value.integer.value = index;
3384f3db423dSLydia Wang 
3385f3db423dSLydia Wang 	return 0;
3386f3db423dSLydia Wang }
3387f3db423dSLydia Wang 
3388f3db423dSLydia Wang static int vt1716s_dmic_put(struct snd_kcontrol *kcontrol,
3389f3db423dSLydia Wang 			   struct snd_ctl_elem_value *ucontrol)
3390f3db423dSLydia Wang {
3391f3db423dSLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
3392f3db423dSLydia Wang 	struct via_spec *spec = codec->spec;
3393f3db423dSLydia Wang 	int index = *ucontrol->value.integer.value;
3394f3db423dSLydia Wang 
3395f3db423dSLydia Wang 	snd_hda_codec_write(codec, 0x26, 0,
3396f3db423dSLydia Wang 					       AC_VERB_SET_CONNECT_SEL, index);
3397f3db423dSLydia Wang 	spec->dmic_enabled = index;
33983e95b9abSLydia Wang 	set_widgets_power_state(codec);
3399f3db423dSLydia Wang 	return 1;
3400f3db423dSLydia Wang }
3401f3db423dSLydia Wang 
340290dd48a1STakashi Iwai static const struct snd_kcontrol_new vt1716s_dmic_mixer[] = {
3403f3db423dSLydia Wang 	HDA_CODEC_VOLUME("Digital Mic Capture Volume", 0x22, 0x0, HDA_INPUT),
3404f3db423dSLydia Wang 	{
3405f3db423dSLydia Wang 	 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
3406f3db423dSLydia Wang 	 .name = "Digital Mic Capture Switch",
34075b0cb1d8SJaroslav Kysela 	 .subdevice = HDA_SUBDEV_NID_FLAG | 0x26,
3408f3db423dSLydia Wang 	 .count = 1,
3409f3db423dSLydia Wang 	 .info = vt1716s_dmic_info,
3410f3db423dSLydia Wang 	 .get = vt1716s_dmic_get,
3411f3db423dSLydia Wang 	 .put = vt1716s_dmic_put,
3412f3db423dSLydia Wang 	 },
3413f3db423dSLydia Wang 	{}			/* end */
3414f3db423dSLydia Wang };
3415f3db423dSLydia Wang 
3416f3db423dSLydia Wang 
3417f3db423dSLydia Wang /* mono-out mixer elements */
341890dd48a1STakashi Iwai static const struct snd_kcontrol_new vt1716S_mono_out_mixer[] = {
3419f3db423dSLydia Wang 	HDA_CODEC_MUTE("Mono Playback Switch", 0x2a, 0x0, HDA_OUTPUT),
3420f3db423dSLydia Wang 	{ } /* end */
3421f3db423dSLydia Wang };
3422f3db423dSLydia Wang 
3423096a8854STakashi Iwai static const struct hda_verb vt1716S_init_verbs[] = {
3424f3db423dSLydia Wang 	/* Enable Boost Volume backdoor */
3425f3db423dSLydia Wang 	{0x1, 0xf8a, 0x80},
3426f3db423dSLydia Wang 	/* don't bybass mixer */
3427f3db423dSLydia Wang 	{0x1, 0xf88, 0xc0},
3428f3db423dSLydia Wang 	/* Enable mono output */
3429f3db423dSLydia Wang 	{0x1, 0xf90, 0x08},
3430f3db423dSLydia Wang 	{ }
3431f3db423dSLydia Wang };
3432f3db423dSLydia Wang 
34333e95b9abSLydia Wang static void set_widgets_power_state_vt1716S(struct hda_codec *codec)
34343e95b9abSLydia Wang {
34353e95b9abSLydia Wang 	struct via_spec *spec = codec->spec;
34363e95b9abSLydia Wang 	int imux_is_smixer;
34373e95b9abSLydia Wang 	unsigned int parm;
34383e95b9abSLydia Wang 	unsigned int mono_out, present;
34393e95b9abSLydia Wang 	/* SW0 (17h) = stereo mixer */
34403e95b9abSLydia Wang 	imux_is_smixer =
34413e95b9abSLydia Wang 	(snd_hda_codec_read(codec, 0x17, 0,
34423e95b9abSLydia Wang 			    AC_VERB_GET_CONNECT_SEL, 0x00) ==  5);
34433e95b9abSLydia Wang 	/* inputs */
34443e95b9abSLydia Wang 	/* PW 1/2/5 (1ah/1bh/1eh) */
34453e95b9abSLydia Wang 	parm = AC_PWRST_D3;
34463e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1a, &parm);
34473e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1b, &parm);
34483e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1e, &parm);
34493e95b9abSLydia Wang 	if (imux_is_smixer)
34503e95b9abSLydia Wang 		parm = AC_PWRST_D0;
34513e95b9abSLydia Wang 	/* SW0 (17h), AIW0(13h) */
3452054d867eSTakashi Iwai 	update_power_state(codec, 0x17, parm);
3453054d867eSTakashi Iwai 	update_power_state(codec, 0x13, parm);
34543e95b9abSLydia Wang 
34553e95b9abSLydia Wang 	parm = AC_PWRST_D3;
34563e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1e, &parm);
34573e95b9abSLydia Wang 	/* PW11 (22h) */
34583e95b9abSLydia Wang 	if (spec->dmic_enabled)
34593e95b9abSLydia Wang 		set_pin_power_state(codec, 0x22, &parm);
34603e95b9abSLydia Wang 	else
3461054d867eSTakashi Iwai 		update_power_state(codec, 0x22, AC_PWRST_D3);
34623e95b9abSLydia Wang 
34633e95b9abSLydia Wang 	/* SW2(26h), AIW1(14h) */
3464054d867eSTakashi Iwai 	update_power_state(codec, 0x26, parm);
3465054d867eSTakashi Iwai 	update_power_state(codec, 0x14, parm);
34663e95b9abSLydia Wang 
34673e95b9abSLydia Wang 	/* outputs */
34683e95b9abSLydia Wang 	/* PW0 (19h), SW1 (18h), AOW1 (11h) */
34693e95b9abSLydia Wang 	parm = AC_PWRST_D3;
34703e95b9abSLydia Wang 	set_pin_power_state(codec, 0x19, &parm);
34713e95b9abSLydia Wang 	/* Smart 5.1 PW2(1bh) */
34723e95b9abSLydia Wang 	if (spec->smart51_enabled)
34733e95b9abSLydia Wang 		set_pin_power_state(codec, 0x1b, &parm);
3474054d867eSTakashi Iwai 	update_power_state(codec, 0x18, parm);
3475054d867eSTakashi Iwai 	update_power_state(codec, 0x11, parm);
34763e95b9abSLydia Wang 
34773e95b9abSLydia Wang 	/* PW7 (23h), SW3 (27h), AOW3 (25h) */
34783e95b9abSLydia Wang 	parm = AC_PWRST_D3;
34793e95b9abSLydia Wang 	set_pin_power_state(codec, 0x23, &parm);
34803e95b9abSLydia Wang 	/* Smart 5.1 PW1(1ah) */
34813e95b9abSLydia Wang 	if (spec->smart51_enabled)
34823e95b9abSLydia Wang 		set_pin_power_state(codec, 0x1a, &parm);
3483054d867eSTakashi Iwai 	update_power_state(codec, 0x27, parm);
34843e95b9abSLydia Wang 
34853e95b9abSLydia Wang 	/* Smart 5.1 PW5(1eh) */
34863e95b9abSLydia Wang 	if (spec->smart51_enabled)
34873e95b9abSLydia Wang 		set_pin_power_state(codec, 0x1e, &parm);
3488054d867eSTakashi Iwai 	update_power_state(codec, 0x25, parm);
34893e95b9abSLydia Wang 
34903e95b9abSLydia Wang 	/* Mono out */
34913e95b9abSLydia Wang 	/* SW4(28h)->MW1(29h)-> PW12 (2ah)*/
34923e95b9abSLydia Wang 	present = snd_hda_jack_detect(codec, 0x1c);
34933e95b9abSLydia Wang 
34943e95b9abSLydia Wang 	if (present)
34953e95b9abSLydia Wang 		mono_out = 0;
34963e95b9abSLydia Wang 	else {
34973e95b9abSLydia Wang 		present = snd_hda_jack_detect(codec, 0x1d);
34983e95b9abSLydia Wang 		if (!spec->hp_independent_mode && present)
34993e95b9abSLydia Wang 			mono_out = 0;
35003e95b9abSLydia Wang 		else
35013e95b9abSLydia Wang 			mono_out = 1;
35023e95b9abSLydia Wang 	}
35033e95b9abSLydia Wang 	parm = mono_out ? AC_PWRST_D0 : AC_PWRST_D3;
3504054d867eSTakashi Iwai 	update_power_state(codec, 0x28, parm);
3505054d867eSTakashi Iwai 	update_power_state(codec, 0x29, parm);
3506054d867eSTakashi Iwai 	update_power_state(codec, 0x2a, parm);
35073e95b9abSLydia Wang 
35083e95b9abSLydia Wang 	/* PW 3/4 (1ch/1dh) */
35093e95b9abSLydia Wang 	parm = AC_PWRST_D3;
35103e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1c, &parm);
35113e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1d, &parm);
35123e95b9abSLydia Wang 	/* HP Independent Mode, power on AOW3 */
35133e95b9abSLydia Wang 	if (spec->hp_independent_mode)
3514054d867eSTakashi Iwai 		update_power_state(codec, 0x25, parm);
35153e95b9abSLydia Wang 
35163e95b9abSLydia Wang 	/* force to D0 for internal Speaker */
35173e95b9abSLydia Wang 	/* MW0 (16h), AOW0 (10h) */
3518054d867eSTakashi Iwai 	update_power_state(codec, 0x16, imux_is_smixer ? AC_PWRST_D0 : parm);
3519054d867eSTakashi Iwai 	update_power_state(codec, 0x10, mono_out ? AC_PWRST_D0 : parm);
35203e95b9abSLydia Wang }
35213e95b9abSLydia Wang 
3522f3db423dSLydia Wang static int patch_vt1716S(struct hda_codec *codec)
3523f3db423dSLydia Wang {
3524f3db423dSLydia Wang 	struct via_spec *spec;
3525f3db423dSLydia Wang 	int err;
3526f3db423dSLydia Wang 
3527f3db423dSLydia Wang 	/* create a codec specific record */
35285b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
3529f3db423dSLydia Wang 	if (spec == NULL)
3530f3db423dSLydia Wang 		return -ENOMEM;
3531f3db423dSLydia Wang 
3532620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x16;
3533d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x1a, 0, 3, 40);
3534d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x1e, 0, 3, 40);
3535620e2b28STakashi Iwai 
3536f3db423dSLydia Wang 	/* automatic parse from the BIOS config */
353712daef65STakashi Iwai 	err = via_parse_auto_config(codec);
3538f3db423dSLydia Wang 	if (err < 0) {
3539f3db423dSLydia Wang 		via_free(codec);
3540f3db423dSLydia Wang 		return err;
3541f3db423dSLydia Wang 	}
3542f3db423dSLydia Wang 
3543096a8854STakashi Iwai 	spec->init_verbs[spec->num_iverbs++]  = vt1716S_init_verbs;
3544f3db423dSLydia Wang 
3545f3db423dSLydia Wang 	spec->mixers[spec->num_mixers] = vt1716s_dmic_mixer;
3546f3db423dSLydia Wang 	spec->num_mixers++;
3547f3db423dSLydia Wang 
3548f3db423dSLydia Wang 	spec->mixers[spec->num_mixers++] = vt1716S_mono_out_mixer;
3549f3db423dSLydia Wang 
3550f3db423dSLydia Wang 	codec->patch_ops = via_patch_ops;
3551f3db423dSLydia Wang 
35523e95b9abSLydia Wang 	spec->set_widgets_power_state = set_widgets_power_state_vt1716S;
3553f3db423dSLydia Wang 	return 0;
3554f3db423dSLydia Wang }
355525eaba2fSLydia Wang 
355625eaba2fSLydia Wang /* for vt2002P */
355725eaba2fSLydia Wang 
3558096a8854STakashi Iwai static const struct hda_verb vt2002P_init_verbs[] = {
3559eadb9a80SLydia Wang 	/* Class-D speaker related verbs */
3560eadb9a80SLydia Wang 	{0x1, 0xfe0, 0x4},
3561eadb9a80SLydia Wang 	{0x1, 0xfe9, 0x80},
3562eadb9a80SLydia Wang 	{0x1, 0xfe2, 0x22},
356325eaba2fSLydia Wang 	/* Enable Boost Volume backdoor */
356425eaba2fSLydia Wang 	{0x1, 0xfb9, 0x24},
356525eaba2fSLydia Wang 	/* Enable AOW0 to MW9 */
356625eaba2fSLydia Wang 	{0x1, 0xfb8, 0x88},
356725eaba2fSLydia Wang 	{ }
356825eaba2fSLydia Wang };
35694a918ffeSTakashi Iwai 
3570096a8854STakashi Iwai static const struct hda_verb vt1802_init_verbs[] = {
357111890956SLydia Wang 	/* Enable Boost Volume backdoor */
357211890956SLydia Wang 	{0x1, 0xfb9, 0x24},
357311890956SLydia Wang 	/* Enable AOW0 to MW9 */
357411890956SLydia Wang 	{0x1, 0xfb8, 0x88},
357511890956SLydia Wang 	{ }
357611890956SLydia Wang };
357725eaba2fSLydia Wang 
35783e95b9abSLydia Wang static void set_widgets_power_state_vt2002P(struct hda_codec *codec)
35793e95b9abSLydia Wang {
35803e95b9abSLydia Wang 	struct via_spec *spec = codec->spec;
35813e95b9abSLydia Wang 	int imux_is_smixer;
35823e95b9abSLydia Wang 	unsigned int parm;
35833e95b9abSLydia Wang 	unsigned int present;
35843e95b9abSLydia Wang 	/* MUX9 (1eh) = stereo mixer */
35853e95b9abSLydia Wang 	imux_is_smixer =
35863e95b9abSLydia Wang 	snd_hda_codec_read(codec, 0x1e, 0, AC_VERB_GET_CONNECT_SEL, 0x00) == 3;
35873e95b9abSLydia Wang 	/* inputs */
35883e95b9abSLydia Wang 	/* PW 5/6/7 (29h/2ah/2bh) */
35893e95b9abSLydia Wang 	parm = AC_PWRST_D3;
35903e95b9abSLydia Wang 	set_pin_power_state(codec, 0x29, &parm);
35913e95b9abSLydia Wang 	set_pin_power_state(codec, 0x2a, &parm);
35923e95b9abSLydia Wang 	set_pin_power_state(codec, 0x2b, &parm);
35933e95b9abSLydia Wang 	parm = AC_PWRST_D0;
35943e95b9abSLydia Wang 	/* MUX9/10 (1eh/1fh), AIW 0/1 (10h/11h) */
3595054d867eSTakashi Iwai 	update_power_state(codec, 0x1e, parm);
3596054d867eSTakashi Iwai 	update_power_state(codec, 0x1f, parm);
3597054d867eSTakashi Iwai 	update_power_state(codec, 0x10, parm);
3598054d867eSTakashi Iwai 	update_power_state(codec, 0x11, parm);
35993e95b9abSLydia Wang 
36003e95b9abSLydia Wang 	/* outputs */
36013e95b9abSLydia Wang 	/* AOW0 (8h)*/
3602054d867eSTakashi Iwai 	update_power_state(codec, 0x8, parm);
36033e95b9abSLydia Wang 
360411890956SLydia Wang 	if (spec->codec_type == VT1802) {
360511890956SLydia Wang 		/* PW4 (28h), MW4 (18h), MUX4(38h) */
360611890956SLydia Wang 		parm = AC_PWRST_D3;
360711890956SLydia Wang 		set_pin_power_state(codec, 0x28, &parm);
3608054d867eSTakashi Iwai 		update_power_state(codec, 0x18, parm);
3609054d867eSTakashi Iwai 		update_power_state(codec, 0x38, parm);
361011890956SLydia Wang 	} else {
36113e95b9abSLydia Wang 		/* PW4 (26h), MW4 (1ch), MUX4(37h) */
36123e95b9abSLydia Wang 		parm = AC_PWRST_D3;
36133e95b9abSLydia Wang 		set_pin_power_state(codec, 0x26, &parm);
3614054d867eSTakashi Iwai 		update_power_state(codec, 0x1c, parm);
3615054d867eSTakashi Iwai 		update_power_state(codec, 0x37, parm);
361611890956SLydia Wang 	}
36173e95b9abSLydia Wang 
361811890956SLydia Wang 	if (spec->codec_type == VT1802) {
361911890956SLydia Wang 		/* PW1 (25h), MW1 (15h), MUX1(35h), AOW1 (9h) */
362011890956SLydia Wang 		parm = AC_PWRST_D3;
362111890956SLydia Wang 		set_pin_power_state(codec, 0x25, &parm);
3622054d867eSTakashi Iwai 		update_power_state(codec, 0x15, parm);
3623054d867eSTakashi Iwai 		update_power_state(codec, 0x35, parm);
362411890956SLydia Wang 	} else {
36253e95b9abSLydia Wang 		/* PW1 (25h), MW1 (19h), MUX1(35h), AOW1 (9h) */
36263e95b9abSLydia Wang 		parm = AC_PWRST_D3;
36273e95b9abSLydia Wang 		set_pin_power_state(codec, 0x25, &parm);
3628054d867eSTakashi Iwai 		update_power_state(codec, 0x19, parm);
3629054d867eSTakashi Iwai 		update_power_state(codec, 0x35, parm);
363011890956SLydia Wang 	}
36313e95b9abSLydia Wang 
36323e95b9abSLydia Wang 	if (spec->hp_independent_mode)
3633054d867eSTakashi Iwai 		update_power_state(codec, 0x9, AC_PWRST_D0);
36343e95b9abSLydia Wang 
36353e95b9abSLydia Wang 	/* Class-D */
36363e95b9abSLydia Wang 	/* PW0 (24h), MW0(18h/14h), MUX0(34h) */
36373e95b9abSLydia Wang 	present = snd_hda_jack_detect(codec, 0x25);
36383e95b9abSLydia Wang 
36393e95b9abSLydia Wang 	parm = AC_PWRST_D3;
36403e95b9abSLydia Wang 	set_pin_power_state(codec, 0x24, &parm);
36413e95b9abSLydia Wang 	parm = present ? AC_PWRST_D3 : AC_PWRST_D0;
364211890956SLydia Wang 	if (spec->codec_type == VT1802)
3643054d867eSTakashi Iwai 		update_power_state(codec, 0x14, parm);
364411890956SLydia Wang 	else
3645054d867eSTakashi Iwai 		update_power_state(codec, 0x18, parm);
3646054d867eSTakashi Iwai 	update_power_state(codec, 0x34, parm);
36473e95b9abSLydia Wang 
36483e95b9abSLydia Wang 	/* Mono Out */
36493e95b9abSLydia Wang 	present = snd_hda_jack_detect(codec, 0x26);
36503e95b9abSLydia Wang 
36513e95b9abSLydia Wang 	parm = present ? AC_PWRST_D3 : AC_PWRST_D0;
365211890956SLydia Wang 	if (spec->codec_type == VT1802) {
365311890956SLydia Wang 		/* PW15 (33h), MW8(1ch), MUX8(3ch) */
3654054d867eSTakashi Iwai 		update_power_state(codec, 0x33, parm);
3655054d867eSTakashi Iwai 		update_power_state(codec, 0x1c, parm);
3656054d867eSTakashi Iwai 		update_power_state(codec, 0x3c, parm);
365711890956SLydia Wang 	} else {
36583e95b9abSLydia Wang 		/* PW15 (31h), MW8(17h), MUX8(3bh) */
3659054d867eSTakashi Iwai 		update_power_state(codec, 0x31, parm);
3660054d867eSTakashi Iwai 		update_power_state(codec, 0x17, parm);
3661054d867eSTakashi Iwai 		update_power_state(codec, 0x3b, parm);
366211890956SLydia Wang 	}
36633e95b9abSLydia Wang 	/* MW9 (21h) */
36643e95b9abSLydia Wang 	if (imux_is_smixer || !is_aa_path_mute(codec))
3665054d867eSTakashi Iwai 		update_power_state(codec, 0x21, AC_PWRST_D0);
36663e95b9abSLydia Wang 	else
3667054d867eSTakashi Iwai 		update_power_state(codec, 0x21, AC_PWRST_D3);
36683e95b9abSLydia Wang }
366925eaba2fSLydia Wang 
36704b527b65SDavid Henningsson /*
36714b527b65SDavid Henningsson  * pin fix-up
36724b527b65SDavid Henningsson  */
36734b527b65SDavid Henningsson enum {
36744b527b65SDavid Henningsson 	VIA_FIXUP_INTMIC_BOOST,
36754b527b65SDavid Henningsson };
36764b527b65SDavid Henningsson 
36774b527b65SDavid Henningsson static void via_fixup_intmic_boost(struct hda_codec *codec,
36784b527b65SDavid Henningsson 				  const struct hda_fixup *fix, int action)
36794b527b65SDavid Henningsson {
36804b527b65SDavid Henningsson 	if (action == HDA_FIXUP_ACT_PRE_PROBE)
36814b527b65SDavid Henningsson 		override_mic_boost(codec, 0x30, 0, 2, 40);
36824b527b65SDavid Henningsson }
36834b527b65SDavid Henningsson 
36844b527b65SDavid Henningsson static const struct hda_fixup via_fixups[] = {
36854b527b65SDavid Henningsson 	[VIA_FIXUP_INTMIC_BOOST] = {
36864b527b65SDavid Henningsson 		.type = HDA_FIXUP_FUNC,
36874b527b65SDavid Henningsson 		.v.func = via_fixup_intmic_boost,
36884b527b65SDavid Henningsson 	},
36894b527b65SDavid Henningsson };
36904b527b65SDavid Henningsson 
36914b527b65SDavid Henningsson static const struct snd_pci_quirk vt2002p_fixups[] = {
36924b527b65SDavid Henningsson 	SND_PCI_QUIRK(0x1043, 0x8532, "Asus X202E", VIA_FIXUP_INTMIC_BOOST),
36934b527b65SDavid Henningsson 	{}
36944b527b65SDavid Henningsson };
36954b527b65SDavid Henningsson 
369625eaba2fSLydia Wang /* patch for vt2002P */
369725eaba2fSLydia Wang static int patch_vt2002P(struct hda_codec *codec)
369825eaba2fSLydia Wang {
369925eaba2fSLydia Wang 	struct via_spec *spec;
370025eaba2fSLydia Wang 	int err;
370125eaba2fSLydia Wang 
370225eaba2fSLydia Wang 	/* create a codec specific record */
37035b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
370425eaba2fSLydia Wang 	if (spec == NULL)
370525eaba2fSLydia Wang 		return -ENOMEM;
370625eaba2fSLydia Wang 
3707620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x21;
3708d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x2b, 0, 3, 40);
3709d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x29, 0, 3, 40);
371030b45033STakashi Iwai 	add_secret_dac_path(codec);
3711620e2b28STakashi Iwai 
37124b527b65SDavid Henningsson 	snd_hda_pick_fixup(codec, NULL, vt2002p_fixups, via_fixups);
37134b527b65SDavid Henningsson 	snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
37144b527b65SDavid Henningsson 
371525eaba2fSLydia Wang 	/* automatic parse from the BIOS config */
371612daef65STakashi Iwai 	err = via_parse_auto_config(codec);
371725eaba2fSLydia Wang 	if (err < 0) {
371825eaba2fSLydia Wang 		via_free(codec);
371925eaba2fSLydia Wang 		return err;
372025eaba2fSLydia Wang 	}
372125eaba2fSLydia Wang 
372211890956SLydia Wang 	if (spec->codec_type == VT1802)
37234a918ffeSTakashi Iwai 		spec->init_verbs[spec->num_iverbs++] = vt1802_init_verbs;
372411890956SLydia Wang 	else
37254a918ffeSTakashi Iwai 		spec->init_verbs[spec->num_iverbs++] = vt2002P_init_verbs;
372611890956SLydia Wang 
372725eaba2fSLydia Wang 	codec->patch_ops = via_patch_ops;
372825eaba2fSLydia Wang 
37293e95b9abSLydia Wang 	spec->set_widgets_power_state =  set_widgets_power_state_vt2002P;
373025eaba2fSLydia Wang 	return 0;
373125eaba2fSLydia Wang }
3732ab6734e7SLydia Wang 
3733ab6734e7SLydia Wang /* for vt1812 */
3734ab6734e7SLydia Wang 
3735096a8854STakashi Iwai static const struct hda_verb vt1812_init_verbs[] = {
3736ab6734e7SLydia Wang 	/* Enable Boost Volume backdoor */
3737ab6734e7SLydia Wang 	{0x1, 0xfb9, 0x24},
3738ab6734e7SLydia Wang 	/* Enable AOW0 to MW9 */
3739ab6734e7SLydia Wang 	{0x1, 0xfb8, 0xa8},
3740ab6734e7SLydia Wang 	{ }
3741ab6734e7SLydia Wang };
3742ab6734e7SLydia Wang 
37433e95b9abSLydia Wang static void set_widgets_power_state_vt1812(struct hda_codec *codec)
37443e95b9abSLydia Wang {
37453e95b9abSLydia Wang 	struct via_spec *spec = codec->spec;
37463e95b9abSLydia Wang 	unsigned int parm;
37473e95b9abSLydia Wang 	unsigned int present;
37483e95b9abSLydia Wang 	/* inputs */
37493e95b9abSLydia Wang 	/* PW 5/6/7 (29h/2ah/2bh) */
37503e95b9abSLydia Wang 	parm = AC_PWRST_D3;
37513e95b9abSLydia Wang 	set_pin_power_state(codec, 0x29, &parm);
37523e95b9abSLydia Wang 	set_pin_power_state(codec, 0x2a, &parm);
37533e95b9abSLydia Wang 	set_pin_power_state(codec, 0x2b, &parm);
37543e95b9abSLydia Wang 	parm = AC_PWRST_D0;
37553e95b9abSLydia Wang 	/* MUX10/11 (1eh/1fh), AIW 0/1 (10h/11h) */
3756054d867eSTakashi Iwai 	update_power_state(codec, 0x1e, parm);
3757054d867eSTakashi Iwai 	update_power_state(codec, 0x1f, parm);
3758054d867eSTakashi Iwai 	update_power_state(codec, 0x10, parm);
3759054d867eSTakashi Iwai 	update_power_state(codec, 0x11, parm);
37603e95b9abSLydia Wang 
37613e95b9abSLydia Wang 	/* outputs */
37623e95b9abSLydia Wang 	/* AOW0 (8h)*/
3763054d867eSTakashi Iwai 	update_power_state(codec, 0x8, AC_PWRST_D0);
37643e95b9abSLydia Wang 
37653e95b9abSLydia Wang 	/* PW4 (28h), MW4 (18h), MUX4(38h) */
37663e95b9abSLydia Wang 	parm = AC_PWRST_D3;
37673e95b9abSLydia Wang 	set_pin_power_state(codec, 0x28, &parm);
3768054d867eSTakashi Iwai 	update_power_state(codec, 0x18, parm);
3769054d867eSTakashi Iwai 	update_power_state(codec, 0x38, parm);
37703e95b9abSLydia Wang 
37713e95b9abSLydia Wang 	/* PW1 (25h), MW1 (15h), MUX1(35h), AOW1 (9h) */
37723e95b9abSLydia Wang 	parm = AC_PWRST_D3;
37733e95b9abSLydia Wang 	set_pin_power_state(codec, 0x25, &parm);
3774054d867eSTakashi Iwai 	update_power_state(codec, 0x15, parm);
3775054d867eSTakashi Iwai 	update_power_state(codec, 0x35, parm);
37763e95b9abSLydia Wang 	if (spec->hp_independent_mode)
3777054d867eSTakashi Iwai 		update_power_state(codec, 0x9, AC_PWRST_D0);
37783e95b9abSLydia Wang 
37793e95b9abSLydia Wang 	/* Internal Speaker */
37803e95b9abSLydia Wang 	/* PW0 (24h), MW0(14h), MUX0(34h) */
37813e95b9abSLydia Wang 	present = snd_hda_jack_detect(codec, 0x25);
37823e95b9abSLydia Wang 
37833e95b9abSLydia Wang 	parm = AC_PWRST_D3;
37843e95b9abSLydia Wang 	set_pin_power_state(codec, 0x24, &parm);
37853e95b9abSLydia Wang 	if (present) {
3786054d867eSTakashi Iwai 		update_power_state(codec, 0x14, AC_PWRST_D3);
3787054d867eSTakashi Iwai 		update_power_state(codec, 0x34, AC_PWRST_D3);
37883e95b9abSLydia Wang 	} else {
3789054d867eSTakashi Iwai 		update_power_state(codec, 0x14, AC_PWRST_D0);
3790054d867eSTakashi Iwai 		update_power_state(codec, 0x34, AC_PWRST_D0);
37913e95b9abSLydia Wang 	}
37923e95b9abSLydia Wang 
37933e95b9abSLydia Wang 
37943e95b9abSLydia Wang 	/* Mono Out */
37953e95b9abSLydia Wang 	/* PW13 (31h), MW13(1ch), MUX13(3ch), MW14(3eh) */
37963e95b9abSLydia Wang 	present = snd_hda_jack_detect(codec, 0x28);
37973e95b9abSLydia Wang 
37983e95b9abSLydia Wang 	parm = AC_PWRST_D3;
37993e95b9abSLydia Wang 	set_pin_power_state(codec, 0x31, &parm);
38003e95b9abSLydia Wang 	if (present) {
3801054d867eSTakashi Iwai 		update_power_state(codec, 0x1c, AC_PWRST_D3);
3802054d867eSTakashi Iwai 		update_power_state(codec, 0x3c, AC_PWRST_D3);
3803054d867eSTakashi Iwai 		update_power_state(codec, 0x3e, AC_PWRST_D3);
38043e95b9abSLydia Wang 	} else {
3805054d867eSTakashi Iwai 		update_power_state(codec, 0x1c, AC_PWRST_D0);
3806054d867eSTakashi Iwai 		update_power_state(codec, 0x3c, AC_PWRST_D0);
3807054d867eSTakashi Iwai 		update_power_state(codec, 0x3e, AC_PWRST_D0);
38083e95b9abSLydia Wang 	}
38093e95b9abSLydia Wang 
38103e95b9abSLydia Wang 	/* PW15 (33h), MW15 (1dh), MUX15(3dh) */
38113e95b9abSLydia Wang 	parm = AC_PWRST_D3;
38123e95b9abSLydia Wang 	set_pin_power_state(codec, 0x33, &parm);
3813054d867eSTakashi Iwai 	update_power_state(codec, 0x1d, parm);
3814054d867eSTakashi Iwai 	update_power_state(codec, 0x3d, parm);
38153e95b9abSLydia Wang 
38163e95b9abSLydia Wang }
3817ab6734e7SLydia Wang 
3818ab6734e7SLydia Wang /* patch for vt1812 */
3819ab6734e7SLydia Wang static int patch_vt1812(struct hda_codec *codec)
3820ab6734e7SLydia Wang {
3821ab6734e7SLydia Wang 	struct via_spec *spec;
3822ab6734e7SLydia Wang 	int err;
3823ab6734e7SLydia Wang 
3824ab6734e7SLydia Wang 	/* create a codec specific record */
38255b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
3826ab6734e7SLydia Wang 	if (spec == NULL)
3827ab6734e7SLydia Wang 		return -ENOMEM;
3828ab6734e7SLydia Wang 
3829620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x21;
3830d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x2b, 0, 3, 40);
3831d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x29, 0, 3, 40);
383230b45033STakashi Iwai 	add_secret_dac_path(codec);
3833620e2b28STakashi Iwai 
3834ab6734e7SLydia Wang 	/* automatic parse from the BIOS config */
383512daef65STakashi Iwai 	err = via_parse_auto_config(codec);
3836ab6734e7SLydia Wang 	if (err < 0) {
3837ab6734e7SLydia Wang 		via_free(codec);
3838ab6734e7SLydia Wang 		return err;
3839ab6734e7SLydia Wang 	}
3840ab6734e7SLydia Wang 
3841096a8854STakashi Iwai 	spec->init_verbs[spec->num_iverbs++]  = vt1812_init_verbs;
3842ab6734e7SLydia Wang 
3843ab6734e7SLydia Wang 	codec->patch_ops = via_patch_ops;
3844ab6734e7SLydia Wang 
38453e95b9abSLydia Wang 	spec->set_widgets_power_state =  set_widgets_power_state_vt1812;
3846ab6734e7SLydia Wang 	return 0;
3847ab6734e7SLydia Wang }
3848ab6734e7SLydia Wang 
3849c577b8a1SJoseph Chan /*
3850c577b8a1SJoseph Chan  * patch entries
3851c577b8a1SJoseph Chan  */
385290dd48a1STakashi Iwai static const struct hda_codec_preset snd_hda_preset_via[] = {
38533218c178STakashi Iwai 	{ .id = 0x11061708, .name = "VT1708", .patch = patch_vt1708},
38543218c178STakashi Iwai 	{ .id = 0x11061709, .name = "VT1708", .patch = patch_vt1708},
38553218c178STakashi Iwai 	{ .id = 0x1106170a, .name = "VT1708", .patch = patch_vt1708},
38563218c178STakashi Iwai 	{ .id = 0x1106170b, .name = "VT1708", .patch = patch_vt1708},
38573218c178STakashi Iwai 	{ .id = 0x1106e710, .name = "VT1709 10-Ch",
3858ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
38593218c178STakashi Iwai 	{ .id = 0x1106e711, .name = "VT1709 10-Ch",
3860ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
38613218c178STakashi Iwai 	{ .id = 0x1106e712, .name = "VT1709 10-Ch",
3862ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
38633218c178STakashi Iwai 	{ .id = 0x1106e713, .name = "VT1709 10-Ch",
3864ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
38653218c178STakashi Iwai 	{ .id = 0x1106e714, .name = "VT1709 6-Ch",
3866ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
38673218c178STakashi Iwai 	{ .id = 0x1106e715, .name = "VT1709 6-Ch",
3868ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
38693218c178STakashi Iwai 	{ .id = 0x1106e716, .name = "VT1709 6-Ch",
3870ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
38713218c178STakashi Iwai 	{ .id = 0x1106e717, .name = "VT1709 6-Ch",
3872ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
38733218c178STakashi Iwai 	{ .id = 0x1106e720, .name = "VT1708B 8-Ch",
3874ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
38753218c178STakashi Iwai 	{ .id = 0x1106e721, .name = "VT1708B 8-Ch",
3876ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
38773218c178STakashi Iwai 	{ .id = 0x1106e722, .name = "VT1708B 8-Ch",
3878ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
38793218c178STakashi Iwai 	{ .id = 0x1106e723, .name = "VT1708B 8-Ch",
3880ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
38813218c178STakashi Iwai 	{ .id = 0x1106e724, .name = "VT1708B 4-Ch",
3882ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
38833218c178STakashi Iwai 	{ .id = 0x1106e725, .name = "VT1708B 4-Ch",
3884ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
38853218c178STakashi Iwai 	{ .id = 0x1106e726, .name = "VT1708B 4-Ch",
3886ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
38873218c178STakashi Iwai 	{ .id = 0x1106e727, .name = "VT1708B 4-Ch",
3888ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
38893218c178STakashi Iwai 	{ .id = 0x11060397, .name = "VT1708S",
3890d949cac1SHarald Welte 	  .patch = patch_vt1708S},
38913218c178STakashi Iwai 	{ .id = 0x11061397, .name = "VT1708S",
3892d949cac1SHarald Welte 	  .patch = patch_vt1708S},
38933218c178STakashi Iwai 	{ .id = 0x11062397, .name = "VT1708S",
3894d949cac1SHarald Welte 	  .patch = patch_vt1708S},
38953218c178STakashi Iwai 	{ .id = 0x11063397, .name = "VT1708S",
3896d949cac1SHarald Welte 	  .patch = patch_vt1708S},
3897bc92df7fSLydia Wang 	{ .id = 0x11064397, .name = "VT1705",
3898d949cac1SHarald Welte 	  .patch = patch_vt1708S},
38993218c178STakashi Iwai 	{ .id = 0x11065397, .name = "VT1708S",
3900d949cac1SHarald Welte 	  .patch = patch_vt1708S},
39013218c178STakashi Iwai 	{ .id = 0x11066397, .name = "VT1708S",
3902d949cac1SHarald Welte 	  .patch = patch_vt1708S},
39033218c178STakashi Iwai 	{ .id = 0x11067397, .name = "VT1708S",
3904d949cac1SHarald Welte 	  .patch = patch_vt1708S},
39053218c178STakashi Iwai 	{ .id = 0x11060398, .name = "VT1702",
3906d949cac1SHarald Welte 	  .patch = patch_vt1702},
39073218c178STakashi Iwai 	{ .id = 0x11061398, .name = "VT1702",
3908d949cac1SHarald Welte 	  .patch = patch_vt1702},
39093218c178STakashi Iwai 	{ .id = 0x11062398, .name = "VT1702",
3910d949cac1SHarald Welte 	  .patch = patch_vt1702},
39113218c178STakashi Iwai 	{ .id = 0x11063398, .name = "VT1702",
3912d949cac1SHarald Welte 	  .patch = patch_vt1702},
39133218c178STakashi Iwai 	{ .id = 0x11064398, .name = "VT1702",
3914d949cac1SHarald Welte 	  .patch = patch_vt1702},
39153218c178STakashi Iwai 	{ .id = 0x11065398, .name = "VT1702",
3916d949cac1SHarald Welte 	  .patch = patch_vt1702},
39173218c178STakashi Iwai 	{ .id = 0x11066398, .name = "VT1702",
3918d949cac1SHarald Welte 	  .patch = patch_vt1702},
39193218c178STakashi Iwai 	{ .id = 0x11067398, .name = "VT1702",
3920d949cac1SHarald Welte 	  .patch = patch_vt1702},
3921eb7188caSLydia Wang 	{ .id = 0x11060428, .name = "VT1718S",
3922eb7188caSLydia Wang 	  .patch = patch_vt1718S},
3923eb7188caSLydia Wang 	{ .id = 0x11064428, .name = "VT1718S",
3924eb7188caSLydia Wang 	  .patch = patch_vt1718S},
3925bb3c6bfcSLydia Wang 	{ .id = 0x11060441, .name = "VT2020",
3926bb3c6bfcSLydia Wang 	  .patch = patch_vt1718S},
3927bb3c6bfcSLydia Wang 	{ .id = 0x11064441, .name = "VT1828S",
3928bb3c6bfcSLydia Wang 	  .patch = patch_vt1718S},
3929f3db423dSLydia Wang 	{ .id = 0x11060433, .name = "VT1716S",
3930f3db423dSLydia Wang 	  .patch = patch_vt1716S},
3931f3db423dSLydia Wang 	{ .id = 0x1106a721, .name = "VT1716S",
3932f3db423dSLydia Wang 	  .patch = patch_vt1716S},
393325eaba2fSLydia Wang 	{ .id = 0x11060438, .name = "VT2002P", .patch = patch_vt2002P},
393425eaba2fSLydia Wang 	{ .id = 0x11064438, .name = "VT2002P", .patch = patch_vt2002P},
3935ab6734e7SLydia Wang 	{ .id = 0x11060448, .name = "VT1812", .patch = patch_vt1812},
393636dd5c4aSLydia Wang 	{ .id = 0x11060440, .name = "VT1818S",
393736dd5c4aSLydia Wang 	  .patch = patch_vt1708S},
393811890956SLydia Wang 	{ .id = 0x11060446, .name = "VT1802",
393911890956SLydia Wang 		.patch = patch_vt2002P},
394011890956SLydia Wang 	{ .id = 0x11068446, .name = "VT1802",
394111890956SLydia Wang 		.patch = patch_vt2002P},
3942c577b8a1SJoseph Chan 	{} /* terminator */
3943c577b8a1SJoseph Chan };
39441289e9e8STakashi Iwai 
39451289e9e8STakashi Iwai MODULE_ALIAS("snd-hda-codec-id:1106*");
39461289e9e8STakashi Iwai 
39471289e9e8STakashi Iwai static struct hda_codec_preset_list via_list = {
39481289e9e8STakashi Iwai 	.preset = snd_hda_preset_via,
39491289e9e8STakashi Iwai 	.owner = THIS_MODULE,
39501289e9e8STakashi Iwai };
39511289e9e8STakashi Iwai 
39521289e9e8STakashi Iwai MODULE_LICENSE("GPL");
39531289e9e8STakashi Iwai MODULE_DESCRIPTION("VIA HD-audio codec");
39541289e9e8STakashi Iwai 
39551289e9e8STakashi Iwai static int __init patch_via_init(void)
39561289e9e8STakashi Iwai {
39571289e9e8STakashi Iwai 	return snd_hda_add_codec_preset(&via_list);
39581289e9e8STakashi Iwai }
39591289e9e8STakashi Iwai 
39601289e9e8STakashi Iwai static void __exit patch_via_exit(void)
39611289e9e8STakashi Iwai {
39621289e9e8STakashi Iwai 	snd_hda_delete_codec_preset(&via_list);
39631289e9e8STakashi Iwai }
39641289e9e8STakashi Iwai 
39651289e9e8STakashi Iwai module_init(patch_via_init)
39661289e9e8STakashi Iwai module_exit(patch_via_exit)
3967