xref: /openbmc/linux/sound/pci/hda/patch_via.c (revision 6121b84af33fcec843a082cb6bc8d39f187faa20)
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,
7943737e0aSLydia Wang 	VT1705CF,
80*6121b84aSLydia Wang 	VT1808,
81d7426329SHarald Welte 	CODEC_TYPES,
82d7426329SHarald Welte };
83d7426329SHarald Welte 
8411890956SLydia Wang #define VT2002P_COMPATIBLE(spec) \
8511890956SLydia Wang 	((spec)->codec_type == VT2002P ||\
8611890956SLydia Wang 	 (spec)->codec_type == VT1812 ||\
8711890956SLydia Wang 	 (spec)->codec_type == VT1802)
8811890956SLydia Wang 
898e3679dcSTakashi Iwai #define MAX_NID_PATH_DEPTH	5
908e3679dcSTakashi Iwai 
9109a9ad69STakashi Iwai /* output-path: DAC -> ... -> pin
9209a9ad69STakashi Iwai  * idx[] contains the source index number of the next widget;
9309a9ad69STakashi Iwai  * e.g. idx[0] is the index of the DAC selected by path[1] widget
9409a9ad69STakashi Iwai  * multi[] indicates whether it's a selector widget with multi-connectors
9509a9ad69STakashi Iwai  * (i.e. the connection selection is mandatory)
9609a9ad69STakashi Iwai  * vol_ctl and mute_ctl contains the NIDs for the assigned mixers
9709a9ad69STakashi Iwai  */
984a79616dSTakashi Iwai struct nid_path {
994a79616dSTakashi Iwai 	int depth;
1008e3679dcSTakashi Iwai 	hda_nid_t path[MAX_NID_PATH_DEPTH];
10109a9ad69STakashi Iwai 	unsigned char idx[MAX_NID_PATH_DEPTH];
10209a9ad69STakashi Iwai 	unsigned char multi[MAX_NID_PATH_DEPTH];
10309a9ad69STakashi Iwai 	unsigned int vol_ctl;
10409a9ad69STakashi Iwai 	unsigned int mute_ctl;
1054a79616dSTakashi Iwai };
1064a79616dSTakashi Iwai 
107a86a88eaSTakashi Iwai /* input-path */
108a86a88eaSTakashi Iwai struct via_input {
109a86a88eaSTakashi Iwai 	hda_nid_t pin;	/* input-pin or aa-mix */
110a86a88eaSTakashi Iwai 	int adc_idx;	/* ADC index to be used */
111a86a88eaSTakashi Iwai 	int mux_idx;	/* MUX index (if any) */
112a86a88eaSTakashi Iwai 	const char *label;	/* input-source label */
113a86a88eaSTakashi Iwai };
114a86a88eaSTakashi Iwai 
115de6c74f3STakashi Iwai #define VIA_MAX_ADCS	3
116de6c74f3STakashi Iwai 
1173b607e3dSTakashi Iwai enum {
1183b607e3dSTakashi Iwai 	STREAM_MULTI_OUT = (1 << 0),
1193b607e3dSTakashi Iwai 	STREAM_INDEP_HP = (1 << 1),
1203b607e3dSTakashi Iwai };
1213b607e3dSTakashi Iwai 
1221f2e99feSLydia Wang struct via_spec {
1237819d1c7STakashi Iwai 	struct hda_gen_spec gen;
1247819d1c7STakashi Iwai 
1251f2e99feSLydia Wang 	/* codec parameterization */
12690dd48a1STakashi Iwai 	const struct snd_kcontrol_new *mixers[6];
1271f2e99feSLydia Wang 	unsigned int num_mixers;
1281f2e99feSLydia Wang 
12990dd48a1STakashi Iwai 	const struct hda_verb *init_verbs[5];
1301f2e99feSLydia Wang 	unsigned int num_iverbs;
1311f2e99feSLydia Wang 
13282673bc8STakashi Iwai 	char stream_name_analog[32];
1337eb56e84STakashi Iwai 	char stream_name_hp[32];
13490dd48a1STakashi Iwai 	const struct hda_pcm_stream *stream_analog_playback;
13590dd48a1STakashi Iwai 	const struct hda_pcm_stream *stream_analog_capture;
1361f2e99feSLydia Wang 
13782673bc8STakashi Iwai 	char stream_name_digital[32];
13890dd48a1STakashi Iwai 	const struct hda_pcm_stream *stream_digital_playback;
13990dd48a1STakashi Iwai 	const struct hda_pcm_stream *stream_digital_capture;
1401f2e99feSLydia Wang 
1411f2e99feSLydia Wang 	/* playback */
1421f2e99feSLydia Wang 	struct hda_multi_out multiout;
1431f2e99feSLydia Wang 	hda_nid_t slave_dig_outs[2];
144ece8d043STakashi Iwai 	hda_nid_t hp_dac_nid;
1453214b966STakashi Iwai 	hda_nid_t speaker_dac_nid;
1463214b966STakashi Iwai 	int hp_indep_shared;	/* indep HP-DAC is shared with side ch */
1473b607e3dSTakashi Iwai 	int opened_streams;	/* STREAM_* bits */
1483b607e3dSTakashi Iwai 	int active_streams;	/* STREAM_* bits */
1493214b966STakashi Iwai 	int aamix_mode;		/* loopback is enabled for output-path? */
1501f2e99feSLydia Wang 
1513214b966STakashi Iwai 	/* Output-paths:
1523214b966STakashi Iwai 	 * There are different output-paths depending on the setup.
1533214b966STakashi Iwai 	 * out_path, hp_path and speaker_path are primary paths.  If both
1543214b966STakashi Iwai 	 * direct DAC and aa-loopback routes are available, these contain
1553214b966STakashi Iwai 	 * the former paths.  Meanwhile *_mix_path contain the paths with
1563214b966STakashi Iwai 	 * loopback mixer.  (Since the loopback is only for front channel,
1573214b966STakashi Iwai 	 * no out_mix_path for surround channels.)
1583214b966STakashi Iwai 	 * The HP output has another path, hp_indep_path, which is used in
1593214b966STakashi Iwai 	 * the independent-HP mode.
1603214b966STakashi Iwai 	 */
161de6c74f3STakashi Iwai 	struct nid_path out_path[HDA_SIDE + 1];
1623214b966STakashi Iwai 	struct nid_path out_mix_path;
1634a79616dSTakashi Iwai 	struct nid_path hp_path;
1643214b966STakashi Iwai 	struct nid_path hp_mix_path;
1653214b966STakashi Iwai 	struct nid_path hp_indep_path;
1664a918ffeSTakashi Iwai 	struct nid_path speaker_path;
1673214b966STakashi Iwai 	struct nid_path speaker_mix_path;
1684a79616dSTakashi Iwai 
1691f2e99feSLydia Wang 	/* capture */
1701f2e99feSLydia Wang 	unsigned int num_adc_nids;
171de6c74f3STakashi Iwai 	hda_nid_t adc_nids[VIA_MAX_ADCS];
172de6c74f3STakashi Iwai 	hda_nid_t mux_nids[VIA_MAX_ADCS];
173620e2b28STakashi Iwai 	hda_nid_t aa_mix_nid;
1741f2e99feSLydia Wang 	hda_nid_t dig_in_nid;
1751f2e99feSLydia Wang 
1761f2e99feSLydia Wang 	/* capture source */
177a86a88eaSTakashi Iwai 	bool dyn_adc_switch;
178a86a88eaSTakashi Iwai 	int num_inputs;
179a86a88eaSTakashi Iwai 	struct via_input inputs[AUTO_CFG_MAX_INS + 1];
180de6c74f3STakashi Iwai 	unsigned int cur_mux[VIA_MAX_ADCS];
1811f2e99feSLydia Wang 
1823b607e3dSTakashi Iwai 	/* dynamic DAC switching */
1833b607e3dSTakashi Iwai 	unsigned int cur_dac_stream_tag;
1843b607e3dSTakashi Iwai 	unsigned int cur_dac_format;
1853b607e3dSTakashi Iwai 	unsigned int cur_hp_stream_tag;
1863b607e3dSTakashi Iwai 	unsigned int cur_hp_format;
1873b607e3dSTakashi Iwai 
188a86a88eaSTakashi Iwai 	/* dynamic ADC switching */
189a86a88eaSTakashi Iwai 	hda_nid_t cur_adc;
190a86a88eaSTakashi Iwai 	unsigned int cur_adc_stream_tag;
191a86a88eaSTakashi Iwai 	unsigned int cur_adc_format;
192a86a88eaSTakashi Iwai 
1931f2e99feSLydia Wang 	/* PCM information */
1941f2e99feSLydia Wang 	struct hda_pcm pcm_rec[3];
1951f2e99feSLydia Wang 
1961f2e99feSLydia Wang 	/* dynamic controls, init_verbs and input_mux */
1971f2e99feSLydia Wang 	struct auto_pin_cfg autocfg;
1981f2e99feSLydia Wang 	struct snd_array kctls;
1991f2e99feSLydia Wang 	hda_nid_t private_dac_nids[AUTO_CFG_MAX_OUTS];
2001f2e99feSLydia Wang 
2011f2e99feSLydia Wang 	/* HP mode source */
2021f2e99feSLydia Wang 	unsigned int hp_independent_mode;
203f3db423dSLydia Wang 	unsigned int dmic_enabled;
20424088a58STakashi Iwai 	unsigned int no_pin_power_ctl;
2051f2e99feSLydia Wang 	enum VIA_HDA_CODEC codec_type;
2061f2e99feSLydia Wang 
207e9d010c2STakashi Iwai 	/* analog low-power control */
208e9d010c2STakashi Iwai 	bool alc_mode;
209e9d010c2STakashi Iwai 
210e3d7a143STakashi Iwai 	/* smart51 setup */
211e3d7a143STakashi Iwai 	unsigned int smart51_nums;
212e3d7a143STakashi Iwai 	hda_nid_t smart51_pins[2];
213e3d7a143STakashi Iwai 	int smart51_idxs[2];
214e3d7a143STakashi Iwai 	const char *smart51_labels[2];
215e3d7a143STakashi Iwai 	unsigned int smart51_enabled;
216e3d7a143STakashi Iwai 
2171f2e99feSLydia Wang 	/* work to check hp jack state */
2181f2e99feSLydia Wang 	struct hda_codec *codec;
2191f2e99feSLydia Wang 	struct delayed_work vt1708_hp_work;
220187d333eSTakashi Iwai 	int hp_work_active;
221e06e5a29STakashi Iwai 	int vt1708_jack_detect;
2221f2e99feSLydia Wang 	int vt1708_hp_present;
2233e95b9abSLydia Wang 
2243e95b9abSLydia Wang 	void (*set_widgets_power_state)(struct hda_codec *codec);
22543737e0aSLydia Wang 	unsigned int dac_stream_tag[4];
2263e95b9abSLydia Wang 
2271f2e99feSLydia Wang 	struct hda_loopback_check loopback;
22813af8e77STakashi Iwai 	int num_loopbacks;
22913af8e77STakashi Iwai 	struct hda_amp_list loopback_list[8];
230a86a88eaSTakashi Iwai 
231a86a88eaSTakashi Iwai 	/* bind capture-volume */
232a86a88eaSTakashi Iwai 	struct hda_bind_ctls *bind_cap_vol;
233a86a88eaSTakashi Iwai 	struct hda_bind_ctls *bind_cap_sw;
2343b607e3dSTakashi Iwai 
2353b607e3dSTakashi Iwai 	struct mutex config_mutex;
2361f2e99feSLydia Wang };
2371f2e99feSLydia Wang 
2380341ccd7SLydia Wang static enum VIA_HDA_CODEC get_codec_type(struct hda_codec *codec);
2395b0cb1d8SJaroslav Kysela static struct via_spec * via_new_spec(struct hda_codec *codec)
2405b0cb1d8SJaroslav Kysela {
2415b0cb1d8SJaroslav Kysela 	struct via_spec *spec;
2425b0cb1d8SJaroslav Kysela 
2435b0cb1d8SJaroslav Kysela 	spec = kzalloc(sizeof(*spec), GFP_KERNEL);
2445b0cb1d8SJaroslav Kysela 	if (spec == NULL)
2455b0cb1d8SJaroslav Kysela 		return NULL;
2465b0cb1d8SJaroslav Kysela 
247361dab3eSTakashi Iwai 	snd_array_init(&spec->kctls, sizeof(struct snd_kcontrol_new), 32);
2483b607e3dSTakashi Iwai 	mutex_init(&spec->config_mutex);
2495b0cb1d8SJaroslav Kysela 	codec->spec = spec;
2505b0cb1d8SJaroslav Kysela 	spec->codec = codec;
2510341ccd7SLydia Wang 	spec->codec_type = get_codec_type(codec);
2520341ccd7SLydia Wang 	/* VT1708BCE & VT1708S are almost same */
2530341ccd7SLydia Wang 	if (spec->codec_type == VT1708BCE)
2540341ccd7SLydia Wang 		spec->codec_type = VT1708S;
2557819d1c7STakashi Iwai 	snd_hda_gen_init(&spec->gen);
2565b0cb1d8SJaroslav Kysela 	return spec;
2575b0cb1d8SJaroslav Kysela }
2585b0cb1d8SJaroslav Kysela 
259744ff5f4SLydia Wang static enum VIA_HDA_CODEC get_codec_type(struct hda_codec *codec)
260d7426329SHarald Welte {
261744ff5f4SLydia Wang 	u32 vendor_id = codec->vendor_id;
262d7426329SHarald Welte 	u16 ven_id = vendor_id >> 16;
263d7426329SHarald Welte 	u16 dev_id = vendor_id & 0xffff;
264d7426329SHarald Welte 	enum VIA_HDA_CODEC codec_type;
265d7426329SHarald Welte 
266d7426329SHarald Welte 	/* get codec type */
267d7426329SHarald Welte 	if (ven_id != 0x1106)
268d7426329SHarald Welte 		codec_type = UNKNOWN;
269d7426329SHarald Welte 	else if (dev_id >= 0x1708 && dev_id <= 0x170b)
270d7426329SHarald Welte 		codec_type = VT1708;
271d7426329SHarald Welte 	else if (dev_id >= 0xe710 && dev_id <= 0xe713)
272d7426329SHarald Welte 		codec_type = VT1709_10CH;
273d7426329SHarald Welte 	else if (dev_id >= 0xe714 && dev_id <= 0xe717)
274d7426329SHarald Welte 		codec_type = VT1709_6CH;
275518bf3baSLydia Wang 	else if (dev_id >= 0xe720 && dev_id <= 0xe723) {
276d7426329SHarald Welte 		codec_type = VT1708B_8CH;
277518bf3baSLydia Wang 		if (snd_hda_param_read(codec, 0x16, AC_PAR_CONNLIST_LEN) == 0x7)
278518bf3baSLydia Wang 			codec_type = VT1708BCE;
279518bf3baSLydia Wang 	} else if (dev_id >= 0xe724 && dev_id <= 0xe727)
280d7426329SHarald Welte 		codec_type = VT1708B_4CH;
281d7426329SHarald Welte 	else if ((dev_id & 0xfff) == 0x397
282d7426329SHarald Welte 		 && (dev_id >> 12) < 8)
283d7426329SHarald Welte 		codec_type = VT1708S;
284d7426329SHarald Welte 	else if ((dev_id & 0xfff) == 0x398
285d7426329SHarald Welte 		 && (dev_id >> 12) < 8)
286d7426329SHarald Welte 		codec_type = VT1702;
287eb7188caSLydia Wang 	else if ((dev_id & 0xfff) == 0x428
288eb7188caSLydia Wang 		 && (dev_id >> 12) < 8)
289eb7188caSLydia Wang 		codec_type = VT1718S;
290f3db423dSLydia Wang 	else if (dev_id == 0x0433 || dev_id == 0xa721)
291f3db423dSLydia Wang 		codec_type = VT1716S;
292bb3c6bfcSLydia Wang 	else if (dev_id == 0x0441 || dev_id == 0x4441)
293bb3c6bfcSLydia Wang 		codec_type = VT1718S;
29425eaba2fSLydia Wang 	else if (dev_id == 0x0438 || dev_id == 0x4438)
29525eaba2fSLydia Wang 		codec_type = VT2002P;
296ab6734e7SLydia Wang 	else if (dev_id == 0x0448)
297ab6734e7SLydia Wang 		codec_type = VT1812;
29836dd5c4aSLydia Wang 	else if (dev_id == 0x0440)
29936dd5c4aSLydia Wang 		codec_type = VT1708S;
30011890956SLydia Wang 	else if ((dev_id & 0xfff) == 0x446)
30111890956SLydia Wang 		codec_type = VT1802;
30243737e0aSLydia Wang 	else if (dev_id == 0x4760)
30343737e0aSLydia Wang 		codec_type = VT1705CF;
304*6121b84aSLydia Wang 	else if (dev_id == 0x4761 || dev_id == 0x4762)
305*6121b84aSLydia Wang 		codec_type = VT1808;
306d7426329SHarald Welte 	else
307d7426329SHarald Welte 		codec_type = UNKNOWN;
308d7426329SHarald Welte 	return codec_type;
309d7426329SHarald Welte };
310d7426329SHarald Welte 
311ec7e7e42SLydia Wang #define VIA_JACK_EVENT		0x20
31269e52a80SHarald Welte #define VIA_HP_EVENT		0x01
3134a918ffeSTakashi Iwai #define VIA_LINE_EVENT		0x03
31469e52a80SHarald Welte 
315c577b8a1SJoseph Chan enum {
316c577b8a1SJoseph Chan 	VIA_CTL_WIDGET_VOL,
317c577b8a1SJoseph Chan 	VIA_CTL_WIDGET_MUTE,
318f5271101SLydia Wang 	VIA_CTL_WIDGET_ANALOG_MUTE,
319c577b8a1SJoseph Chan };
320c577b8a1SJoseph Chan 
321ada509ecSTakashi Iwai static void analog_low_current_mode(struct hda_codec *codec);
322ada509ecSTakashi Iwai static bool is_aa_path_mute(struct hda_codec *codec);
3231f2e99feSLydia Wang 
324187d333eSTakashi Iwai #define hp_detect_with_aa(codec) \
325187d333eSTakashi Iwai 	(snd_hda_get_bool_hint(codec, "analog_loopback_hp_detect") == 1 && \
326187d333eSTakashi Iwai 	 !is_aa_path_mute(codec))
3271f2e99feSLydia Wang 
3281f2e99feSLydia Wang static void vt1708_stop_hp_work(struct via_spec *spec)
3291f2e99feSLydia Wang {
3301f2e99feSLydia Wang 	if (spec->codec_type != VT1708 || spec->autocfg.hp_pins[0] == 0)
3311f2e99feSLydia Wang 		return;
332187d333eSTakashi Iwai 	if (spec->hp_work_active) {
333187d333eSTakashi Iwai 		snd_hda_codec_write(spec->codec, 0x1, 0, 0xf81, 1);
3345b84ba26STejun Heo 		cancel_delayed_work_sync(&spec->vt1708_hp_work);
335187d333eSTakashi Iwai 		spec->hp_work_active = 0;
336187d333eSTakashi Iwai 	}
337187d333eSTakashi Iwai }
338187d333eSTakashi Iwai 
339187d333eSTakashi Iwai static void vt1708_update_hp_work(struct via_spec *spec)
340187d333eSTakashi Iwai {
341187d333eSTakashi Iwai 	if (spec->codec_type != VT1708 || spec->autocfg.hp_pins[0] == 0)
342187d333eSTakashi Iwai 		return;
343187d333eSTakashi Iwai 	if (spec->vt1708_jack_detect &&
344187d333eSTakashi Iwai 	    (spec->active_streams || hp_detect_with_aa(spec->codec))) {
345187d333eSTakashi Iwai 		if (!spec->hp_work_active) {
346187d333eSTakashi Iwai 			snd_hda_codec_write(spec->codec, 0x1, 0, 0xf81, 0);
347187d333eSTakashi Iwai 			schedule_delayed_work(&spec->vt1708_hp_work,
348187d333eSTakashi Iwai 					      msecs_to_jiffies(100));
349187d333eSTakashi Iwai 			spec->hp_work_active = 1;
350187d333eSTakashi Iwai 		}
351187d333eSTakashi Iwai 	} else if (!hp_detect_with_aa(spec->codec))
352187d333eSTakashi Iwai 		vt1708_stop_hp_work(spec);
3531f2e99feSLydia Wang }
354f5271101SLydia Wang 
3553e95b9abSLydia Wang static void set_widgets_power_state(struct hda_codec *codec)
3563e95b9abSLydia Wang {
3573e95b9abSLydia Wang 	struct via_spec *spec = codec->spec;
3583e95b9abSLydia Wang 	if (spec->set_widgets_power_state)
3593e95b9abSLydia Wang 		spec->set_widgets_power_state(codec);
3603e95b9abSLydia Wang }
36125eaba2fSLydia Wang 
362f5271101SLydia Wang static int analog_input_switch_put(struct snd_kcontrol *kcontrol,
363f5271101SLydia Wang 				   struct snd_ctl_elem_value *ucontrol)
364f5271101SLydia Wang {
365f5271101SLydia Wang 	int change = snd_hda_mixer_amp_switch_put(kcontrol, ucontrol);
366f5271101SLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
367f5271101SLydia Wang 
3683e95b9abSLydia Wang 	set_widgets_power_state(codec);
369ada509ecSTakashi Iwai 	analog_low_current_mode(snd_kcontrol_chip(kcontrol));
370187d333eSTakashi Iwai 	vt1708_update_hp_work(codec->spec);
371f5271101SLydia Wang 	return change;
372f5271101SLydia Wang }
373f5271101SLydia Wang 
374f5271101SLydia Wang /* modify .put = snd_hda_mixer_amp_switch_put */
375f5271101SLydia Wang #define ANALOG_INPUT_MUTE						\
376f5271101SLydia Wang 	{		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,		\
377f5271101SLydia Wang 			.name = NULL,					\
378f5271101SLydia Wang 			.index = 0,					\
379f5271101SLydia Wang 			.info = snd_hda_mixer_amp_switch_info,		\
380f5271101SLydia Wang 			.get = snd_hda_mixer_amp_switch_get,		\
381f5271101SLydia Wang 			.put = analog_input_switch_put,			\
382f5271101SLydia Wang 			.private_value = HDA_COMPOSE_AMP_VAL(0, 3, 0, 0) }
383f5271101SLydia Wang 
38490dd48a1STakashi Iwai static const struct snd_kcontrol_new via_control_templates[] = {
385c577b8a1SJoseph Chan 	HDA_CODEC_VOLUME(NULL, 0, 0, 0),
386c577b8a1SJoseph Chan 	HDA_CODEC_MUTE(NULL, 0, 0, 0),
387f5271101SLydia Wang 	ANALOG_INPUT_MUTE,
388c577b8a1SJoseph Chan };
389c577b8a1SJoseph Chan 
390ab6734e7SLydia Wang 
391c577b8a1SJoseph Chan /* add dynamic controls */
392291c9e33STakashi Iwai static struct snd_kcontrol_new *__via_clone_ctl(struct via_spec *spec,
393291c9e33STakashi Iwai 				const struct snd_kcontrol_new *tmpl,
394291c9e33STakashi Iwai 				const char *name)
395c577b8a1SJoseph Chan {
396c577b8a1SJoseph Chan 	struct snd_kcontrol_new *knew;
397c577b8a1SJoseph Chan 
398603c4019STakashi Iwai 	knew = snd_array_new(&spec->kctls);
399c577b8a1SJoseph Chan 	if (!knew)
400291c9e33STakashi Iwai 		return NULL;
401291c9e33STakashi Iwai 	*knew = *tmpl;
402291c9e33STakashi Iwai 	if (!name)
403291c9e33STakashi Iwai 		name = tmpl->name;
404291c9e33STakashi Iwai 	if (name) {
405c577b8a1SJoseph Chan 		knew->name = kstrdup(name, GFP_KERNEL);
406c577b8a1SJoseph Chan 		if (!knew->name)
407291c9e33STakashi Iwai 			return NULL;
408291c9e33STakashi Iwai 	}
409291c9e33STakashi Iwai 	return knew;
410291c9e33STakashi Iwai }
411291c9e33STakashi Iwai 
412291c9e33STakashi Iwai static int __via_add_control(struct via_spec *spec, int type, const char *name,
413291c9e33STakashi Iwai 			     int idx, unsigned long val)
414291c9e33STakashi Iwai {
415291c9e33STakashi Iwai 	struct snd_kcontrol_new *knew;
416291c9e33STakashi Iwai 
417291c9e33STakashi Iwai 	knew = __via_clone_ctl(spec, &via_control_templates[type], name);
418291c9e33STakashi Iwai 	if (!knew)
419c577b8a1SJoseph Chan 		return -ENOMEM;
420d7a99cceSTakashi Iwai 	knew->index = idx;
4214d02d1b6SJaroslav Kysela 	if (get_amp_nid_(val))
4225e26dfd0SJaroslav Kysela 		knew->subdevice = HDA_SUBDEV_AMP_FLAG;
423c577b8a1SJoseph Chan 	knew->private_value = val;
424c577b8a1SJoseph Chan 	return 0;
425c577b8a1SJoseph Chan }
426c577b8a1SJoseph Chan 
4277b315bb4STakashi Iwai #define via_add_control(spec, type, name, val) \
4287b315bb4STakashi Iwai 	__via_add_control(spec, type, name, 0, val)
4297b315bb4STakashi Iwai 
430291c9e33STakashi Iwai #define via_clone_control(spec, tmpl) __via_clone_ctl(spec, tmpl, NULL)
4315b0cb1d8SJaroslav Kysela 
432603c4019STakashi Iwai static void via_free_kctls(struct hda_codec *codec)
433603c4019STakashi Iwai {
434603c4019STakashi Iwai 	struct via_spec *spec = codec->spec;
435603c4019STakashi Iwai 
436603c4019STakashi Iwai 	if (spec->kctls.list) {
437603c4019STakashi Iwai 		struct snd_kcontrol_new *kctl = spec->kctls.list;
438603c4019STakashi Iwai 		int i;
439603c4019STakashi Iwai 		for (i = 0; i < spec->kctls.used; i++)
440603c4019STakashi Iwai 			kfree(kctl[i].name);
441603c4019STakashi Iwai 	}
442603c4019STakashi Iwai 	snd_array_free(&spec->kctls);
443603c4019STakashi Iwai }
444603c4019STakashi Iwai 
445c577b8a1SJoseph Chan /* create input playback/capture controls for the given pin */
4469510e8ddSLydia Wang static int via_new_analog_input(struct via_spec *spec, const char *ctlname,
4477b315bb4STakashi Iwai 				int type_idx, int idx, int mix_nid)
448c577b8a1SJoseph Chan {
449c577b8a1SJoseph Chan 	char name[32];
450c577b8a1SJoseph Chan 	int err;
451c577b8a1SJoseph Chan 
452c577b8a1SJoseph Chan 	sprintf(name, "%s Playback Volume", ctlname);
4537b315bb4STakashi Iwai 	err = __via_add_control(spec, VIA_CTL_WIDGET_VOL, name, type_idx,
454c577b8a1SJoseph Chan 			      HDA_COMPOSE_AMP_VAL(mix_nid, 3, idx, HDA_INPUT));
455c577b8a1SJoseph Chan 	if (err < 0)
456c577b8a1SJoseph Chan 		return err;
457c577b8a1SJoseph Chan 	sprintf(name, "%s Playback Switch", ctlname);
4587b315bb4STakashi Iwai 	err = __via_add_control(spec, VIA_CTL_WIDGET_ANALOG_MUTE, name, type_idx,
459c577b8a1SJoseph Chan 			      HDA_COMPOSE_AMP_VAL(mix_nid, 3, idx, HDA_INPUT));
460c577b8a1SJoseph Chan 	if (err < 0)
461c577b8a1SJoseph Chan 		return err;
462c577b8a1SJoseph Chan 	return 0;
463c577b8a1SJoseph Chan }
464c577b8a1SJoseph Chan 
4655d41762aSTakashi Iwai #define get_connection_index(codec, mux, nid) \
4668d087c76STakashi Iwai 	snd_hda_get_conn_index(codec, mux, nid, 0)
4675d41762aSTakashi Iwai 
4688df2a312STakashi Iwai static bool check_amp_caps(struct hda_codec *codec, hda_nid_t nid, int dir,
4698df2a312STakashi Iwai 			   unsigned int mask)
4708df2a312STakashi Iwai {
471a934d5a9STakashi Iwai 	unsigned int caps;
472a934d5a9STakashi Iwai 	if (!nid)
473a934d5a9STakashi Iwai 		return false;
474a934d5a9STakashi Iwai 	caps = get_wcaps(codec, nid);
4758df2a312STakashi Iwai 	if (dir == HDA_INPUT)
4768df2a312STakashi Iwai 		caps &= AC_WCAP_IN_AMP;
4778df2a312STakashi Iwai 	else
4788df2a312STakashi Iwai 		caps &= AC_WCAP_OUT_AMP;
4798df2a312STakashi Iwai 	if (!caps)
4808df2a312STakashi Iwai 		return false;
4818df2a312STakashi Iwai 	if (query_amp_caps(codec, nid, dir) & mask)
4828df2a312STakashi Iwai 		return true;
4838df2a312STakashi Iwai 	return false;
4848df2a312STakashi Iwai }
4858df2a312STakashi Iwai 
48609a9ad69STakashi Iwai #define have_mute(codec, nid, dir) \
48709a9ad69STakashi Iwai 	check_amp_caps(codec, nid, dir, AC_AMPCAP_MUTE)
4888df2a312STakashi Iwai 
489d69607b3SLydia Wang /* enable/disable the output-route mixers */
490d69607b3SLydia Wang static void activate_output_mix(struct hda_codec *codec, struct nid_path *path,
4913214b966STakashi Iwai 				hda_nid_t mix_nid, int idx, bool enable)
492d69607b3SLydia Wang {
493d69607b3SLydia Wang 	int i, num, val;
494d69607b3SLydia Wang 
495d69607b3SLydia Wang 	if (!path)
496d69607b3SLydia Wang 		return;
49709cf03b8STakashi Iwai 	num = snd_hda_get_num_conns(codec, mix_nid);
498d69607b3SLydia Wang 	for (i = 0; i < num; i++) {
4993214b966STakashi Iwai 		if (i == idx)
500d69607b3SLydia Wang 			val = AMP_IN_UNMUTE(i);
501d69607b3SLydia Wang 		else
502d69607b3SLydia Wang 			val = AMP_IN_MUTE(i);
503d69607b3SLydia Wang 		snd_hda_codec_write(codec, mix_nid, 0,
504d69607b3SLydia Wang 				    AC_VERB_SET_AMP_GAIN_MUTE, val);
505d69607b3SLydia Wang 	}
506d69607b3SLydia Wang }
507d69607b3SLydia Wang 
50809a9ad69STakashi Iwai /* enable/disable the output-route */
50909a9ad69STakashi Iwai static void activate_output_path(struct hda_codec *codec, struct nid_path *path,
51009a9ad69STakashi Iwai 				 bool enable, bool force)
5115d41762aSTakashi Iwai {
512d69607b3SLydia Wang 	struct via_spec *spec = codec->spec;
5133214b966STakashi Iwai 	int i;
51409a9ad69STakashi Iwai 	for (i = 0; i < path->depth; i++) {
51509a9ad69STakashi Iwai 		hda_nid_t src, dst;
51609a9ad69STakashi Iwai 		int idx = path->idx[i];
51709a9ad69STakashi Iwai 		src = path->path[i];
51809a9ad69STakashi Iwai 		if (i < path->depth - 1)
51909a9ad69STakashi Iwai 			dst = path->path[i + 1];
52009a9ad69STakashi Iwai 		else
52109a9ad69STakashi Iwai 			dst = 0;
52209a9ad69STakashi Iwai 		if (enable && path->multi[i])
52309a9ad69STakashi Iwai 			snd_hda_codec_write(codec, dst, 0,
5245d41762aSTakashi Iwai 					    AC_VERB_SET_CONNECT_SEL, idx);
5253214b966STakashi Iwai 		if (!force && (dst == spec->aa_mix_nid))
526e5e14681SLydia Wang 			continue;
5273214b966STakashi Iwai 		if (have_mute(codec, dst, HDA_INPUT))
5283214b966STakashi Iwai 			activate_output_mix(codec, path, dst, idx, enable);
52909a9ad69STakashi Iwai 		if (!force && (src == path->vol_ctl || src == path->mute_ctl))
53009a9ad69STakashi Iwai 			continue;
53109a9ad69STakashi Iwai 		if (have_mute(codec, src, HDA_OUTPUT)) {
53209a9ad69STakashi Iwai 			int val = enable ? AMP_OUT_UNMUTE : AMP_OUT_MUTE;
53309a9ad69STakashi Iwai 			snd_hda_codec_write(codec, src, 0,
53409a9ad69STakashi Iwai 					    AC_VERB_SET_AMP_GAIN_MUTE, val);
53509a9ad69STakashi Iwai 		}
53609a9ad69STakashi Iwai 	}
5375d41762aSTakashi Iwai }
5385d41762aSTakashi Iwai 
5395d41762aSTakashi Iwai /* set the given pin as output */
5405d41762aSTakashi Iwai static void init_output_pin(struct hda_codec *codec, hda_nid_t pin,
5415d41762aSTakashi Iwai 			    int pin_type)
5425d41762aSTakashi Iwai {
5435d41762aSTakashi Iwai 	if (!pin)
5445d41762aSTakashi Iwai 		return;
545cdd03cedSTakashi Iwai 	snd_hda_set_pin_ctl(codec, pin, pin_type);
5465d41762aSTakashi Iwai 	if (snd_hda_query_pin_caps(codec, pin) & AC_PINCAP_EAPD)
5475d41762aSTakashi Iwai 		snd_hda_codec_write(codec, pin, 0,
548d3a11e60STakashi Iwai 				    AC_VERB_SET_EAPD_BTLENABLE, 0x02);
549c577b8a1SJoseph Chan }
550c577b8a1SJoseph Chan 
55109a9ad69STakashi Iwai static void via_auto_init_output(struct hda_codec *codec,
552a353fbb1STakashi Iwai 				 struct nid_path *path, int pin_type)
5535d41762aSTakashi Iwai {
5545d41762aSTakashi Iwai 	unsigned int caps;
555d69607b3SLydia Wang 	hda_nid_t pin;
5565d41762aSTakashi Iwai 
55709a9ad69STakashi Iwai 	if (!path->depth)
5585d41762aSTakashi Iwai 		return;
55909a9ad69STakashi Iwai 	pin = path->path[path->depth - 1];
5605d41762aSTakashi Iwai 
5615d41762aSTakashi Iwai 	init_output_pin(codec, pin, pin_type);
56277e314f7STakashi Iwai 	if (get_wcaps(codec, pin) & AC_WCAP_OUT_AMP)
5635d41762aSTakashi Iwai 		caps = query_amp_caps(codec, pin, HDA_OUTPUT);
56477e314f7STakashi Iwai 	else
56577e314f7STakashi Iwai 		caps = 0;
5665d41762aSTakashi Iwai 	if (caps & AC_AMPCAP_MUTE) {
5675d41762aSTakashi Iwai 		unsigned int val;
5685d41762aSTakashi Iwai 		val = (caps & AC_AMPCAP_OFFSET) >> AC_AMPCAP_OFFSET_SHIFT;
5695d41762aSTakashi Iwai 		snd_hda_codec_write(codec, pin, 0, AC_VERB_SET_AMP_GAIN_MUTE,
5705d41762aSTakashi Iwai 				    AMP_OUT_MUTE | val);
5715d41762aSTakashi Iwai 	}
572a353fbb1STakashi Iwai 	activate_output_path(codec, path, true, true); /* force on */
57309a9ad69STakashi Iwai }
574c577b8a1SJoseph Chan 
575c577b8a1SJoseph Chan static void via_auto_init_multi_out(struct hda_codec *codec)
576c577b8a1SJoseph Chan {
577c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
5783214b966STakashi Iwai 	struct nid_path *path;
579c577b8a1SJoseph Chan 	int i;
580c577b8a1SJoseph Chan 
5813214b966STakashi Iwai 	for (i = 0; i < spec->autocfg.line_outs + spec->smart51_nums; i++) {
5823214b966STakashi Iwai 		path = &spec->out_path[i];
5833214b966STakashi Iwai 		if (!i && spec->aamix_mode && spec->out_mix_path.depth)
5843214b966STakashi Iwai 			path = &spec->out_mix_path;
585a353fbb1STakashi Iwai 		via_auto_init_output(codec, path, PIN_OUT);
5863214b966STakashi Iwai 	}
587c577b8a1SJoseph Chan }
588c577b8a1SJoseph Chan 
589020066d1STakashi Iwai /* deactivate the inactive headphone-paths */
590020066d1STakashi Iwai static void deactivate_hp_paths(struct hda_codec *codec)
591c577b8a1SJoseph Chan {
592c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
5933214b966STakashi Iwai 	int shared = spec->hp_indep_shared;
594c577b8a1SJoseph Chan 
59509a9ad69STakashi Iwai 	if (spec->hp_independent_mode) {
59609a9ad69STakashi Iwai 		activate_output_path(codec, &spec->hp_path, false, false);
5973214b966STakashi Iwai 		activate_output_path(codec, &spec->hp_mix_path, false, false);
5983214b966STakashi Iwai 		if (shared)
5993214b966STakashi Iwai 			activate_output_path(codec, &spec->out_path[shared],
6003214b966STakashi Iwai 					     false, false);
601020066d1STakashi Iwai 	} else if (spec->aamix_mode || !spec->hp_path.depth) {
602020066d1STakashi Iwai 		activate_output_path(codec, &spec->hp_indep_path, false, false);
6033214b966STakashi Iwai 		activate_output_path(codec, &spec->hp_path, false, false);
6043214b966STakashi Iwai 	} else {
605020066d1STakashi Iwai 		activate_output_path(codec, &spec->hp_indep_path, false, false);
6063214b966STakashi Iwai 		activate_output_path(codec, &spec->hp_mix_path, false, false);
60709a9ad69STakashi Iwai 	}
60825eaba2fSLydia Wang }
609c577b8a1SJoseph Chan 
610020066d1STakashi Iwai static void via_auto_init_hp_out(struct hda_codec *codec)
611020066d1STakashi Iwai {
612020066d1STakashi Iwai 	struct via_spec *spec = codec->spec;
613020066d1STakashi Iwai 
614020066d1STakashi Iwai 	if (!spec->hp_path.depth) {
615a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->hp_mix_path, PIN_HP);
616020066d1STakashi Iwai 		return;
617020066d1STakashi Iwai 	}
618020066d1STakashi Iwai 	deactivate_hp_paths(codec);
619020066d1STakashi Iwai 	if (spec->hp_independent_mode)
620a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->hp_indep_path, PIN_HP);
621020066d1STakashi Iwai 	else if (spec->aamix_mode)
622a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->hp_mix_path, PIN_HP);
623020066d1STakashi Iwai 	else
624a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->hp_path, PIN_HP);
625020066d1STakashi Iwai }
626020066d1STakashi Iwai 
6274a918ffeSTakashi Iwai static void via_auto_init_speaker_out(struct hda_codec *codec)
6284a918ffeSTakashi Iwai {
6294a918ffeSTakashi Iwai 	struct via_spec *spec = codec->spec;
6304a918ffeSTakashi Iwai 
6313214b966STakashi Iwai 	if (!spec->autocfg.speaker_outs)
6323214b966STakashi Iwai 		return;
6333214b966STakashi Iwai 	if (!spec->speaker_path.depth) {
634a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->speaker_mix_path, PIN_OUT);
6353214b966STakashi Iwai 		return;
6363214b966STakashi Iwai 	}
6373214b966STakashi Iwai 	if (!spec->aamix_mode) {
6383214b966STakashi Iwai 		activate_output_path(codec, &spec->speaker_mix_path,
6393214b966STakashi Iwai 				     false, false);
640a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->speaker_path, PIN_OUT);
6413214b966STakashi Iwai 	} else {
6423214b966STakashi Iwai 		activate_output_path(codec, &spec->speaker_path, false, false);
643a353fbb1STakashi Iwai 		via_auto_init_output(codec, &spec->speaker_mix_path, PIN_OUT);
6443214b966STakashi Iwai 	}
6454a918ffeSTakashi Iwai }
6464a918ffeSTakashi Iwai 
647f4a7828bSTakashi Iwai static bool is_smart51_pins(struct hda_codec *codec, hda_nid_t pin);
6486e969d91STakashi Iwai static void via_hp_automute(struct hda_codec *codec);
64932e0191dSClemens Ladisch 
650c577b8a1SJoseph Chan static void via_auto_init_analog_input(struct hda_codec *codec)
651c577b8a1SJoseph Chan {
652c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
6537b315bb4STakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
654096a8854STakashi Iwai 	hda_nid_t conn[HDA_MAX_CONNECTIONS];
65532e0191dSClemens Ladisch 	unsigned int ctl;
656096a8854STakashi Iwai 	int i, num_conns;
657c577b8a1SJoseph Chan 
658096a8854STakashi Iwai 	/* init ADCs */
659096a8854STakashi Iwai 	for (i = 0; i < spec->num_adc_nids; i++) {
66077e314f7STakashi Iwai 		hda_nid_t nid = spec->adc_nids[i];
66177e314f7STakashi Iwai 		if (!(get_wcaps(codec, nid) & AC_WCAP_IN_AMP) ||
66277e314f7STakashi Iwai 		    !(query_amp_caps(codec, nid, HDA_INPUT) & AC_AMPCAP_MUTE))
66377e314f7STakashi Iwai 			continue;
664096a8854STakashi Iwai 		snd_hda_codec_write(codec, spec->adc_nids[i], 0,
665096a8854STakashi Iwai 				    AC_VERB_SET_AMP_GAIN_MUTE,
666096a8854STakashi Iwai 				    AMP_IN_UNMUTE(0));
667096a8854STakashi Iwai 	}
668096a8854STakashi Iwai 
669096a8854STakashi Iwai 	/* init pins */
6707b315bb4STakashi Iwai 	for (i = 0; i < cfg->num_inputs; i++) {
6717b315bb4STakashi Iwai 		hda_nid_t nid = cfg->inputs[i].pin;
672f4a7828bSTakashi Iwai 		if (spec->smart51_enabled && is_smart51_pins(codec, nid))
67332e0191dSClemens Ladisch 			ctl = PIN_OUT;
6744740860bSTakashi Iwai 		else {
67532e0191dSClemens Ladisch 			ctl = PIN_IN;
6764740860bSTakashi Iwai 			if (cfg->inputs[i].type == AUTO_PIN_MIC)
6774740860bSTakashi Iwai 				ctl |= snd_hda_get_default_vref(codec, nid);
6784740860bSTakashi Iwai 		}
679cdd03cedSTakashi Iwai 		snd_hda_set_pin_ctl(codec, nid, ctl);
680c577b8a1SJoseph Chan 	}
681096a8854STakashi Iwai 
682096a8854STakashi Iwai 	/* init input-src */
683096a8854STakashi Iwai 	for (i = 0; i < spec->num_adc_nids; i++) {
684a86a88eaSTakashi Iwai 		int adc_idx = spec->inputs[spec->cur_mux[i]].adc_idx;
685fc1156c0STakashi Iwai 		/* secondary ADCs must have the unique MUX */
686fc1156c0STakashi Iwai 		if (i > 0 && !spec->mux_nids[i])
687fc1156c0STakashi Iwai 			break;
688a86a88eaSTakashi Iwai 		if (spec->mux_nids[adc_idx]) {
689a86a88eaSTakashi Iwai 			int mux_idx = spec->inputs[spec->cur_mux[i]].mux_idx;
690a86a88eaSTakashi Iwai 			snd_hda_codec_write(codec, spec->mux_nids[adc_idx], 0,
691096a8854STakashi Iwai 					    AC_VERB_SET_CONNECT_SEL,
692a86a88eaSTakashi Iwai 					    mux_idx);
693a86a88eaSTakashi Iwai 		}
694a86a88eaSTakashi Iwai 		if (spec->dyn_adc_switch)
695a86a88eaSTakashi Iwai 			break; /* only one input-src */
696096a8854STakashi Iwai 	}
697096a8854STakashi Iwai 
698096a8854STakashi Iwai 	/* init aa-mixer */
699096a8854STakashi Iwai 	if (!spec->aa_mix_nid)
700096a8854STakashi Iwai 		return;
701096a8854STakashi Iwai 	num_conns = snd_hda_get_connections(codec, spec->aa_mix_nid, conn,
702096a8854STakashi Iwai 					    ARRAY_SIZE(conn));
703096a8854STakashi Iwai 	for (i = 0; i < num_conns; i++) {
704096a8854STakashi Iwai 		unsigned int caps = get_wcaps(codec, conn[i]);
705096a8854STakashi Iwai 		if (get_wcaps_type(caps) == AC_WID_PIN)
706096a8854STakashi Iwai 			snd_hda_codec_write(codec, spec->aa_mix_nid, 0,
707096a8854STakashi Iwai 					    AC_VERB_SET_AMP_GAIN_MUTE,
708096a8854STakashi Iwai 					    AMP_IN_MUTE(i));
709096a8854STakashi Iwai 	}
710c577b8a1SJoseph Chan }
711f5271101SLydia Wang 
712054d867eSTakashi Iwai static void update_power_state(struct hda_codec *codec, hda_nid_t nid,
713054d867eSTakashi Iwai 			       unsigned int parm)
714054d867eSTakashi Iwai {
715054d867eSTakashi Iwai 	if (snd_hda_codec_read(codec, nid, 0,
716054d867eSTakashi Iwai 			       AC_VERB_GET_POWER_STATE, 0) == parm)
717054d867eSTakashi Iwai 		return;
718054d867eSTakashi Iwai 	snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_POWER_STATE, parm);
719054d867eSTakashi Iwai }
720054d867eSTakashi Iwai 
72143737e0aSLydia Wang static void update_conv_power_state(struct hda_codec *codec, hda_nid_t nid,
72243737e0aSLydia Wang 			       unsigned int parm, unsigned int index)
72343737e0aSLydia Wang {
72443737e0aSLydia Wang 	struct via_spec *spec = codec->spec;
72543737e0aSLydia Wang 	unsigned int format;
72643737e0aSLydia Wang 	if (snd_hda_codec_read(codec, nid, 0,
72743737e0aSLydia Wang 			       AC_VERB_GET_POWER_STATE, 0) == parm)
72843737e0aSLydia Wang 		return;
72943737e0aSLydia Wang 	format = snd_hda_codec_read(codec, nid, 0, AC_VERB_GET_CONV, 0);
73043737e0aSLydia Wang 	if (format && (spec->dac_stream_tag[index] != format))
73143737e0aSLydia Wang 		spec->dac_stream_tag[index] = format;
73243737e0aSLydia Wang 
73343737e0aSLydia Wang 	snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_POWER_STATE, parm);
73443737e0aSLydia Wang 	if (parm == AC_PWRST_D0) {
73543737e0aSLydia Wang 		format = snd_hda_codec_read(codec, nid, 0, AC_VERB_GET_CONV, 0);
73643737e0aSLydia Wang 		if (!format && (spec->dac_stream_tag[index] != format))
73743737e0aSLydia Wang 			snd_hda_codec_write(codec, nid, 0,
73843737e0aSLydia Wang 						  AC_VERB_SET_CHANNEL_STREAMID,
73943737e0aSLydia Wang 						  spec->dac_stream_tag[index]);
74043737e0aSLydia Wang 	}
74143737e0aSLydia Wang }
74243737e0aSLydia Wang 
743f5271101SLydia Wang static void set_pin_power_state(struct hda_codec *codec, hda_nid_t nid,
744f5271101SLydia Wang 				unsigned int *affected_parm)
745f5271101SLydia Wang {
746f5271101SLydia Wang 	unsigned parm;
747f5271101SLydia Wang 	unsigned def_conf = snd_hda_codec_get_pincfg(codec, nid);
748f5271101SLydia Wang 	unsigned no_presence = (def_conf & AC_DEFCFG_MISC)
749f5271101SLydia Wang 		>> AC_DEFCFG_MISC_SHIFT
750f5271101SLydia Wang 		& AC_DEFCFG_MISC_NO_PRESENCE; /* do not support pin sense */
7511564b287SLydia Wang 	struct via_spec *spec = codec->spec;
75224088a58STakashi Iwai 	unsigned present = 0;
75324088a58STakashi Iwai 
75424088a58STakashi Iwai 	no_presence |= spec->no_pin_power_ctl;
75524088a58STakashi Iwai 	if (!no_presence)
75624088a58STakashi Iwai 		present = snd_hda_jack_detect(codec, nid);
757f4a7828bSTakashi Iwai 	if ((spec->smart51_enabled && is_smart51_pins(codec, nid))
7581564b287SLydia Wang 	    || ((no_presence || present)
7591564b287SLydia Wang 		&& get_defcfg_connect(def_conf) != AC_JACK_PORT_NONE)) {
760f5271101SLydia Wang 		*affected_parm = AC_PWRST_D0; /* if it's connected */
761f5271101SLydia Wang 		parm = AC_PWRST_D0;
762f5271101SLydia Wang 	} else
763f5271101SLydia Wang 		parm = AC_PWRST_D3;
764f5271101SLydia Wang 
765054d867eSTakashi Iwai 	update_power_state(codec, nid, parm);
766f5271101SLydia Wang }
767f5271101SLydia Wang 
76824088a58STakashi Iwai static int via_pin_power_ctl_info(struct snd_kcontrol *kcontrol,
76924088a58STakashi Iwai 				  struct snd_ctl_elem_info *uinfo)
77024088a58STakashi Iwai {
771dda415d4STakashi Iwai 	return snd_hda_enum_bool_helper_info(kcontrol, uinfo);
77224088a58STakashi Iwai }
77324088a58STakashi Iwai 
77424088a58STakashi Iwai static int via_pin_power_ctl_get(struct snd_kcontrol *kcontrol,
77524088a58STakashi Iwai 				 struct snd_ctl_elem_value *ucontrol)
77624088a58STakashi Iwai {
77724088a58STakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
77824088a58STakashi Iwai 	struct via_spec *spec = codec->spec;
77924088a58STakashi Iwai 	ucontrol->value.enumerated.item[0] = !spec->no_pin_power_ctl;
78024088a58STakashi Iwai 	return 0;
78124088a58STakashi Iwai }
78224088a58STakashi Iwai 
78324088a58STakashi Iwai static int via_pin_power_ctl_put(struct snd_kcontrol *kcontrol,
78424088a58STakashi Iwai 				 struct snd_ctl_elem_value *ucontrol)
78524088a58STakashi Iwai {
78624088a58STakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
78724088a58STakashi Iwai 	struct via_spec *spec = codec->spec;
78824088a58STakashi Iwai 	unsigned int val = !ucontrol->value.enumerated.item[0];
78924088a58STakashi Iwai 
79024088a58STakashi Iwai 	if (val == spec->no_pin_power_ctl)
79124088a58STakashi Iwai 		return 0;
79224088a58STakashi Iwai 	spec->no_pin_power_ctl = val;
79324088a58STakashi Iwai 	set_widgets_power_state(codec);
794e9d010c2STakashi Iwai 	analog_low_current_mode(codec);
79524088a58STakashi Iwai 	return 1;
79624088a58STakashi Iwai }
79724088a58STakashi Iwai 
79824088a58STakashi Iwai static const struct snd_kcontrol_new via_pin_power_ctl_enum = {
79924088a58STakashi Iwai 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
80024088a58STakashi Iwai 	.name = "Dynamic Power-Control",
80124088a58STakashi Iwai 	.info = via_pin_power_ctl_info,
80224088a58STakashi Iwai 	.get = via_pin_power_ctl_get,
80324088a58STakashi Iwai 	.put = via_pin_power_ctl_put,
80424088a58STakashi Iwai };
80524088a58STakashi Iwai 
80624088a58STakashi Iwai 
8070aa62aefSHarald Welte static int via_independent_hp_info(struct snd_kcontrol *kcontrol,
8080aa62aefSHarald Welte 				   struct snd_ctl_elem_info *uinfo)
8090aa62aefSHarald Welte {
8108df2a312STakashi Iwai 	static const char * const texts[] = { "OFF", "ON" };
8118df2a312STakashi Iwai 
8128df2a312STakashi Iwai 	uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
8138df2a312STakashi Iwai 	uinfo->count = 1;
8148df2a312STakashi Iwai 	uinfo->value.enumerated.items = 2;
8158df2a312STakashi Iwai 	if (uinfo->value.enumerated.item >= 2)
8168df2a312STakashi Iwai 		uinfo->value.enumerated.item = 1;
8178df2a312STakashi Iwai 	strcpy(uinfo->value.enumerated.name,
8188df2a312STakashi Iwai 	       texts[uinfo->value.enumerated.item]);
8198df2a312STakashi Iwai 	return 0;
8200aa62aefSHarald Welte }
8210aa62aefSHarald Welte 
8220aa62aefSHarald Welte static int via_independent_hp_get(struct snd_kcontrol *kcontrol,
8230aa62aefSHarald Welte 				  struct snd_ctl_elem_value *ucontrol)
8240aa62aefSHarald Welte {
8250aa62aefSHarald Welte 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
826cdc1784dSLydia Wang 	struct via_spec *spec = codec->spec;
827cdc1784dSLydia Wang 
828ece8d043STakashi Iwai 	ucontrol->value.enumerated.item[0] = spec->hp_independent_mode;
829cdc1784dSLydia Wang 	return 0;
830cdc1784dSLydia Wang }
831cdc1784dSLydia Wang 
8323b607e3dSTakashi Iwai /* adjust spec->multiout setup according to the current flags */
8333b607e3dSTakashi Iwai static void setup_playback_multi_pcm(struct via_spec *spec)
8343b607e3dSTakashi Iwai {
8353b607e3dSTakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
8363b607e3dSTakashi Iwai 	spec->multiout.num_dacs = cfg->line_outs + spec->smart51_nums;
8373b607e3dSTakashi Iwai 	spec->multiout.hp_nid = 0;
8383b607e3dSTakashi Iwai 	if (!spec->hp_independent_mode) {
8393b607e3dSTakashi Iwai 		if (!spec->hp_indep_shared)
8403b607e3dSTakashi Iwai 			spec->multiout.hp_nid = spec->hp_dac_nid;
8413b607e3dSTakashi Iwai 	} else {
8423b607e3dSTakashi Iwai 		if (spec->hp_indep_shared)
8433b607e3dSTakashi Iwai 			spec->multiout.num_dacs = cfg->line_outs - 1;
8443b607e3dSTakashi Iwai 	}
8453b607e3dSTakashi Iwai }
8463b607e3dSTakashi Iwai 
8473b607e3dSTakashi Iwai /* update DAC setups according to indep-HP switch;
8483b607e3dSTakashi Iwai  * this function is called only when indep-HP is modified
8493b607e3dSTakashi Iwai  */
8503b607e3dSTakashi Iwai static void switch_indep_hp_dacs(struct hda_codec *codec)
8513b607e3dSTakashi Iwai {
8523b607e3dSTakashi Iwai 	struct via_spec *spec = codec->spec;
8533b607e3dSTakashi Iwai 	int shared = spec->hp_indep_shared;
8543b607e3dSTakashi Iwai 	hda_nid_t shared_dac, hp_dac;
8553b607e3dSTakashi Iwai 
8563b607e3dSTakashi Iwai 	if (!spec->opened_streams)
8573b607e3dSTakashi Iwai 		return;
8583b607e3dSTakashi Iwai 
8593b607e3dSTakashi Iwai 	shared_dac = shared ? spec->multiout.dac_nids[shared] : 0;
8603b607e3dSTakashi Iwai 	hp_dac = spec->hp_dac_nid;
8613b607e3dSTakashi Iwai 	if (spec->hp_independent_mode) {
8623b607e3dSTakashi Iwai 		/* switch to indep-HP mode */
8633b607e3dSTakashi Iwai 		if (spec->active_streams & STREAM_MULTI_OUT) {
8643b607e3dSTakashi Iwai 			__snd_hda_codec_cleanup_stream(codec, hp_dac, 1);
8653b607e3dSTakashi Iwai 			__snd_hda_codec_cleanup_stream(codec, shared_dac, 1);
8663b607e3dSTakashi Iwai 		}
8673b607e3dSTakashi Iwai 		if (spec->active_streams & STREAM_INDEP_HP)
8683b607e3dSTakashi Iwai 			snd_hda_codec_setup_stream(codec, hp_dac,
8693b607e3dSTakashi Iwai 						   spec->cur_hp_stream_tag, 0,
8703b607e3dSTakashi Iwai 						   spec->cur_hp_format);
8713b607e3dSTakashi Iwai 	} else {
8723b607e3dSTakashi Iwai 		/* back to HP or shared-DAC */
8733b607e3dSTakashi Iwai 		if (spec->active_streams & STREAM_INDEP_HP)
8743b607e3dSTakashi Iwai 			__snd_hda_codec_cleanup_stream(codec, hp_dac, 1);
8753b607e3dSTakashi Iwai 		if (spec->active_streams & STREAM_MULTI_OUT) {
8763b607e3dSTakashi Iwai 			hda_nid_t dac;
8773b607e3dSTakashi Iwai 			int ch;
8783b607e3dSTakashi Iwai 			if (shared_dac) { /* reset mutli-ch DAC */
8793b607e3dSTakashi Iwai 				dac = shared_dac;
8803b607e3dSTakashi Iwai 				ch = shared * 2;
8813b607e3dSTakashi Iwai 			} else { /* reset HP DAC */
8823b607e3dSTakashi Iwai 				dac = hp_dac;
8833b607e3dSTakashi Iwai 				ch = 0;
8843b607e3dSTakashi Iwai 			}
8853b607e3dSTakashi Iwai 			snd_hda_codec_setup_stream(codec, dac,
8863b607e3dSTakashi Iwai 						   spec->cur_dac_stream_tag, ch,
8873b607e3dSTakashi Iwai 						   spec->cur_dac_format);
8883b607e3dSTakashi Iwai 		}
8893b607e3dSTakashi Iwai 	}
8903b607e3dSTakashi Iwai 	setup_playback_multi_pcm(spec);
8913b607e3dSTakashi Iwai }
8923b607e3dSTakashi Iwai 
8930aa62aefSHarald Welte static int via_independent_hp_put(struct snd_kcontrol *kcontrol,
8940aa62aefSHarald Welte 				  struct snd_ctl_elem_value *ucontrol)
8950aa62aefSHarald Welte {
8960aa62aefSHarald Welte 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
8970aa62aefSHarald Welte 	struct via_spec *spec = codec->spec;
8983214b966STakashi Iwai 	int cur, shared;
8998df2a312STakashi Iwai 
9003b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
90125250505STakashi Iwai 	cur = !!ucontrol->value.enumerated.item[0];
9023b607e3dSTakashi Iwai 	if (spec->hp_independent_mode == cur) {
9033b607e3dSTakashi Iwai 		mutex_unlock(&spec->config_mutex);
90425250505STakashi Iwai 		return 0;
9053b607e3dSTakashi Iwai 	}
90625250505STakashi Iwai 	spec->hp_independent_mode = cur;
9073214b966STakashi Iwai 	shared = spec->hp_indep_shared;
908020066d1STakashi Iwai 	deactivate_hp_paths(codec);
909020066d1STakashi Iwai 	if (cur)
910020066d1STakashi Iwai 		activate_output_path(codec, &spec->hp_indep_path, true, false);
911020066d1STakashi Iwai 	else {
9123214b966STakashi Iwai 		if (shared)
9133214b966STakashi Iwai 			activate_output_path(codec, &spec->out_path[shared],
91425250505STakashi Iwai 					     true, false);
915020066d1STakashi Iwai 		if (spec->aamix_mode || !spec->hp_path.depth)
916020066d1STakashi Iwai 			activate_output_path(codec, &spec->hp_mix_path,
917020066d1STakashi Iwai 					     true, false);
918020066d1STakashi Iwai 		else
919020066d1STakashi Iwai 			activate_output_path(codec, &spec->hp_path,
920020066d1STakashi Iwai 					     true, false);
9218df2a312STakashi Iwai 	}
9220aa62aefSHarald Welte 
9233b607e3dSTakashi Iwai 	switch_indep_hp_dacs(codec);
9243b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
9253b607e3dSTakashi Iwai 
926ce0e5a9eSLydia Wang 	/* update jack power state */
9273e95b9abSLydia Wang 	set_widgets_power_state(codec);
9286e969d91STakashi Iwai 	via_hp_automute(codec);
92925250505STakashi Iwai 	return 1;
9300aa62aefSHarald Welte }
9310aa62aefSHarald Welte 
932ece8d043STakashi Iwai static const struct snd_kcontrol_new via_hp_mixer = {
9330aa62aefSHarald Welte 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
9340aa62aefSHarald Welte 	.name = "Independent HP",
9350aa62aefSHarald Welte 	.info = via_independent_hp_info,
9360aa62aefSHarald Welte 	.get = via_independent_hp_get,
9370aa62aefSHarald Welte 	.put = via_independent_hp_put,
9380aa62aefSHarald Welte };
9390aa62aefSHarald Welte 
9403d83e577STakashi Iwai static int via_hp_build(struct hda_codec *codec)
9415b0cb1d8SJaroslav Kysela {
9423d83e577STakashi Iwai 	struct via_spec *spec = codec->spec;
9435b0cb1d8SJaroslav Kysela 	struct snd_kcontrol_new *knew;
9445b0cb1d8SJaroslav Kysela 	hda_nid_t nid;
9455b0cb1d8SJaroslav Kysela 
9465b0cb1d8SJaroslav Kysela 	nid = spec->autocfg.hp_pins[0];
947ece8d043STakashi Iwai 	knew = via_clone_control(spec, &via_hp_mixer);
9483d83e577STakashi Iwai 	if (knew == NULL)
9493d83e577STakashi Iwai 		return -ENOMEM;
9503d83e577STakashi Iwai 
9515b0cb1d8SJaroslav Kysela 	knew->subdevice = HDA_SUBDEV_NID_FLAG | nid;
9525b0cb1d8SJaroslav Kysela 
9535b0cb1d8SJaroslav Kysela 	return 0;
9545b0cb1d8SJaroslav Kysela }
9555b0cb1d8SJaroslav Kysela 
9561564b287SLydia Wang static void notify_aa_path_ctls(struct hda_codec *codec)
9571564b287SLydia Wang {
958e3d7a143STakashi Iwai 	struct via_spec *spec = codec->spec;
9591564b287SLydia Wang 	int i;
9601564b287SLydia Wang 
961e3d7a143STakashi Iwai 	for (i = 0; i < spec->smart51_nums; i++) {
962e3d7a143STakashi Iwai 		struct snd_kcontrol *ctl;
963e3d7a143STakashi Iwai 		struct snd_ctl_elem_id id;
9641564b287SLydia Wang 		memset(&id, 0, sizeof(id));
9651564b287SLydia Wang 		id.iface = SNDRV_CTL_ELEM_IFACE_MIXER;
966e3d7a143STakashi Iwai 		sprintf(id.name, "%s Playback Volume", spec->smart51_labels[i]);
967525566cbSLydia Wang 		ctl = snd_hda_find_mixer_ctl(codec, id.name);
968525566cbSLydia Wang 		if (ctl)
969525566cbSLydia Wang 			snd_ctl_notify(codec->bus->card,
970525566cbSLydia Wang 					SNDRV_CTL_EVENT_MASK_VALUE,
971525566cbSLydia Wang 					&ctl->id);
9721564b287SLydia Wang 	}
9731564b287SLydia Wang }
9741564b287SLydia Wang 
9751564b287SLydia Wang static void mute_aa_path(struct hda_codec *codec, int mute)
9761564b287SLydia Wang {
9771564b287SLydia Wang 	struct via_spec *spec = codec->spec;
9781564b287SLydia Wang 	int val = mute ? HDA_AMP_MUTE : HDA_AMP_UNMUTE;
979e3d7a143STakashi Iwai 	int i;
980e3d7a143STakashi Iwai 
981e3d7a143STakashi Iwai 	/* check AA path's mute status */
982e3d7a143STakashi Iwai 	for (i = 0; i < spec->smart51_nums; i++) {
983e3d7a143STakashi Iwai 		if (spec->smart51_idxs[i] < 0)
984e3d7a143STakashi Iwai 			continue;
985e3d7a143STakashi Iwai 		snd_hda_codec_amp_stereo(codec, spec->aa_mix_nid,
986e3d7a143STakashi Iwai 					 HDA_INPUT, spec->smart51_idxs[i],
9871564b287SLydia Wang 					 HDA_AMP_MUTE, val);
9881564b287SLydia Wang 	}
9891564b287SLydia Wang }
990f4a7828bSTakashi Iwai 
991e3d7a143STakashi Iwai static bool is_smart51_pins(struct hda_codec *codec, hda_nid_t pin)
992e3d7a143STakashi Iwai {
993e3d7a143STakashi Iwai 	struct via_spec *spec = codec->spec;
994e3d7a143STakashi Iwai 	int i;
995e3d7a143STakashi Iwai 
996e3d7a143STakashi Iwai 	for (i = 0; i < spec->smart51_nums; i++)
997e3d7a143STakashi Iwai 		if (spec->smart51_pins[i] == pin)
998e3d7a143STakashi Iwai 			return true;
999e3d7a143STakashi Iwai 	return false;
1000e3d7a143STakashi Iwai }
1001e3d7a143STakashi Iwai 
10021564b287SLydia Wang static int via_smart51_get(struct snd_kcontrol *kcontrol,
10031564b287SLydia Wang 			   struct snd_ctl_elem_value *ucontrol)
10041564b287SLydia Wang {
10051564b287SLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
10061564b287SLydia Wang 	struct via_spec *spec = codec->spec;
10071564b287SLydia Wang 
1008f2b1c9f0STakashi Iwai 	*ucontrol->value.integer.value = spec->smart51_enabled;
10091564b287SLydia Wang 	return 0;
10101564b287SLydia Wang }
10111564b287SLydia Wang 
10121564b287SLydia Wang static int via_smart51_put(struct snd_kcontrol *kcontrol,
10131564b287SLydia Wang 			   struct snd_ctl_elem_value *ucontrol)
10141564b287SLydia Wang {
10151564b287SLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
10161564b287SLydia Wang 	struct via_spec *spec = codec->spec;
10171564b287SLydia Wang 	int out_in = *ucontrol->value.integer.value
10181564b287SLydia Wang 		? AC_PINCTL_OUT_EN : AC_PINCTL_IN_EN;
10191564b287SLydia Wang 	int i;
10201564b287SLydia Wang 
1021e3d7a143STakashi Iwai 	for (i = 0; i < spec->smart51_nums; i++) {
1022e3d7a143STakashi Iwai 		hda_nid_t nid = spec->smart51_pins[i];
10237b315bb4STakashi Iwai 		unsigned int parm;
10247b315bb4STakashi Iwai 
10257b315bb4STakashi Iwai 		parm = snd_hda_codec_read(codec, nid, 0,
10261564b287SLydia Wang 					  AC_VERB_GET_PIN_WIDGET_CONTROL, 0);
10271564b287SLydia Wang 		parm &= ~(AC_PINCTL_IN_EN | AC_PINCTL_OUT_EN);
10281564b287SLydia Wang 		parm |= out_in;
1029cdd03cedSTakashi Iwai 		snd_hda_set_pin_ctl(codec, nid, parm);
10301564b287SLydia Wang 		if (out_in == AC_PINCTL_OUT_EN) {
10311564b287SLydia Wang 			mute_aa_path(codec, 1);
10321564b287SLydia Wang 			notify_aa_path_ctls(codec);
10331564b287SLydia Wang 		}
10341564b287SLydia Wang 	}
10351564b287SLydia Wang 	spec->smart51_enabled = *ucontrol->value.integer.value;
10363e95b9abSLydia Wang 	set_widgets_power_state(codec);
10371564b287SLydia Wang 	return 1;
10381564b287SLydia Wang }
10391564b287SLydia Wang 
10405f4b36d6STakashi Iwai static const struct snd_kcontrol_new via_smart51_mixer = {
10411564b287SLydia Wang 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
10421564b287SLydia Wang 	.name = "Smart 5.1",
10431564b287SLydia Wang 	.count = 1,
1044f2b1c9f0STakashi Iwai 	.info = snd_ctl_boolean_mono_info,
10451564b287SLydia Wang 	.get = via_smart51_get,
10461564b287SLydia Wang 	.put = via_smart51_put,
10471564b287SLydia Wang };
10481564b287SLydia Wang 
1049f4a7828bSTakashi Iwai static int via_smart51_build(struct hda_codec *codec)
10505b0cb1d8SJaroslav Kysela {
1051f4a7828bSTakashi Iwai 	struct via_spec *spec = codec->spec;
10525b0cb1d8SJaroslav Kysela 
1053e3d7a143STakashi Iwai 	if (!spec->smart51_nums)
1054cb34c207SLydia Wang 		return 0;
1055e3d7a143STakashi Iwai 	if (!via_clone_control(spec, &via_smart51_mixer))
10565b0cb1d8SJaroslav Kysela 		return -ENOMEM;
10575b0cb1d8SJaroslav Kysela 	return 0;
10585b0cb1d8SJaroslav Kysela }
10595b0cb1d8SJaroslav Kysela 
1060f5271101SLydia Wang /* check AA path's mute status */
1061ada509ecSTakashi Iwai static bool is_aa_path_mute(struct hda_codec *codec)
1062ada509ecSTakashi Iwai {
1063ada509ecSTakashi Iwai 	struct via_spec *spec = codec->spec;
1064ada509ecSTakashi Iwai 	const struct hda_amp_list *p;
1065ada509ecSTakashi Iwai 	int i, ch, v;
1066ada509ecSTakashi Iwai 
1067ada509ecSTakashi Iwai 	for (i = 0; i < spec->num_loopbacks; i++) {
1068ada509ecSTakashi Iwai 		p = &spec->loopback_list[i];
1069ada509ecSTakashi Iwai 		for (ch = 0; ch < 2; ch++) {
1070ada509ecSTakashi Iwai 			v = snd_hda_codec_amp_read(codec, p->nid, ch, p->dir,
1071ada509ecSTakashi Iwai 						   p->idx);
1072ada509ecSTakashi Iwai 			if (!(v & HDA_AMP_MUTE) && v > 0)
1073ada509ecSTakashi Iwai 				return false;
1074f5271101SLydia Wang 		}
1075f5271101SLydia Wang 	}
1076ada509ecSTakashi Iwai 	return true;
1077f5271101SLydia Wang }
1078f5271101SLydia Wang 
1079f5271101SLydia Wang /* enter/exit analog low-current mode */
1080e9d010c2STakashi Iwai static void __analog_low_current_mode(struct hda_codec *codec, bool force)
1081f5271101SLydia Wang {
1082f5271101SLydia Wang 	struct via_spec *spec = codec->spec;
1083ada509ecSTakashi Iwai 	bool enable;
1084ada509ecSTakashi Iwai 	unsigned int verb, parm;
1085f5271101SLydia Wang 
1086e9d010c2STakashi Iwai 	if (spec->no_pin_power_ctl)
1087e9d010c2STakashi Iwai 		enable = false;
1088e9d010c2STakashi Iwai 	else
108992433923STakashi Iwai 		enable = is_aa_path_mute(codec) && !spec->opened_streams;
1090e9d010c2STakashi Iwai 	if (enable == spec->alc_mode && !force)
1091e9d010c2STakashi Iwai 		return;
1092e9d010c2STakashi Iwai 	spec->alc_mode = enable;
1093f5271101SLydia Wang 
1094f5271101SLydia Wang 	/* decide low current mode's verb & parameter */
1095f5271101SLydia Wang 	switch (spec->codec_type) {
1096f5271101SLydia Wang 	case VT1708B_8CH:
1097f5271101SLydia Wang 	case VT1708B_4CH:
1098f5271101SLydia Wang 		verb = 0xf70;
1099f5271101SLydia Wang 		parm = enable ? 0x02 : 0x00; /* 0x02: 2/3x, 0x00: 1x */
1100f5271101SLydia Wang 		break;
1101f5271101SLydia Wang 	case VT1708S:
1102eb7188caSLydia Wang 	case VT1718S:
1103f3db423dSLydia Wang 	case VT1716S:
1104f5271101SLydia Wang 		verb = 0xf73;
1105f5271101SLydia Wang 		parm = enable ? 0x51 : 0xe1; /* 0x51: 4/28x, 0xe1: 1x */
1106f5271101SLydia Wang 		break;
1107f5271101SLydia Wang 	case VT1702:
1108f5271101SLydia Wang 		verb = 0xf73;
1109f5271101SLydia Wang 		parm = enable ? 0x01 : 0x1d; /* 0x01: 4/40x, 0x1d: 1x */
1110f5271101SLydia Wang 		break;
111125eaba2fSLydia Wang 	case VT2002P:
1112ab6734e7SLydia Wang 	case VT1812:
111311890956SLydia Wang 	case VT1802:
111425eaba2fSLydia Wang 		verb = 0xf93;
111525eaba2fSLydia Wang 		parm = enable ? 0x00 : 0xe0; /* 0x00: 4/40x, 0xe0: 1x */
111625eaba2fSLydia Wang 		break;
111743737e0aSLydia Wang 	case VT1705CF:
1118*6121b84aSLydia Wang 	case VT1808:
111943737e0aSLydia Wang 		verb = 0xf82;
112043737e0aSLydia Wang 		parm = enable ? 0x00 : 0xe0;  /* 0x00: 4/40x, 0xe0: 1x */
112143737e0aSLydia Wang 		break;
1122f5271101SLydia Wang 	default:
1123f5271101SLydia Wang 		return;		/* other codecs are not supported */
1124f5271101SLydia Wang 	}
1125f5271101SLydia Wang 	/* send verb */
1126f5271101SLydia Wang 	snd_hda_codec_write(codec, codec->afg, 0, verb, parm);
1127f5271101SLydia Wang }
1128f5271101SLydia Wang 
1129e9d010c2STakashi Iwai static void analog_low_current_mode(struct hda_codec *codec)
1130e9d010c2STakashi Iwai {
1131e9d010c2STakashi Iwai 	return __analog_low_current_mode(codec, false);
1132e9d010c2STakashi Iwai }
1133e9d010c2STakashi Iwai 
1134c577b8a1SJoseph Chan /*
1135c577b8a1SJoseph Chan  * generic initialization of ADC, input mixers and output mixers
1136c577b8a1SJoseph Chan  */
1137096a8854STakashi Iwai static const struct hda_verb vt1708_init_verbs[] = {
1138aa266fccSLydia Wang 	/* power down jack detect function */
1139aa266fccSLydia Wang 	{0x1, 0xf81, 0x1},
1140f7278fd0SJosepch Chan 	{ }
1141c577b8a1SJoseph Chan };
1142c577b8a1SJoseph Chan 
11433b607e3dSTakashi Iwai static void set_stream_open(struct hda_codec *codec, int bit, bool active)
11447eb56e84STakashi Iwai {
1145ada509ecSTakashi Iwai 	struct via_spec *spec = codec->spec;
1146ada509ecSTakashi Iwai 
1147ada509ecSTakashi Iwai 	if (active)
11483b607e3dSTakashi Iwai 		spec->opened_streams |= bit;
1149ada509ecSTakashi Iwai 	else
11503b607e3dSTakashi Iwai 		spec->opened_streams &= ~bit;
1151ada509ecSTakashi Iwai 	analog_low_current_mode(codec);
11527eb56e84STakashi Iwai }
11537eb56e84STakashi Iwai 
1154ece8d043STakashi Iwai static int via_playback_multi_pcm_open(struct hda_pcm_stream *hinfo,
1155c577b8a1SJoseph Chan 				 struct hda_codec *codec,
1156c577b8a1SJoseph Chan 				 struct snd_pcm_substream *substream)
1157c577b8a1SJoseph Chan {
1158c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
115925250505STakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
1160ada509ecSTakashi Iwai 	int err;
1161ece8d043STakashi Iwai 
116225250505STakashi Iwai 	spec->multiout.num_dacs = cfg->line_outs + spec->smart51_nums;
116325250505STakashi Iwai 	spec->multiout.max_channels = spec->multiout.num_dacs * 2;
11643b607e3dSTakashi Iwai 	set_stream_open(codec, STREAM_MULTI_OUT, true);
1165ada509ecSTakashi Iwai 	err = snd_hda_multi_out_analog_open(codec, &spec->multiout, substream,
11669a08160bSTakashi Iwai 					    hinfo);
1167ada509ecSTakashi Iwai 	if (err < 0) {
11683b607e3dSTakashi Iwai 		set_stream_open(codec, STREAM_MULTI_OUT, false);
1169ada509ecSTakashi Iwai 		return err;
1170ada509ecSTakashi Iwai 	}
1171ada509ecSTakashi Iwai 	return 0;
1172c577b8a1SJoseph Chan }
1173c577b8a1SJoseph Chan 
1174ece8d043STakashi Iwai static int via_playback_multi_pcm_close(struct hda_pcm_stream *hinfo,
11759af74210STakashi Iwai 				  struct hda_codec *codec,
11769af74210STakashi Iwai 				  struct snd_pcm_substream *substream)
11779af74210STakashi Iwai {
11783b607e3dSTakashi Iwai 	set_stream_open(codec, STREAM_MULTI_OUT, false);
11799af74210STakashi Iwai 	return 0;
11809af74210STakashi Iwai }
11819af74210STakashi Iwai 
11827eb56e84STakashi Iwai static int via_playback_hp_pcm_open(struct hda_pcm_stream *hinfo,
11837eb56e84STakashi Iwai 				    struct hda_codec *codec,
11847eb56e84STakashi Iwai 				    struct snd_pcm_substream *substream)
11857eb56e84STakashi Iwai {
11867eb56e84STakashi Iwai 	struct via_spec *spec = codec->spec;
11877eb56e84STakashi Iwai 
1188ece8d043STakashi Iwai 	if (snd_BUG_ON(!spec->hp_dac_nid))
11897eb56e84STakashi Iwai 		return -EINVAL;
11903b607e3dSTakashi Iwai 	set_stream_open(codec, STREAM_INDEP_HP, true);
1191ece8d043STakashi Iwai 	return 0;
1192ece8d043STakashi Iwai }
1193ece8d043STakashi Iwai 
1194ece8d043STakashi Iwai static int via_playback_hp_pcm_close(struct hda_pcm_stream *hinfo,
1195ece8d043STakashi Iwai 				     struct hda_codec *codec,
1196ece8d043STakashi Iwai 				     struct snd_pcm_substream *substream)
1197ece8d043STakashi Iwai {
11983b607e3dSTakashi Iwai 	set_stream_open(codec, STREAM_INDEP_HP, false);
11997eb56e84STakashi Iwai 	return 0;
12007eb56e84STakashi Iwai }
12017eb56e84STakashi Iwai 
12027eb56e84STakashi Iwai static int via_playback_multi_pcm_prepare(struct hda_pcm_stream *hinfo,
12037eb56e84STakashi Iwai 					  struct hda_codec *codec,
12040aa62aefSHarald Welte 					  unsigned int stream_tag,
12050aa62aefSHarald Welte 					  unsigned int format,
12060aa62aefSHarald Welte 					  struct snd_pcm_substream *substream)
12070aa62aefSHarald Welte {
12080aa62aefSHarald Welte 	struct via_spec *spec = codec->spec;
12090aa62aefSHarald Welte 
12103b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
12113b607e3dSTakashi Iwai 	setup_playback_multi_pcm(spec);
1212ece8d043STakashi Iwai 	snd_hda_multi_out_analog_prepare(codec, &spec->multiout, stream_tag,
1213ece8d043STakashi Iwai 					 format, substream);
12143b607e3dSTakashi Iwai 	/* remember for dynamic DAC switch with indep-HP */
12153b607e3dSTakashi Iwai 	spec->active_streams |= STREAM_MULTI_OUT;
12163b607e3dSTakashi Iwai 	spec->cur_dac_stream_tag = stream_tag;
12173b607e3dSTakashi Iwai 	spec->cur_dac_format = format;
12183b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
1219187d333eSTakashi Iwai 	vt1708_update_hp_work(spec);
12207eb56e84STakashi Iwai 	return 0;
12210aa62aefSHarald Welte }
12220aa62aefSHarald Welte 
12237eb56e84STakashi Iwai static int via_playback_hp_pcm_prepare(struct hda_pcm_stream *hinfo,
12240aa62aefSHarald Welte 				       struct hda_codec *codec,
12250aa62aefSHarald Welte 				       unsigned int stream_tag,
12260aa62aefSHarald Welte 				       unsigned int format,
12270aa62aefSHarald Welte 				       struct snd_pcm_substream *substream)
12280aa62aefSHarald Welte {
12290aa62aefSHarald Welte 	struct via_spec *spec = codec->spec;
12300aa62aefSHarald Welte 
12313b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
12323b607e3dSTakashi Iwai 	if (spec->hp_independent_mode)
1233ece8d043STakashi Iwai 		snd_hda_codec_setup_stream(codec, spec->hp_dac_nid,
1234ece8d043STakashi Iwai 					   stream_tag, 0, format);
12353b607e3dSTakashi Iwai 	spec->active_streams |= STREAM_INDEP_HP;
12363b607e3dSTakashi Iwai 	spec->cur_hp_stream_tag = stream_tag;
12373b607e3dSTakashi Iwai 	spec->cur_hp_format = format;
12383b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
1239187d333eSTakashi Iwai 	vt1708_update_hp_work(spec);
12400aa62aefSHarald Welte 	return 0;
12410aa62aefSHarald Welte }
12420aa62aefSHarald Welte 
12430aa62aefSHarald Welte static int via_playback_multi_pcm_cleanup(struct hda_pcm_stream *hinfo,
12440aa62aefSHarald Welte 				    struct hda_codec *codec,
12450aa62aefSHarald Welte 				    struct snd_pcm_substream *substream)
12460aa62aefSHarald Welte {
12470aa62aefSHarald Welte 	struct via_spec *spec = codec->spec;
12480aa62aefSHarald Welte 
12493b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
1250ece8d043STakashi Iwai 	snd_hda_multi_out_analog_cleanup(codec, &spec->multiout);
12513b607e3dSTakashi Iwai 	spec->active_streams &= ~STREAM_MULTI_OUT;
12523b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
1253187d333eSTakashi Iwai 	vt1708_update_hp_work(spec);
12547eb56e84STakashi Iwai 	return 0;
12550aa62aefSHarald Welte }
12567eb56e84STakashi Iwai 
12577eb56e84STakashi Iwai static int via_playback_hp_pcm_cleanup(struct hda_pcm_stream *hinfo,
12587eb56e84STakashi Iwai 				       struct hda_codec *codec,
12597eb56e84STakashi Iwai 				       struct snd_pcm_substream *substream)
12607eb56e84STakashi Iwai {
12617eb56e84STakashi Iwai 	struct via_spec *spec = codec->spec;
12627eb56e84STakashi Iwai 
12633b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
12643b607e3dSTakashi Iwai 	if (spec->hp_independent_mode)
1265ece8d043STakashi Iwai 		snd_hda_codec_setup_stream(codec, spec->hp_dac_nid, 0, 0, 0);
12663b607e3dSTakashi Iwai 	spec->active_streams &= ~STREAM_INDEP_HP;
12673b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
1268187d333eSTakashi Iwai 	vt1708_update_hp_work(spec);
12690aa62aefSHarald Welte 	return 0;
12700aa62aefSHarald Welte }
12710aa62aefSHarald Welte 
1272c577b8a1SJoseph Chan /*
1273c577b8a1SJoseph Chan  * Digital out
1274c577b8a1SJoseph Chan  */
1275c577b8a1SJoseph Chan static int via_dig_playback_pcm_open(struct hda_pcm_stream *hinfo,
1276c577b8a1SJoseph Chan 				     struct hda_codec *codec,
1277c577b8a1SJoseph Chan 				     struct snd_pcm_substream *substream)
1278c577b8a1SJoseph Chan {
1279c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
1280c577b8a1SJoseph Chan 	return snd_hda_multi_out_dig_open(codec, &spec->multiout);
1281c577b8a1SJoseph Chan }
1282c577b8a1SJoseph Chan 
1283c577b8a1SJoseph Chan static int via_dig_playback_pcm_close(struct hda_pcm_stream *hinfo,
1284c577b8a1SJoseph Chan 				      struct hda_codec *codec,
1285c577b8a1SJoseph Chan 				      struct snd_pcm_substream *substream)
1286c577b8a1SJoseph Chan {
1287c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
1288c577b8a1SJoseph Chan 	return snd_hda_multi_out_dig_close(codec, &spec->multiout);
1289c577b8a1SJoseph Chan }
1290c577b8a1SJoseph Chan 
12915691ec7fSHarald Welte static int via_dig_playback_pcm_prepare(struct hda_pcm_stream *hinfo,
129298aa34c0SHarald Welte 					struct hda_codec *codec,
129398aa34c0SHarald Welte 					unsigned int stream_tag,
129498aa34c0SHarald Welte 					unsigned int format,
129598aa34c0SHarald Welte 					struct snd_pcm_substream *substream)
129698aa34c0SHarald Welte {
129798aa34c0SHarald Welte 	struct via_spec *spec = codec->spec;
12989da29271STakashi Iwai 	return snd_hda_multi_out_dig_prepare(codec, &spec->multiout,
12999da29271STakashi Iwai 					     stream_tag, format, substream);
13009da29271STakashi Iwai }
13015691ec7fSHarald Welte 
13029da29271STakashi Iwai static int via_dig_playback_pcm_cleanup(struct hda_pcm_stream *hinfo,
13039da29271STakashi Iwai 					struct hda_codec *codec,
13049da29271STakashi Iwai 					struct snd_pcm_substream *substream)
13059da29271STakashi Iwai {
13069da29271STakashi Iwai 	struct via_spec *spec = codec->spec;
13079da29271STakashi Iwai 	snd_hda_multi_out_dig_cleanup(codec, &spec->multiout);
130898aa34c0SHarald Welte 	return 0;
130998aa34c0SHarald Welte }
131098aa34c0SHarald Welte 
1311c577b8a1SJoseph Chan /*
1312c577b8a1SJoseph Chan  * Analog capture
1313c577b8a1SJoseph Chan  */
1314c577b8a1SJoseph Chan static int via_capture_pcm_prepare(struct hda_pcm_stream *hinfo,
1315c577b8a1SJoseph Chan 				   struct hda_codec *codec,
1316c577b8a1SJoseph Chan 				   unsigned int stream_tag,
1317c577b8a1SJoseph Chan 				   unsigned int format,
1318c577b8a1SJoseph Chan 				   struct snd_pcm_substream *substream)
1319c577b8a1SJoseph Chan {
1320c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
1321c577b8a1SJoseph Chan 
1322c577b8a1SJoseph Chan 	snd_hda_codec_setup_stream(codec, spec->adc_nids[substream->number],
1323c577b8a1SJoseph Chan 				   stream_tag, 0, format);
1324c577b8a1SJoseph Chan 	return 0;
1325c577b8a1SJoseph Chan }
1326c577b8a1SJoseph Chan 
1327c577b8a1SJoseph Chan static int via_capture_pcm_cleanup(struct hda_pcm_stream *hinfo,
1328c577b8a1SJoseph Chan 				   struct hda_codec *codec,
1329c577b8a1SJoseph Chan 				   struct snd_pcm_substream *substream)
1330c577b8a1SJoseph Chan {
1331c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
1332888afa15STakashi Iwai 	snd_hda_codec_cleanup_stream(codec, spec->adc_nids[substream->number]);
1333c577b8a1SJoseph Chan 	return 0;
1334c577b8a1SJoseph Chan }
1335c577b8a1SJoseph Chan 
1336a86a88eaSTakashi Iwai /* analog capture with dynamic ADC switching */
1337a86a88eaSTakashi Iwai static int via_dyn_adc_capture_pcm_prepare(struct hda_pcm_stream *hinfo,
1338a86a88eaSTakashi Iwai 					   struct hda_codec *codec,
1339a86a88eaSTakashi Iwai 					   unsigned int stream_tag,
1340a86a88eaSTakashi Iwai 					   unsigned int format,
1341a86a88eaSTakashi Iwai 					   struct snd_pcm_substream *substream)
1342a86a88eaSTakashi Iwai {
1343a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
1344a86a88eaSTakashi Iwai 	int adc_idx = spec->inputs[spec->cur_mux[0]].adc_idx;
1345a86a88eaSTakashi Iwai 
13463b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
1347a86a88eaSTakashi Iwai 	spec->cur_adc = spec->adc_nids[adc_idx];
1348a86a88eaSTakashi Iwai 	spec->cur_adc_stream_tag = stream_tag;
1349a86a88eaSTakashi Iwai 	spec->cur_adc_format = format;
1350a86a88eaSTakashi Iwai 	snd_hda_codec_setup_stream(codec, spec->cur_adc, stream_tag, 0, format);
13513b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
1352a86a88eaSTakashi Iwai 	return 0;
1353a86a88eaSTakashi Iwai }
1354a86a88eaSTakashi Iwai 
1355a86a88eaSTakashi Iwai static int via_dyn_adc_capture_pcm_cleanup(struct hda_pcm_stream *hinfo,
1356a86a88eaSTakashi Iwai 					   struct hda_codec *codec,
1357a86a88eaSTakashi Iwai 					   struct snd_pcm_substream *substream)
1358a86a88eaSTakashi Iwai {
1359a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
1360a86a88eaSTakashi Iwai 
13613b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
1362a86a88eaSTakashi Iwai 	snd_hda_codec_cleanup_stream(codec, spec->cur_adc);
1363a86a88eaSTakashi Iwai 	spec->cur_adc = 0;
13643b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
1365a86a88eaSTakashi Iwai 	return 0;
1366a86a88eaSTakashi Iwai }
1367a86a88eaSTakashi Iwai 
1368a86a88eaSTakashi Iwai /* re-setup the stream if running; called from input-src put */
1369a86a88eaSTakashi Iwai static bool via_dyn_adc_pcm_resetup(struct hda_codec *codec, int cur)
1370a86a88eaSTakashi Iwai {
1371a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
1372a86a88eaSTakashi Iwai 	int adc_idx = spec->inputs[cur].adc_idx;
1373a86a88eaSTakashi Iwai 	hda_nid_t adc = spec->adc_nids[adc_idx];
13743b607e3dSTakashi Iwai 	bool ret = false;
1375a86a88eaSTakashi Iwai 
13763b607e3dSTakashi Iwai 	mutex_lock(&spec->config_mutex);
1377a86a88eaSTakashi Iwai 	if (spec->cur_adc && spec->cur_adc != adc) {
1378a86a88eaSTakashi Iwai 		/* stream is running, let's swap the current ADC */
1379a86a88eaSTakashi Iwai 		__snd_hda_codec_cleanup_stream(codec, spec->cur_adc, 1);
1380a86a88eaSTakashi Iwai 		spec->cur_adc = adc;
1381a86a88eaSTakashi Iwai 		snd_hda_codec_setup_stream(codec, adc,
1382a86a88eaSTakashi Iwai 					   spec->cur_adc_stream_tag, 0,
1383a86a88eaSTakashi Iwai 					   spec->cur_adc_format);
13843b607e3dSTakashi Iwai 		ret = true;
1385a86a88eaSTakashi Iwai 	}
13863b607e3dSTakashi Iwai 	mutex_unlock(&spec->config_mutex);
13873b607e3dSTakashi Iwai 	return ret;
1388a86a88eaSTakashi Iwai }
1389a86a88eaSTakashi Iwai 
13909af74210STakashi Iwai static const struct hda_pcm_stream via_pcm_analog_playback = {
13917eb56e84STakashi Iwai 	.substreams = 1,
1392c577b8a1SJoseph Chan 	.channels_min = 2,
1393c577b8a1SJoseph Chan 	.channels_max = 8,
13949af74210STakashi Iwai 	/* NID is set in via_build_pcms */
1395c577b8a1SJoseph Chan 	.ops = {
1396ece8d043STakashi Iwai 		.open = via_playback_multi_pcm_open,
1397ece8d043STakashi Iwai 		.close = via_playback_multi_pcm_close,
13980aa62aefSHarald Welte 		.prepare = via_playback_multi_pcm_prepare,
13990aa62aefSHarald Welte 		.cleanup = via_playback_multi_pcm_cleanup
1400c577b8a1SJoseph Chan 	},
1401c577b8a1SJoseph Chan };
1402c577b8a1SJoseph Chan 
14037eb56e84STakashi Iwai static const struct hda_pcm_stream via_pcm_hp_playback = {
14047eb56e84STakashi Iwai 	.substreams = 1,
14057eb56e84STakashi Iwai 	.channels_min = 2,
14067eb56e84STakashi Iwai 	.channels_max = 2,
14077eb56e84STakashi Iwai 	/* NID is set in via_build_pcms */
14087eb56e84STakashi Iwai 	.ops = {
14097eb56e84STakashi Iwai 		.open = via_playback_hp_pcm_open,
1410ece8d043STakashi Iwai 		.close = via_playback_hp_pcm_close,
14117eb56e84STakashi Iwai 		.prepare = via_playback_hp_pcm_prepare,
14127eb56e84STakashi Iwai 		.cleanup = via_playback_hp_pcm_cleanup
14137eb56e84STakashi Iwai 	},
14147eb56e84STakashi Iwai };
14157eb56e84STakashi Iwai 
141690dd48a1STakashi Iwai static const struct hda_pcm_stream vt1708_pcm_analog_s16_playback = {
14177eb56e84STakashi Iwai 	.substreams = 1,
1418bc9b5623STakashi Iwai 	.channels_min = 2,
1419bc9b5623STakashi Iwai 	.channels_max = 8,
14209af74210STakashi Iwai 	/* NID is set in via_build_pcms */
1421bc9b5623STakashi Iwai 	/* We got noisy outputs on the right channel on VT1708 when
1422bc9b5623STakashi Iwai 	 * 24bit samples are used.  Until any workaround is found,
1423bc9b5623STakashi Iwai 	 * disable the 24bit format, so far.
1424bc9b5623STakashi Iwai 	 */
1425bc9b5623STakashi Iwai 	.formats = SNDRV_PCM_FMTBIT_S16_LE,
1426bc9b5623STakashi Iwai 	.ops = {
1427ece8d043STakashi Iwai 		.open = via_playback_multi_pcm_open,
1428ece8d043STakashi Iwai 		.close = via_playback_multi_pcm_close,
1429c873cc25SLydia Wang 		.prepare = via_playback_multi_pcm_prepare,
1430c873cc25SLydia Wang 		.cleanup = via_playback_multi_pcm_cleanup
1431bc9b5623STakashi Iwai 	},
1432bc9b5623STakashi Iwai };
1433bc9b5623STakashi Iwai 
14349af74210STakashi Iwai static const struct hda_pcm_stream via_pcm_analog_capture = {
14357eb56e84STakashi Iwai 	.substreams = 1, /* will be changed in via_build_pcms() */
1436c577b8a1SJoseph Chan 	.channels_min = 2,
1437c577b8a1SJoseph Chan 	.channels_max = 2,
14389af74210STakashi Iwai 	/* NID is set in via_build_pcms */
1439c577b8a1SJoseph Chan 	.ops = {
1440c577b8a1SJoseph Chan 		.prepare = via_capture_pcm_prepare,
1441c577b8a1SJoseph Chan 		.cleanup = via_capture_pcm_cleanup
1442c577b8a1SJoseph Chan 	},
1443c577b8a1SJoseph Chan };
1444c577b8a1SJoseph Chan 
1445a86a88eaSTakashi Iwai static const struct hda_pcm_stream via_pcm_dyn_adc_analog_capture = {
1446a86a88eaSTakashi Iwai 	.substreams = 1,
1447a86a88eaSTakashi Iwai 	.channels_min = 2,
1448a86a88eaSTakashi Iwai 	.channels_max = 2,
1449a86a88eaSTakashi Iwai 	/* NID is set in via_build_pcms */
1450a86a88eaSTakashi Iwai 	.ops = {
1451a86a88eaSTakashi Iwai 		.prepare = via_dyn_adc_capture_pcm_prepare,
1452a86a88eaSTakashi Iwai 		.cleanup = via_dyn_adc_capture_pcm_cleanup,
1453a86a88eaSTakashi Iwai 	},
1454a86a88eaSTakashi Iwai };
1455a86a88eaSTakashi Iwai 
14569af74210STakashi Iwai static const struct hda_pcm_stream via_pcm_digital_playback = {
1457c577b8a1SJoseph Chan 	.substreams = 1,
1458c577b8a1SJoseph Chan 	.channels_min = 2,
1459c577b8a1SJoseph Chan 	.channels_max = 2,
1460c577b8a1SJoseph Chan 	/* NID is set in via_build_pcms */
1461c577b8a1SJoseph Chan 	.ops = {
1462c577b8a1SJoseph Chan 		.open = via_dig_playback_pcm_open,
14636b97eb45STakashi Iwai 		.close = via_dig_playback_pcm_close,
14649da29271STakashi Iwai 		.prepare = via_dig_playback_pcm_prepare,
14659da29271STakashi Iwai 		.cleanup = via_dig_playback_pcm_cleanup
1466c577b8a1SJoseph Chan 	},
1467c577b8a1SJoseph Chan };
1468c577b8a1SJoseph Chan 
14699af74210STakashi Iwai static const struct hda_pcm_stream via_pcm_digital_capture = {
1470c577b8a1SJoseph Chan 	.substreams = 1,
1471c577b8a1SJoseph Chan 	.channels_min = 2,
1472c577b8a1SJoseph Chan 	.channels_max = 2,
1473c577b8a1SJoseph Chan };
1474c577b8a1SJoseph Chan 
1475370bafbdSTakashi Iwai /*
1476370bafbdSTakashi Iwai  * slave controls for virtual master
1477370bafbdSTakashi Iwai  */
14789322ca54STakashi Iwai static const char * const via_slave_pfxs[] = {
14799322ca54STakashi Iwai 	"Front", "Surround", "Center", "LFE", "Side",
1480f37bc7a8STakashi Iwai 	"Headphone", "Speaker", "Bass Speaker",
1481370bafbdSTakashi Iwai 	NULL,
1482370bafbdSTakashi Iwai };
1483370bafbdSTakashi Iwai 
1484c577b8a1SJoseph Chan static int via_build_controls(struct hda_codec *codec)
1485c577b8a1SJoseph Chan {
1486c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
14875b0cb1d8SJaroslav Kysela 	struct snd_kcontrol *kctl;
14885b0cb1d8SJaroslav Kysela 	int err, i;
1489c577b8a1SJoseph Chan 
1490b5bcc189STakashi Iwai 	spec->no_pin_power_ctl = 1;
149124088a58STakashi Iwai 	if (spec->set_widgets_power_state)
149224088a58STakashi Iwai 		if (!via_clone_control(spec, &via_pin_power_ctl_enum))
149324088a58STakashi Iwai 			return -ENOMEM;
149424088a58STakashi Iwai 
1495c577b8a1SJoseph Chan 	for (i = 0; i < spec->num_mixers; i++) {
1496c577b8a1SJoseph Chan 		err = snd_hda_add_new_ctls(codec, spec->mixers[i]);
1497c577b8a1SJoseph Chan 		if (err < 0)
1498c577b8a1SJoseph Chan 			return err;
1499c577b8a1SJoseph Chan 	}
1500c577b8a1SJoseph Chan 
1501c577b8a1SJoseph Chan 	if (spec->multiout.dig_out_nid) {
1502c577b8a1SJoseph Chan 		err = snd_hda_create_spdif_out_ctls(codec,
150374b654c9SStephen Warren 						    spec->multiout.dig_out_nid,
1504c577b8a1SJoseph Chan 						    spec->multiout.dig_out_nid);
1505c577b8a1SJoseph Chan 		if (err < 0)
1506c577b8a1SJoseph Chan 			return err;
15079a08160bSTakashi Iwai 		err = snd_hda_create_spdif_share_sw(codec,
15089a08160bSTakashi Iwai 						    &spec->multiout);
15099a08160bSTakashi Iwai 		if (err < 0)
15109a08160bSTakashi Iwai 			return err;
15119a08160bSTakashi Iwai 		spec->multiout.share_spdif = 1;
1512c577b8a1SJoseph Chan 	}
1513c577b8a1SJoseph Chan 	if (spec->dig_in_nid) {
1514c577b8a1SJoseph Chan 		err = snd_hda_create_spdif_in_ctls(codec, spec->dig_in_nid);
1515c577b8a1SJoseph Chan 		if (err < 0)
1516c577b8a1SJoseph Chan 			return err;
1517c577b8a1SJoseph Chan 	}
151817314379SLydia Wang 
1519370bafbdSTakashi Iwai 	/* if we have no master control, let's create it */
1520370bafbdSTakashi Iwai 	if (!snd_hda_find_mixer_ctl(codec, "Master Playback Volume")) {
1521370bafbdSTakashi Iwai 		unsigned int vmaster_tlv[4];
1522370bafbdSTakashi Iwai 		snd_hda_set_vmaster_tlv(codec, spec->multiout.dac_nids[0],
1523370bafbdSTakashi Iwai 					HDA_OUTPUT, vmaster_tlv);
1524370bafbdSTakashi Iwai 		err = snd_hda_add_vmaster(codec, "Master Playback Volume",
15259322ca54STakashi Iwai 					  vmaster_tlv, via_slave_pfxs,
15269322ca54STakashi Iwai 					  "Playback Volume");
1527370bafbdSTakashi Iwai 		if (err < 0)
1528370bafbdSTakashi Iwai 			return err;
1529370bafbdSTakashi Iwai 	}
1530370bafbdSTakashi Iwai 	if (!snd_hda_find_mixer_ctl(codec, "Master Playback Switch")) {
1531370bafbdSTakashi Iwai 		err = snd_hda_add_vmaster(codec, "Master Playback Switch",
15329322ca54STakashi Iwai 					  NULL, via_slave_pfxs,
15339322ca54STakashi Iwai 					  "Playback Switch");
1534370bafbdSTakashi Iwai 		if (err < 0)
1535370bafbdSTakashi Iwai 			return err;
1536370bafbdSTakashi Iwai 	}
1537370bafbdSTakashi Iwai 
15385b0cb1d8SJaroslav Kysela 	/* assign Capture Source enums to NID */
15395b0cb1d8SJaroslav Kysela 	kctl = snd_hda_find_mixer_ctl(codec, "Input Source");
15405b0cb1d8SJaroslav Kysela 	for (i = 0; kctl && i < kctl->count; i++) {
154177e314f7STakashi Iwai 		if (!spec->mux_nids[i])
154277e314f7STakashi Iwai 			continue;
154321949f00STakashi Iwai 		err = snd_hda_add_nid(codec, kctl, i, spec->mux_nids[i]);
15445b0cb1d8SJaroslav Kysela 		if (err < 0)
15455b0cb1d8SJaroslav Kysela 			return err;
15465b0cb1d8SJaroslav Kysela 	}
15475b0cb1d8SJaroslav Kysela 
1548603c4019STakashi Iwai 	via_free_kctls(codec); /* no longer needed */
154901a61e12STakashi Iwai 
155001a61e12STakashi Iwai 	err = snd_hda_jack_add_kctls(codec, &spec->autocfg);
155101a61e12STakashi Iwai 	if (err < 0)
155201a61e12STakashi Iwai 		return err;
155301a61e12STakashi Iwai 
1554c577b8a1SJoseph Chan 	return 0;
1555c577b8a1SJoseph Chan }
1556c577b8a1SJoseph Chan 
1557c577b8a1SJoseph Chan static int via_build_pcms(struct hda_codec *codec)
1558c577b8a1SJoseph Chan {
1559c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
1560c577b8a1SJoseph Chan 	struct hda_pcm *info = spec->pcm_rec;
1561c577b8a1SJoseph Chan 
1562a5973103STakashi Iwai 	codec->num_pcms = 0;
1563c577b8a1SJoseph Chan 	codec->pcm_info = info;
1564c577b8a1SJoseph Chan 
1565a5973103STakashi Iwai 	if (spec->multiout.num_dacs || spec->num_adc_nids) {
1566a5973103STakashi Iwai 		snprintf(spec->stream_name_analog,
1567a5973103STakashi Iwai 			 sizeof(spec->stream_name_analog),
156882673bc8STakashi Iwai 			 "%s Analog", codec->chip_name);
1569c577b8a1SJoseph Chan 		info->name = spec->stream_name_analog;
15709af74210STakashi Iwai 
1571a5973103STakashi Iwai 		if (spec->multiout.num_dacs) {
15729af74210STakashi Iwai 			if (!spec->stream_analog_playback)
1573a5973103STakashi Iwai 				spec->stream_analog_playback =
1574a5973103STakashi Iwai 					&via_pcm_analog_playback;
1575377ff31aSLydia Wang 			info->stream[SNDRV_PCM_STREAM_PLAYBACK] =
15769af74210STakashi Iwai 				*spec->stream_analog_playback;
1577377ff31aSLydia Wang 			info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid =
1578377ff31aSLydia Wang 				spec->multiout.dac_nids[0];
1579c577b8a1SJoseph Chan 			info->stream[SNDRV_PCM_STREAM_PLAYBACK].channels_max =
1580c577b8a1SJoseph Chan 				spec->multiout.max_channels;
1581ee81abb6STakashi Iwai 			if (spec->autocfg.line_out_type == AUTO_PIN_SPEAKER_OUT
1582ee81abb6STakashi Iwai 			    && spec->autocfg.line_outs == 2)
1583ee81abb6STakashi Iwai 				info->stream[SNDRV_PCM_STREAM_PLAYBACK].chmap =
1584ee81abb6STakashi Iwai 					snd_pcm_2_1_chmaps;
1585a5973103STakashi Iwai 		}
15869af74210STakashi Iwai 
1587a86a88eaSTakashi Iwai 		if (!spec->stream_analog_capture) {
1588a86a88eaSTakashi Iwai 			if (spec->dyn_adc_switch)
1589a86a88eaSTakashi Iwai 				spec->stream_analog_capture =
1590a86a88eaSTakashi Iwai 					&via_pcm_dyn_adc_analog_capture;
1591a86a88eaSTakashi Iwai 			else
1592a5973103STakashi Iwai 				spec->stream_analog_capture =
1593a5973103STakashi Iwai 					&via_pcm_analog_capture;
1594a86a88eaSTakashi Iwai 		}
1595a5973103STakashi Iwai 		if (spec->num_adc_nids) {
15969af74210STakashi Iwai 			info->stream[SNDRV_PCM_STREAM_CAPTURE] =
15979af74210STakashi Iwai 				*spec->stream_analog_capture;
1598a5973103STakashi Iwai 			info->stream[SNDRV_PCM_STREAM_CAPTURE].nid =
1599a5973103STakashi Iwai 				spec->adc_nids[0];
1600a86a88eaSTakashi Iwai 			if (!spec->dyn_adc_switch)
16019af74210STakashi Iwai 				info->stream[SNDRV_PCM_STREAM_CAPTURE].substreams =
16029af74210STakashi Iwai 					spec->num_adc_nids;
1603a5973103STakashi Iwai 		}
1604c577b8a1SJoseph Chan 		codec->num_pcms++;
1605c577b8a1SJoseph Chan 		info++;
1606a5973103STakashi Iwai 	}
1607a5973103STakashi Iwai 
1608a5973103STakashi Iwai 	if (spec->multiout.dig_out_nid || spec->dig_in_nid) {
160982673bc8STakashi Iwai 		snprintf(spec->stream_name_digital,
161082673bc8STakashi Iwai 			 sizeof(spec->stream_name_digital),
161182673bc8STakashi Iwai 			 "%s Digital", codec->chip_name);
1612c577b8a1SJoseph Chan 		info->name = spec->stream_name_digital;
16137ba72ba1STakashi Iwai 		info->pcm_type = HDA_PCM_TYPE_SPDIF;
1614c577b8a1SJoseph Chan 		if (spec->multiout.dig_out_nid) {
16159af74210STakashi Iwai 			if (!spec->stream_digital_playback)
16169af74210STakashi Iwai 				spec->stream_digital_playback =
16179af74210STakashi Iwai 					&via_pcm_digital_playback;
1618c577b8a1SJoseph Chan 			info->stream[SNDRV_PCM_STREAM_PLAYBACK] =
16199af74210STakashi Iwai 				*spec->stream_digital_playback;
1620c577b8a1SJoseph Chan 			info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid =
1621c577b8a1SJoseph Chan 				spec->multiout.dig_out_nid;
1622c577b8a1SJoseph Chan 		}
1623c577b8a1SJoseph Chan 		if (spec->dig_in_nid) {
16249af74210STakashi Iwai 			if (!spec->stream_digital_capture)
16259af74210STakashi Iwai 				spec->stream_digital_capture =
16269af74210STakashi Iwai 					&via_pcm_digital_capture;
1627c577b8a1SJoseph Chan 			info->stream[SNDRV_PCM_STREAM_CAPTURE] =
16289af74210STakashi Iwai 				*spec->stream_digital_capture;
1629c577b8a1SJoseph Chan 			info->stream[SNDRV_PCM_STREAM_CAPTURE].nid =
1630c577b8a1SJoseph Chan 				spec->dig_in_nid;
1631c577b8a1SJoseph Chan 		}
1632a5973103STakashi Iwai 		codec->num_pcms++;
1633a5973103STakashi Iwai 		info++;
1634c577b8a1SJoseph Chan 	}
1635c577b8a1SJoseph Chan 
1636ece8d043STakashi Iwai 	if (spec->hp_dac_nid) {
16377eb56e84STakashi Iwai 		snprintf(spec->stream_name_hp, sizeof(spec->stream_name_hp),
16387eb56e84STakashi Iwai 			 "%s HP", codec->chip_name);
16397eb56e84STakashi Iwai 		info->name = spec->stream_name_hp;
16407eb56e84STakashi Iwai 		info->stream[SNDRV_PCM_STREAM_PLAYBACK] = via_pcm_hp_playback;
16417eb56e84STakashi Iwai 		info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid =
1642ece8d043STakashi Iwai 			spec->hp_dac_nid;
1643a5973103STakashi Iwai 		codec->num_pcms++;
1644a5973103STakashi Iwai 		info++;
16457eb56e84STakashi Iwai 	}
1646c577b8a1SJoseph Chan 	return 0;
1647c577b8a1SJoseph Chan }
1648c577b8a1SJoseph Chan 
1649c577b8a1SJoseph Chan static void via_free(struct hda_codec *codec)
1650c577b8a1SJoseph Chan {
1651c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
1652c577b8a1SJoseph Chan 
1653c577b8a1SJoseph Chan 	if (!spec)
1654c577b8a1SJoseph Chan 		return;
1655c577b8a1SJoseph Chan 
1656603c4019STakashi Iwai 	via_free_kctls(codec);
16571f2e99feSLydia Wang 	vt1708_stop_hp_work(spec);
1658a86a88eaSTakashi Iwai 	kfree(spec->bind_cap_vol);
1659a86a88eaSTakashi Iwai 	kfree(spec->bind_cap_sw);
16607819d1c7STakashi Iwai 	snd_hda_gen_free(&spec->gen);
1661a86a88eaSTakashi Iwai 	kfree(spec);
1662c577b8a1SJoseph Chan }
1663c577b8a1SJoseph Chan 
166464be285bSTakashi Iwai /* mute/unmute outputs */
166564be285bSTakashi Iwai static void toggle_output_mutes(struct hda_codec *codec, int num_pins,
166664be285bSTakashi Iwai 				hda_nid_t *pins, bool mute)
166764be285bSTakashi Iwai {
166864be285bSTakashi Iwai 	int i;
166994994734STakashi Iwai 	for (i = 0; i < num_pins; i++) {
167094994734STakashi Iwai 		unsigned int parm = snd_hda_codec_read(codec, pins[i], 0,
167194994734STakashi Iwai 					  AC_VERB_GET_PIN_WIDGET_CONTROL, 0);
167294994734STakashi Iwai 		if (parm & AC_PINCTL_IN_EN)
167394994734STakashi Iwai 			continue;
167494994734STakashi Iwai 		if (mute)
167594994734STakashi Iwai 			parm &= ~AC_PINCTL_OUT_EN;
167694994734STakashi Iwai 		else
167794994734STakashi Iwai 			parm |= AC_PINCTL_OUT_EN;
1678cdd03cedSTakashi Iwai 		snd_hda_set_pin_ctl(codec, pins[i], parm);
167994994734STakashi Iwai 	}
168064be285bSTakashi Iwai }
168164be285bSTakashi Iwai 
16824a918ffeSTakashi Iwai /* mute internal speaker if line-out is plugged */
16834a918ffeSTakashi Iwai static void via_line_automute(struct hda_codec *codec, int present)
16844a918ffeSTakashi Iwai {
16854a918ffeSTakashi Iwai 	struct via_spec *spec = codec->spec;
16864a918ffeSTakashi Iwai 
16874a918ffeSTakashi Iwai 	if (!spec->autocfg.speaker_outs)
16884a918ffeSTakashi Iwai 		return;
16894a918ffeSTakashi Iwai 	if (!present)
16904a918ffeSTakashi Iwai 		present = snd_hda_jack_detect(codec,
16914a918ffeSTakashi Iwai 					      spec->autocfg.line_out_pins[0]);
16924a918ffeSTakashi Iwai 	toggle_output_mutes(codec, spec->autocfg.speaker_outs,
16934a918ffeSTakashi Iwai 			    spec->autocfg.speaker_pins,
16944a918ffeSTakashi Iwai 			    present);
16954a918ffeSTakashi Iwai }
16964a918ffeSTakashi Iwai 
169769e52a80SHarald Welte /* mute internal speaker if HP is plugged */
169869e52a80SHarald Welte static void via_hp_automute(struct hda_codec *codec)
169969e52a80SHarald Welte {
17004a918ffeSTakashi Iwai 	int present = 0;
17016e969d91STakashi Iwai 	int nums;
170269e52a80SHarald Welte 	struct via_spec *spec = codec->spec;
170369e52a80SHarald Welte 
1704187d333eSTakashi Iwai 	if (!spec->hp_independent_mode && spec->autocfg.hp_pins[0] &&
1705cf55e904SHerton Ronaldo Krzesinski 	    (spec->codec_type != VT1708 || spec->vt1708_jack_detect) &&
1706cf55e904SHerton Ronaldo Krzesinski 	    is_jack_detectable(codec, spec->autocfg.hp_pins[0]))
1707d56757abSTakashi Iwai 		present = snd_hda_jack_detect(codec, spec->autocfg.hp_pins[0]);
17086e969d91STakashi Iwai 
1709f2b1c9f0STakashi Iwai 	if (spec->smart51_enabled)
1710f2b1c9f0STakashi Iwai 		nums = spec->autocfg.line_outs + spec->smart51_nums;
1711f2b1c9f0STakashi Iwai 	else
1712f2b1c9f0STakashi Iwai 		nums = spec->autocfg.line_outs;
17136e969d91STakashi Iwai 	toggle_output_mutes(codec, nums, spec->autocfg.line_out_pins, present);
17146e969d91STakashi Iwai 
17154a918ffeSTakashi Iwai 	via_line_automute(codec, present);
1716f3db423dSLydia Wang }
1717f3db423dSLydia Wang 
17182a43952aSTakashi Iwai #ifdef CONFIG_PM
171968cb2b55STakashi Iwai static int via_suspend(struct hda_codec *codec)
17201f2e99feSLydia Wang {
17211f2e99feSLydia Wang 	struct via_spec *spec = codec->spec;
17221f2e99feSLydia Wang 	vt1708_stop_hp_work(spec);
172394c142a1SDavid Henningsson 
172494c142a1SDavid Henningsson 	if (spec->codec_type == VT1802) {
172594c142a1SDavid Henningsson 		/* Fix pop noise on headphones */
172694c142a1SDavid Henningsson 		int i;
172794c142a1SDavid Henningsson 		for (i = 0; i < spec->autocfg.hp_outs; i++)
172894c142a1SDavid Henningsson 			snd_hda_set_pin_ctl(codec, spec->autocfg.hp_pins[i], 0);
172994c142a1SDavid Henningsson 	}
173094c142a1SDavid Henningsson 
17311f2e99feSLydia Wang 	return 0;
17321f2e99feSLydia Wang }
17331f2e99feSLydia Wang #endif
17341f2e99feSLydia Wang 
173583012a7cSTakashi Iwai #ifdef CONFIG_PM
1736cb53c626STakashi Iwai static int via_check_power_status(struct hda_codec *codec, hda_nid_t nid)
1737cb53c626STakashi Iwai {
1738cb53c626STakashi Iwai 	struct via_spec *spec = codec->spec;
1739cb53c626STakashi Iwai 	return snd_hda_check_amp_list_power(codec, &spec->loopback, nid);
1740cb53c626STakashi Iwai }
1741cb53c626STakashi Iwai #endif
1742cb53c626STakashi Iwai 
1743c577b8a1SJoseph Chan /*
1744c577b8a1SJoseph Chan  */
17455d41762aSTakashi Iwai 
17465d41762aSTakashi Iwai static int via_init(struct hda_codec *codec);
17475d41762aSTakashi Iwai 
174890dd48a1STakashi Iwai static const struct hda_codec_ops via_patch_ops = {
1749c577b8a1SJoseph Chan 	.build_controls = via_build_controls,
1750c577b8a1SJoseph Chan 	.build_pcms = via_build_pcms,
1751c577b8a1SJoseph Chan 	.init = via_init,
1752c577b8a1SJoseph Chan 	.free = via_free,
17534e2d16d3SDavid Henningsson 	.unsol_event = snd_hda_jack_unsol_event,
17542a43952aSTakashi Iwai #ifdef CONFIG_PM
17551f2e99feSLydia Wang 	.suspend = via_suspend,
1756cb53c626STakashi Iwai 	.check_power_status = via_check_power_status,
1757cb53c626STakashi Iwai #endif
1758c577b8a1SJoseph Chan };
1759c577b8a1SJoseph Chan 
17604a79616dSTakashi Iwai static bool is_empty_dac(struct hda_codec *codec, hda_nid_t dac)
1761c577b8a1SJoseph Chan {
17624a79616dSTakashi Iwai 	struct via_spec *spec = codec->spec;
17634a79616dSTakashi Iwai 	int i;
17644a79616dSTakashi Iwai 
17654a79616dSTakashi Iwai 	for (i = 0; i < spec->multiout.num_dacs; i++) {
17664a79616dSTakashi Iwai 		if (spec->multiout.dac_nids[i] == dac)
17674a79616dSTakashi Iwai 			return false;
17684a79616dSTakashi Iwai 	}
1769ece8d043STakashi Iwai 	if (spec->hp_dac_nid == dac)
17704a79616dSTakashi Iwai 		return false;
17714a79616dSTakashi Iwai 	return true;
17724a79616dSTakashi Iwai }
17734a79616dSTakashi Iwai 
17748e3679dcSTakashi Iwai static bool __parse_output_path(struct hda_codec *codec, hda_nid_t nid,
17753214b966STakashi Iwai 				hda_nid_t target_dac, int with_aa_mix,
17763214b966STakashi Iwai 				struct nid_path *path, int depth)
17774a79616dSTakashi Iwai {
17783214b966STakashi Iwai 	struct via_spec *spec = codec->spec;
17794a79616dSTakashi Iwai 	hda_nid_t conn[8];
17804a79616dSTakashi Iwai 	int i, nums;
17814a79616dSTakashi Iwai 
17823214b966STakashi Iwai 	if (nid == spec->aa_mix_nid) {
17833214b966STakashi Iwai 		if (!with_aa_mix)
17843214b966STakashi Iwai 			return false;
17853214b966STakashi Iwai 		with_aa_mix = 2; /* mark aa-mix is included */
17863214b966STakashi Iwai 	}
17873214b966STakashi Iwai 
17884a79616dSTakashi Iwai 	nums = snd_hda_get_connections(codec, nid, conn, ARRAY_SIZE(conn));
17894a79616dSTakashi Iwai 	for (i = 0; i < nums; i++) {
17904a79616dSTakashi Iwai 		if (get_wcaps_type(get_wcaps(codec, conn[i])) != AC_WID_AUD_OUT)
17914a79616dSTakashi Iwai 			continue;
17923214b966STakashi Iwai 		if (conn[i] == target_dac || is_empty_dac(codec, conn[i])) {
17933214b966STakashi Iwai 			/* aa-mix is requested but not included? */
17943214b966STakashi Iwai 			if (!(spec->aa_mix_nid && with_aa_mix == 1))
179509a9ad69STakashi Iwai 				goto found;
17964a79616dSTakashi Iwai 		}
17973214b966STakashi Iwai 	}
17988e3679dcSTakashi Iwai 	if (depth >= MAX_NID_PATH_DEPTH)
17994a79616dSTakashi Iwai 		return false;
18004a79616dSTakashi Iwai 	for (i = 0; i < nums; i++) {
18014a79616dSTakashi Iwai 		unsigned int type;
18024a79616dSTakashi Iwai 		type = get_wcaps_type(get_wcaps(codec, conn[i]));
18033214b966STakashi Iwai 		if (type == AC_WID_AUD_OUT)
18044a79616dSTakashi Iwai 			continue;
18058e3679dcSTakashi Iwai 		if (__parse_output_path(codec, conn[i], target_dac,
18063214b966STakashi Iwai 					with_aa_mix, path, depth + 1))
180709a9ad69STakashi Iwai 			goto found;
18084a79616dSTakashi Iwai 	}
18094a79616dSTakashi Iwai 	return false;
181009a9ad69STakashi Iwai 
181109a9ad69STakashi Iwai  found:
181209a9ad69STakashi Iwai 	path->path[path->depth] = conn[i];
181309a9ad69STakashi Iwai 	path->idx[path->depth] = i;
181409a9ad69STakashi Iwai 	if (nums > 1 && get_wcaps_type(get_wcaps(codec, nid)) != AC_WID_AUD_MIX)
181509a9ad69STakashi Iwai 		path->multi[path->depth] = 1;
181609a9ad69STakashi Iwai 	path->depth++;
181709a9ad69STakashi Iwai 	return true;
18184a79616dSTakashi Iwai }
18194a79616dSTakashi Iwai 
18208e3679dcSTakashi Iwai static bool parse_output_path(struct hda_codec *codec, hda_nid_t nid,
18213214b966STakashi Iwai 			      hda_nid_t target_dac, int with_aa_mix,
18223214b966STakashi Iwai 			      struct nid_path *path)
18238e3679dcSTakashi Iwai {
18243214b966STakashi Iwai 	if (__parse_output_path(codec, nid, target_dac, with_aa_mix, path, 1)) {
18258e3679dcSTakashi Iwai 		path->path[path->depth] = nid;
18268e3679dcSTakashi Iwai 		path->depth++;
18273214b966STakashi Iwai 		snd_printdd("output-path: depth=%d, %02x/%02x/%02x/%02x/%02x\n",
18283214b966STakashi Iwai 			    path->depth, path->path[0], path->path[1],
18293214b966STakashi Iwai 			    path->path[2], path->path[3], path->path[4]);
18308e3679dcSTakashi Iwai 		return true;
18318e3679dcSTakashi Iwai 	}
18328e3679dcSTakashi Iwai 	return false;
18338e3679dcSTakashi Iwai }
18348e3679dcSTakashi Iwai 
18354a79616dSTakashi Iwai static int via_auto_fill_dac_nids(struct hda_codec *codec)
18364a79616dSTakashi Iwai {
18374a79616dSTakashi Iwai 	struct via_spec *spec = codec->spec;
18384a79616dSTakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
18395b376195STakashi Iwai 	int i;
1840c577b8a1SJoseph Chan 	hda_nid_t nid;
1841c577b8a1SJoseph Chan 
18425b376195STakashi Iwai 	spec->multiout.num_dacs = 0;
1843c577b8a1SJoseph Chan 	spec->multiout.dac_nids = spec->private_dac_nids;
18444a79616dSTakashi Iwai 	for (i = 0; i < cfg->line_outs; i++) {
18453214b966STakashi Iwai 		hda_nid_t dac = 0;
1846c577b8a1SJoseph Chan 		nid = cfg->line_out_pins[i];
18474a79616dSTakashi Iwai 		if (!nid)
18484a79616dSTakashi Iwai 			continue;
18493214b966STakashi Iwai 		if (parse_output_path(codec, nid, 0, 0, &spec->out_path[i]))
18503214b966STakashi Iwai 			dac = spec->out_path[i].path[0];
18513214b966STakashi Iwai 		if (!i && parse_output_path(codec, nid, dac, 1,
18523214b966STakashi Iwai 					    &spec->out_mix_path))
18533214b966STakashi Iwai 			dac = spec->out_mix_path.path[0];
18545b376195STakashi Iwai 		if (dac)
18555b376195STakashi Iwai 			spec->private_dac_nids[spec->multiout.num_dacs++] = dac;
18565c9a5615SLydia Wang 	}
18573214b966STakashi Iwai 	if (!spec->out_path[0].depth && spec->out_mix_path.depth) {
18583214b966STakashi Iwai 		spec->out_path[0] = spec->out_mix_path;
18593214b966STakashi Iwai 		spec->out_mix_path.depth = 0;
18603214b966STakashi Iwai 	}
1861c577b8a1SJoseph Chan 	return 0;
1862c577b8a1SJoseph Chan }
1863c577b8a1SJoseph Chan 
18644a79616dSTakashi Iwai static int create_ch_ctls(struct hda_codec *codec, const char *pfx,
186509a9ad69STakashi Iwai 			  int chs, bool check_dac, struct nid_path *path)
1866c577b8a1SJoseph Chan {
18674a79616dSTakashi Iwai 	struct via_spec *spec = codec->spec;
1868c577b8a1SJoseph Chan 	char name[32];
186909a9ad69STakashi Iwai 	hda_nid_t dac, pin, sel, nid;
187009a9ad69STakashi Iwai 	int err;
1871a934d5a9STakashi Iwai 
187209a9ad69STakashi Iwai 	dac = check_dac ? path->path[0] : 0;
187309a9ad69STakashi Iwai 	pin = path->path[path->depth - 1];
187409a9ad69STakashi Iwai 	sel = path->depth > 1 ? path->path[1] : 0;
1875c577b8a1SJoseph Chan 
18768df2a312STakashi Iwai 	if (dac && check_amp_caps(codec, dac, HDA_OUTPUT, AC_AMPCAP_NUM_STEPS))
18774a79616dSTakashi Iwai 		nid = dac;
18788df2a312STakashi Iwai 	else if (check_amp_caps(codec, pin, HDA_OUTPUT, AC_AMPCAP_NUM_STEPS))
18794a79616dSTakashi Iwai 		nid = pin;
1880a934d5a9STakashi Iwai 	else if (check_amp_caps(codec, sel, HDA_OUTPUT, AC_AMPCAP_NUM_STEPS))
1881a934d5a9STakashi Iwai 		nid = sel;
18824a79616dSTakashi Iwai 	else
18834a79616dSTakashi Iwai 		nid = 0;
18844a79616dSTakashi Iwai 	if (nid) {
18854a79616dSTakashi Iwai 		sprintf(name, "%s Playback Volume", pfx);
1886c577b8a1SJoseph Chan 		err = via_add_control(spec, VIA_CTL_WIDGET_VOL, name,
1887a00a5fadSLydia Wang 			      HDA_COMPOSE_AMP_VAL(nid, chs, 0, HDA_OUTPUT));
1888c577b8a1SJoseph Chan 		if (err < 0)
1889c577b8a1SJoseph Chan 			return err;
189009a9ad69STakashi Iwai 		path->vol_ctl = nid;
1891c577b8a1SJoseph Chan 	}
18924a79616dSTakashi Iwai 
18938df2a312STakashi Iwai 	if (dac && check_amp_caps(codec, dac, HDA_OUTPUT, AC_AMPCAP_MUTE))
18944a79616dSTakashi Iwai 		nid = dac;
18958df2a312STakashi Iwai 	else if (check_amp_caps(codec, pin, HDA_OUTPUT, AC_AMPCAP_MUTE))
18964a79616dSTakashi Iwai 		nid = pin;
1897a934d5a9STakashi Iwai 	else if (check_amp_caps(codec, sel, HDA_OUTPUT, AC_AMPCAP_MUTE))
1898a934d5a9STakashi Iwai 		nid = sel;
18994a79616dSTakashi Iwai 	else
19004a79616dSTakashi Iwai 		nid = 0;
19014a79616dSTakashi Iwai 	if (nid) {
19024a79616dSTakashi Iwai 		sprintf(name, "%s Playback Switch", pfx);
19034a79616dSTakashi Iwai 		err = via_add_control(spec, VIA_CTL_WIDGET_MUTE, name,
19044a79616dSTakashi Iwai 			      HDA_COMPOSE_AMP_VAL(nid, chs, 0, HDA_OUTPUT));
19054a79616dSTakashi Iwai 		if (err < 0)
19064a79616dSTakashi Iwai 			return err;
190709a9ad69STakashi Iwai 		path->mute_ctl = nid;
19084a79616dSTakashi Iwai 	}
19094a79616dSTakashi Iwai 	return 0;
19104a79616dSTakashi Iwai }
19114a79616dSTakashi Iwai 
1912f4a7828bSTakashi Iwai static void mangle_smart51(struct hda_codec *codec)
1913f4a7828bSTakashi Iwai {
1914f4a7828bSTakashi Iwai 	struct via_spec *spec = codec->spec;
1915f4a7828bSTakashi Iwai 	struct auto_pin_cfg *cfg = &spec->autocfg;
19160f98c24bSTakashi Iwai 	struct auto_pin_cfg_item *ins = cfg->inputs;
19170f98c24bSTakashi Iwai 	int i, j, nums, attr;
19180f98c24bSTakashi Iwai 	int pins[AUTO_CFG_MAX_INS];
1919f4a7828bSTakashi Iwai 
19200f98c24bSTakashi Iwai 	for (attr = INPUT_PIN_ATTR_REAR; attr >= INPUT_PIN_ATTR_NORMAL; attr--) {
19210f98c24bSTakashi Iwai 		nums = 0;
1922f4a7828bSTakashi Iwai 		for (i = 0; i < cfg->num_inputs; i++) {
19230f98c24bSTakashi Iwai 			unsigned int def;
19240f98c24bSTakashi Iwai 			if (ins[i].type > AUTO_PIN_LINE_IN)
19250f98c24bSTakashi Iwai 				continue;
19260f98c24bSTakashi Iwai 			def = snd_hda_codec_get_pincfg(codec, ins[i].pin);
19270f98c24bSTakashi Iwai 			if (snd_hda_get_input_pin_attr(def) != attr)
19280f98c24bSTakashi Iwai 				continue;
19290f98c24bSTakashi Iwai 			for (j = 0; j < nums; j++)
19300f98c24bSTakashi Iwai 				if (ins[pins[j]].type < ins[i].type) {
19310f98c24bSTakashi Iwai 					memmove(pins + j + 1, pins + j,
193221d45d2bSTakashi Iwai 						(nums - j) * sizeof(int));
19330f98c24bSTakashi Iwai 					break;
19340f98c24bSTakashi Iwai 				}
19350f98c24bSTakashi Iwai 			pins[j] = i;
1936e3d7a143STakashi Iwai 			nums++;
1937e3d7a143STakashi Iwai 		}
1938e3d7a143STakashi Iwai 		if (cfg->line_outs + nums < 3)
1939f4a7828bSTakashi Iwai 			continue;
19400f98c24bSTakashi Iwai 		for (i = 0; i < nums; i++) {
19410f98c24bSTakashi Iwai 			hda_nid_t pin = ins[pins[i]].pin;
19420f98c24bSTakashi Iwai 			spec->smart51_pins[spec->smart51_nums++] = pin;
19430f98c24bSTakashi Iwai 			cfg->line_out_pins[cfg->line_outs++] = pin;
1944f4a7828bSTakashi Iwai 			if (cfg->line_outs == 3)
1945f4a7828bSTakashi Iwai 				break;
1946f4a7828bSTakashi Iwai 		}
19470f98c24bSTakashi Iwai 		return;
19480f98c24bSTakashi Iwai 	}
1949f4a7828bSTakashi Iwai }
1950f4a7828bSTakashi Iwai 
1951020066d1STakashi Iwai static void copy_path_mixer_ctls(struct nid_path *dst, struct nid_path *src)
1952020066d1STakashi Iwai {
1953020066d1STakashi Iwai 	dst->vol_ctl = src->vol_ctl;
1954020066d1STakashi Iwai 	dst->mute_ctl = src->mute_ctl;
1955020066d1STakashi Iwai }
1956020066d1STakashi Iwai 
19574a79616dSTakashi Iwai /* add playback controls from the parsed DAC table */
19584a79616dSTakashi Iwai static int via_auto_create_multi_out_ctls(struct hda_codec *codec)
19594a79616dSTakashi Iwai {
19604a79616dSTakashi Iwai 	struct via_spec *spec = codec->spec;
1961f4a7828bSTakashi Iwai 	struct auto_pin_cfg *cfg = &spec->autocfg;
19623214b966STakashi Iwai 	struct nid_path *path;
19634a79616dSTakashi Iwai 	static const char * const chname[4] = {
196434ca8d33SDavid Henningsson 		"Front", "Surround", NULL /* "CLFE" */, "Side"
19654a79616dSTakashi Iwai 	};
19664a79616dSTakashi Iwai 	int i, idx, err;
1967f4a7828bSTakashi Iwai 	int old_line_outs;
1968f4a7828bSTakashi Iwai 
1969f4a7828bSTakashi Iwai 	/* check smart51 */
1970f4a7828bSTakashi Iwai 	old_line_outs = cfg->line_outs;
1971f4a7828bSTakashi Iwai 	if (cfg->line_outs == 1)
1972f4a7828bSTakashi Iwai 		mangle_smart51(codec);
19734a79616dSTakashi Iwai 
1974e3d7a143STakashi Iwai 	err = via_auto_fill_dac_nids(codec);
1975e3d7a143STakashi Iwai 	if (err < 0)
1976e3d7a143STakashi Iwai 		return err;
1977e3d7a143STakashi Iwai 
19785c9a5615SLydia Wang 	if (spec->multiout.num_dacs < 3) {
19795c9a5615SLydia Wang 		spec->smart51_nums = 0;
19805c9a5615SLydia Wang 		cfg->line_outs = old_line_outs;
19815c9a5615SLydia Wang 	}
19824a79616dSTakashi Iwai 	for (i = 0; i < cfg->line_outs; i++) {
19834a79616dSTakashi Iwai 		hda_nid_t pin, dac;
19844a79616dSTakashi Iwai 		pin = cfg->line_out_pins[i];
19854a79616dSTakashi Iwai 		dac = spec->multiout.dac_nids[i];
19864a79616dSTakashi Iwai 		if (!pin || !dac)
19874a79616dSTakashi Iwai 			continue;
19883214b966STakashi Iwai 		path = spec->out_path + i;
19890fe0adf8STakashi Iwai 		if (i == HDA_CLFE) {
19903214b966STakashi Iwai 			err = create_ch_ctls(codec, "Center", 1, true, path);
19914a79616dSTakashi Iwai 			if (err < 0)
19924a79616dSTakashi Iwai 				return err;
19933214b966STakashi Iwai 			err = create_ch_ctls(codec, "LFE", 2, true, path);
19944a79616dSTakashi Iwai 			if (err < 0)
19954a79616dSTakashi Iwai 				return err;
19964a79616dSTakashi Iwai 		} else {
19976aadf41dSTakashi Iwai 			const char *pfx = chname[i];
19986aadf41dSTakashi Iwai 			if (cfg->line_out_type == AUTO_PIN_SPEAKER_OUT &&
1999f37bc7a8STakashi Iwai 			    cfg->line_outs <= 2)
2000f37bc7a8STakashi Iwai 				pfx = i ? "Bass Speaker" : "Speaker";
20013214b966STakashi Iwai 			err = create_ch_ctls(codec, pfx, 3, true, path);
20024a79616dSTakashi Iwai 			if (err < 0)
20034a79616dSTakashi Iwai 				return err;
20044a79616dSTakashi Iwai 		}
2005020066d1STakashi Iwai 		if (path != spec->out_path + i)
2006020066d1STakashi Iwai 			copy_path_mixer_ctls(&spec->out_path[i], path);
2007020066d1STakashi Iwai 		if (path == spec->out_path && spec->out_mix_path.depth)
2008020066d1STakashi Iwai 			copy_path_mixer_ctls(&spec->out_mix_path, path);
20094a79616dSTakashi Iwai 	}
20104a79616dSTakashi Iwai 
20114a79616dSTakashi Iwai 	idx = get_connection_index(codec, spec->aa_mix_nid,
20124a79616dSTakashi Iwai 				   spec->multiout.dac_nids[0]);
20134a79616dSTakashi Iwai 	if (idx >= 0) {
20144a79616dSTakashi Iwai 		/* add control to mixer */
20153214b966STakashi Iwai 		const char *name;
20163214b966STakashi Iwai 		name = spec->out_mix_path.depth ?
20173214b966STakashi Iwai 			"PCM Loopback Playback Volume" : "PCM Playback Volume";
20183214b966STakashi Iwai 		err = via_add_control(spec, VIA_CTL_WIDGET_VOL, name,
20194a79616dSTakashi Iwai 				      HDA_COMPOSE_AMP_VAL(spec->aa_mix_nid, 3,
20204a79616dSTakashi Iwai 							  idx, HDA_INPUT));
20214a79616dSTakashi Iwai 		if (err < 0)
20224a79616dSTakashi Iwai 			return err;
20233214b966STakashi Iwai 		name = spec->out_mix_path.depth ?
20243214b966STakashi Iwai 			"PCM Loopback Playback Switch" : "PCM Playback Switch";
20253214b966STakashi Iwai 		err = via_add_control(spec, VIA_CTL_WIDGET_MUTE, name,
20264a79616dSTakashi Iwai 				      HDA_COMPOSE_AMP_VAL(spec->aa_mix_nid, 3,
20274a79616dSTakashi Iwai 							  idx, HDA_INPUT));
20284a79616dSTakashi Iwai 		if (err < 0)
20294a79616dSTakashi Iwai 			return err;
2030c577b8a1SJoseph Chan 	}
2031c577b8a1SJoseph Chan 
2032f4a7828bSTakashi Iwai 	cfg->line_outs = old_line_outs;
2033f4a7828bSTakashi Iwai 
2034c577b8a1SJoseph Chan 	return 0;
2035c577b8a1SJoseph Chan }
2036c577b8a1SJoseph Chan 
20374a79616dSTakashi Iwai static int via_auto_create_hp_ctls(struct hda_codec *codec, hda_nid_t pin)
2038c577b8a1SJoseph Chan {
20394a79616dSTakashi Iwai 	struct via_spec *spec = codec->spec;
204009a9ad69STakashi Iwai 	struct nid_path *path;
204118bd2c44STakashi Iwai 	bool check_dac;
20423214b966STakashi Iwai 	int i, err;
2043c577b8a1SJoseph Chan 
2044c577b8a1SJoseph Chan 	if (!pin)
2045c577b8a1SJoseph Chan 		return 0;
2046c577b8a1SJoseph Chan 
20473214b966STakashi Iwai 	if (!parse_output_path(codec, pin, 0, 0, &spec->hp_indep_path)) {
20483214b966STakashi Iwai 		for (i = HDA_SIDE; i >= HDA_CLFE; i--) {
20493214b966STakashi Iwai 			if (i < spec->multiout.num_dacs &&
205025250505STakashi Iwai 			    parse_output_path(codec, pin,
20513214b966STakashi Iwai 					      spec->multiout.dac_nids[i], 0,
20523214b966STakashi Iwai 					      &spec->hp_indep_path)) {
20533214b966STakashi Iwai 				spec->hp_indep_shared = i;
20543214b966STakashi Iwai 				break;
205525250505STakashi Iwai 			}
20563214b966STakashi Iwai 		}
20573214b966STakashi Iwai 	}
20583214b966STakashi Iwai 	if (spec->hp_indep_path.depth) {
20593214b966STakashi Iwai 		spec->hp_dac_nid = spec->hp_indep_path.path[0];
20603214b966STakashi Iwai 		if (!spec->hp_indep_shared)
20613214b966STakashi Iwai 			spec->hp_path = spec->hp_indep_path;
20623214b966STakashi Iwai 	}
20633214b966STakashi Iwai 	/* optionally check front-path w/o AA-mix */
20643214b966STakashi Iwai 	if (!spec->hp_path.depth)
20653214b966STakashi Iwai 		parse_output_path(codec, pin,
20663214b966STakashi Iwai 				  spec->multiout.dac_nids[HDA_FRONT], 0,
20673214b966STakashi Iwai 				  &spec->hp_path);
20684a79616dSTakashi Iwai 
2069ece8d043STakashi Iwai 	if (!parse_output_path(codec, pin, spec->multiout.dac_nids[HDA_FRONT],
20703214b966STakashi Iwai 			       1, &spec->hp_mix_path) && !spec->hp_path.depth)
2071ece8d043STakashi Iwai 		return 0;
2072ece8d043STakashi Iwai 
20733214b966STakashi Iwai 	if (spec->hp_path.depth) {
207409a9ad69STakashi Iwai 		path = &spec->hp_path;
207518bd2c44STakashi Iwai 		check_dac = true;
207618bd2c44STakashi Iwai 	} else {
20773214b966STakashi Iwai 		path = &spec->hp_mix_path;
207818bd2c44STakashi Iwai 		check_dac = false;
207918bd2c44STakashi Iwai 	}
208018bd2c44STakashi Iwai 	err = create_ch_ctls(codec, "Headphone", 3, check_dac, path);
20814a79616dSTakashi Iwai 	if (err < 0)
20824a79616dSTakashi Iwai 		return err;
2083020066d1STakashi Iwai 	if (check_dac)
2084020066d1STakashi Iwai 		copy_path_mixer_ctls(&spec->hp_mix_path, path);
2085020066d1STakashi Iwai 	else
2086020066d1STakashi Iwai 		copy_path_mixer_ctls(&spec->hp_path, path);
2087020066d1STakashi Iwai 	if (spec->hp_indep_path.depth)
2088020066d1STakashi Iwai 		copy_path_mixer_ctls(&spec->hp_indep_path, path);
2089c577b8a1SJoseph Chan 	return 0;
2090c577b8a1SJoseph Chan }
2091c577b8a1SJoseph Chan 
20924a918ffeSTakashi Iwai static int via_auto_create_speaker_ctls(struct hda_codec *codec)
20934a918ffeSTakashi Iwai {
20944a918ffeSTakashi Iwai 	struct via_spec *spec = codec->spec;
20953214b966STakashi Iwai 	struct nid_path *path;
20963214b966STakashi Iwai 	bool check_dac;
209781c0a78bSWang Shaoyan 	hda_nid_t pin, dac = 0;
20983214b966STakashi Iwai 	int err;
20994a918ffeSTakashi Iwai 
21004a918ffeSTakashi Iwai 	pin = spec->autocfg.speaker_pins[0];
21014a918ffeSTakashi Iwai 	if (!spec->autocfg.speaker_outs || !pin)
21024a918ffeSTakashi Iwai 		return 0;
21034a918ffeSTakashi Iwai 
21043214b966STakashi Iwai 	if (parse_output_path(codec, pin, 0, 0, &spec->speaker_path))
21058e3679dcSTakashi Iwai 		dac = spec->speaker_path.path[0];
21063214b966STakashi Iwai 	if (!dac)
21073214b966STakashi Iwai 		parse_output_path(codec, pin,
21083214b966STakashi Iwai 				  spec->multiout.dac_nids[HDA_FRONT], 0,
210909a9ad69STakashi Iwai 				  &spec->speaker_path);
21103214b966STakashi Iwai 	if (!parse_output_path(codec, pin, spec->multiout.dac_nids[HDA_FRONT],
21113214b966STakashi Iwai 			       1, &spec->speaker_mix_path) && !dac)
21123214b966STakashi Iwai 		return 0;
21134a918ffeSTakashi Iwai 
21143214b966STakashi Iwai 	/* no AA-path for front? */
21153214b966STakashi Iwai 	if (!spec->out_mix_path.depth && spec->speaker_mix_path.depth)
21163214b966STakashi Iwai 		dac = 0;
21173214b966STakashi Iwai 
21183214b966STakashi Iwai 	spec->speaker_dac_nid = dac;
21193214b966STakashi Iwai 	spec->multiout.extra_out_nid[0] = dac;
21203214b966STakashi Iwai 	if (dac) {
21213214b966STakashi Iwai 		path = &spec->speaker_path;
21223214b966STakashi Iwai 		check_dac = true;
21233214b966STakashi Iwai 	} else {
21243214b966STakashi Iwai 		path = &spec->speaker_mix_path;
21253214b966STakashi Iwai 		check_dac = false;
21263214b966STakashi Iwai 	}
21273214b966STakashi Iwai 	err = create_ch_ctls(codec, "Speaker", 3, check_dac, path);
21283214b966STakashi Iwai 	if (err < 0)
21293214b966STakashi Iwai 		return err;
2130020066d1STakashi Iwai 	if (check_dac)
2131020066d1STakashi Iwai 		copy_path_mixer_ctls(&spec->speaker_mix_path, path);
2132020066d1STakashi Iwai 	else
2133020066d1STakashi Iwai 		copy_path_mixer_ctls(&spec->speaker_path, path);
21343214b966STakashi Iwai 	return 0;
21353214b966STakashi Iwai }
21363214b966STakashi Iwai 
21373214b966STakashi Iwai #define via_aamix_ctl_info	via_pin_power_ctl_info
21383214b966STakashi Iwai 
21393214b966STakashi Iwai static int via_aamix_ctl_get(struct snd_kcontrol *kcontrol,
21403214b966STakashi Iwai 			     struct snd_ctl_elem_value *ucontrol)
21413214b966STakashi Iwai {
21423214b966STakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
21433214b966STakashi Iwai 	struct via_spec *spec = codec->spec;
21443214b966STakashi Iwai 	ucontrol->value.enumerated.item[0] = spec->aamix_mode;
21453214b966STakashi Iwai 	return 0;
21463214b966STakashi Iwai }
21473214b966STakashi Iwai 
21483214b966STakashi Iwai static void update_aamix_paths(struct hda_codec *codec, int do_mix,
21493214b966STakashi Iwai 			       struct nid_path *nomix, struct nid_path *mix)
21503214b966STakashi Iwai {
21513214b966STakashi Iwai 	if (do_mix) {
21523214b966STakashi Iwai 		activate_output_path(codec, nomix, false, false);
21533214b966STakashi Iwai 		activate_output_path(codec, mix, true, false);
21543214b966STakashi Iwai 	} else {
21553214b966STakashi Iwai 		activate_output_path(codec, mix, false, false);
21563214b966STakashi Iwai 		activate_output_path(codec, nomix, true, false);
21573214b966STakashi Iwai 	}
21583214b966STakashi Iwai }
21593214b966STakashi Iwai 
21603214b966STakashi Iwai static int via_aamix_ctl_put(struct snd_kcontrol *kcontrol,
21613214b966STakashi Iwai 			     struct snd_ctl_elem_value *ucontrol)
21623214b966STakashi Iwai {
21633214b966STakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
21643214b966STakashi Iwai 	struct via_spec *spec = codec->spec;
21653214b966STakashi Iwai 	unsigned int val = ucontrol->value.enumerated.item[0];
21663214b966STakashi Iwai 
21673214b966STakashi Iwai 	if (val == spec->aamix_mode)
21683214b966STakashi Iwai 		return 0;
21693214b966STakashi Iwai 	spec->aamix_mode = val;
21703214b966STakashi Iwai 	/* update front path */
21713214b966STakashi Iwai 	update_aamix_paths(codec, val, &spec->out_path[0], &spec->out_mix_path);
21723214b966STakashi Iwai 	/* update HP path */
21733214b966STakashi Iwai 	if (!spec->hp_independent_mode) {
21743214b966STakashi Iwai 		update_aamix_paths(codec, val, &spec->hp_path,
21753214b966STakashi Iwai 				   &spec->hp_mix_path);
21763214b966STakashi Iwai 	}
21773214b966STakashi Iwai 	/* update speaker path */
21783214b966STakashi Iwai 	update_aamix_paths(codec, val, &spec->speaker_path,
21793214b966STakashi Iwai 			   &spec->speaker_mix_path);
21803214b966STakashi Iwai 	return 1;
21813214b966STakashi Iwai }
21823214b966STakashi Iwai 
21833214b966STakashi Iwai static const struct snd_kcontrol_new via_aamix_ctl_enum = {
21843214b966STakashi Iwai 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
21853214b966STakashi Iwai 	.name = "Loopback Mixing",
21863214b966STakashi Iwai 	.info = via_aamix_ctl_info,
21873214b966STakashi Iwai 	.get = via_aamix_ctl_get,
21883214b966STakashi Iwai 	.put = via_aamix_ctl_put,
21893214b966STakashi Iwai };
21903214b966STakashi Iwai 
21913214b966STakashi Iwai static int via_auto_create_loopback_switch(struct hda_codec *codec)
21923214b966STakashi Iwai {
21933214b966STakashi Iwai 	struct via_spec *spec = codec->spec;
21943214b966STakashi Iwai 
21954808d12dSTakashi Iwai 	if (!spec->aa_mix_nid)
21964808d12dSTakashi Iwai 		return 0; /* no loopback switching available */
21974808d12dSTakashi Iwai 	if (!(spec->out_mix_path.depth || spec->hp_mix_path.depth ||
21984808d12dSTakashi Iwai 	      spec->speaker_path.depth))
21993214b966STakashi Iwai 		return 0; /* no loopback switching available */
22003214b966STakashi Iwai 	if (!via_clone_control(spec, &via_aamix_ctl_enum))
22013214b966STakashi Iwai 		return -ENOMEM;
22024a918ffeSTakashi Iwai 	return 0;
22034a918ffeSTakashi Iwai }
22044a918ffeSTakashi Iwai 
2205a766d0d7STakashi Iwai /* look for ADCs */
2206a766d0d7STakashi Iwai static int via_fill_adcs(struct hda_codec *codec)
2207a766d0d7STakashi Iwai {
2208a766d0d7STakashi Iwai 	struct via_spec *spec = codec->spec;
2209a766d0d7STakashi Iwai 	hda_nid_t nid = codec->start_nid;
2210a766d0d7STakashi Iwai 	int i;
2211a766d0d7STakashi Iwai 
2212a766d0d7STakashi Iwai 	for (i = 0; i < codec->num_nodes; i++, nid++) {
2213a766d0d7STakashi Iwai 		unsigned int wcaps = get_wcaps(codec, nid);
2214a766d0d7STakashi Iwai 		if (get_wcaps_type(wcaps) != AC_WID_AUD_IN)
2215a766d0d7STakashi Iwai 			continue;
2216a766d0d7STakashi Iwai 		if (wcaps & AC_WCAP_DIGITAL)
2217a766d0d7STakashi Iwai 			continue;
2218a766d0d7STakashi Iwai 		if (!(wcaps & AC_WCAP_CONN_LIST))
2219a766d0d7STakashi Iwai 			continue;
2220a766d0d7STakashi Iwai 		if (spec->num_adc_nids >= ARRAY_SIZE(spec->adc_nids))
2221a766d0d7STakashi Iwai 			return -ENOMEM;
2222a766d0d7STakashi Iwai 		spec->adc_nids[spec->num_adc_nids++] = nid;
2223a766d0d7STakashi Iwai 	}
2224a766d0d7STakashi Iwai 	return 0;
2225a766d0d7STakashi Iwai }
2226a766d0d7STakashi Iwai 
2227a86a88eaSTakashi Iwai /* input-src control */
2228a86a88eaSTakashi Iwai static int via_mux_enum_info(struct snd_kcontrol *kcontrol,
2229a86a88eaSTakashi Iwai 			     struct snd_ctl_elem_info *uinfo)
2230a86a88eaSTakashi Iwai {
2231a86a88eaSTakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2232a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2233a86a88eaSTakashi Iwai 
2234a86a88eaSTakashi Iwai 	uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
2235a86a88eaSTakashi Iwai 	uinfo->count = 1;
2236a86a88eaSTakashi Iwai 	uinfo->value.enumerated.items = spec->num_inputs;
2237a86a88eaSTakashi Iwai 	if (uinfo->value.enumerated.item >= spec->num_inputs)
2238a86a88eaSTakashi Iwai 		uinfo->value.enumerated.item = spec->num_inputs - 1;
2239a86a88eaSTakashi Iwai 	strcpy(uinfo->value.enumerated.name,
2240a86a88eaSTakashi Iwai 	       spec->inputs[uinfo->value.enumerated.item].label);
2241a86a88eaSTakashi Iwai 	return 0;
2242a86a88eaSTakashi Iwai }
2243a86a88eaSTakashi Iwai 
2244a86a88eaSTakashi Iwai static int via_mux_enum_get(struct snd_kcontrol *kcontrol,
2245a86a88eaSTakashi Iwai 			    struct snd_ctl_elem_value *ucontrol)
2246a86a88eaSTakashi Iwai {
2247a86a88eaSTakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2248a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2249a86a88eaSTakashi Iwai 	unsigned int idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
2250a86a88eaSTakashi Iwai 
2251a86a88eaSTakashi Iwai 	ucontrol->value.enumerated.item[0] = spec->cur_mux[idx];
2252a86a88eaSTakashi Iwai 	return 0;
2253a86a88eaSTakashi Iwai }
2254a86a88eaSTakashi Iwai 
2255a86a88eaSTakashi Iwai static int via_mux_enum_put(struct snd_kcontrol *kcontrol,
2256a86a88eaSTakashi Iwai 			    struct snd_ctl_elem_value *ucontrol)
2257a86a88eaSTakashi Iwai {
2258a86a88eaSTakashi Iwai 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2259a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2260a86a88eaSTakashi Iwai 	unsigned int idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
2261a86a88eaSTakashi Iwai 	hda_nid_t mux;
2262a86a88eaSTakashi Iwai 	int cur;
2263a86a88eaSTakashi Iwai 
2264a86a88eaSTakashi Iwai 	cur = ucontrol->value.enumerated.item[0];
2265a86a88eaSTakashi Iwai 	if (cur < 0 || cur >= spec->num_inputs)
2266a86a88eaSTakashi Iwai 		return -EINVAL;
2267a86a88eaSTakashi Iwai 	if (spec->cur_mux[idx] == cur)
2268a86a88eaSTakashi Iwai 		return 0;
2269a86a88eaSTakashi Iwai 	spec->cur_mux[idx] = cur;
2270a86a88eaSTakashi Iwai 	if (spec->dyn_adc_switch) {
2271a86a88eaSTakashi Iwai 		int adc_idx = spec->inputs[cur].adc_idx;
2272a86a88eaSTakashi Iwai 		mux = spec->mux_nids[adc_idx];
2273a86a88eaSTakashi Iwai 		via_dyn_adc_pcm_resetup(codec, cur);
2274a86a88eaSTakashi Iwai 	} else {
2275a86a88eaSTakashi Iwai 		mux = spec->mux_nids[idx];
2276a86a88eaSTakashi Iwai 		if (snd_BUG_ON(!mux))
2277a86a88eaSTakashi Iwai 			return -EINVAL;
2278a86a88eaSTakashi Iwai 	}
2279a86a88eaSTakashi Iwai 
2280a86a88eaSTakashi Iwai 	if (mux) {
2281a86a88eaSTakashi Iwai 		/* switch to D0 beofre change index */
2282054d867eSTakashi Iwai 		update_power_state(codec, mux, AC_PWRST_D0);
2283a86a88eaSTakashi Iwai 		snd_hda_codec_write(codec, mux, 0,
2284a86a88eaSTakashi Iwai 				    AC_VERB_SET_CONNECT_SEL,
2285a86a88eaSTakashi Iwai 				    spec->inputs[cur].mux_idx);
2286a86a88eaSTakashi Iwai 	}
2287a86a88eaSTakashi Iwai 
2288a86a88eaSTakashi Iwai 	/* update jack power state */
2289a86a88eaSTakashi Iwai 	set_widgets_power_state(codec);
2290a86a88eaSTakashi Iwai 	return 0;
2291a86a88eaSTakashi Iwai }
2292a766d0d7STakashi Iwai 
2293d7a99cceSTakashi Iwai static const struct snd_kcontrol_new via_input_src_ctl = {
2294d7a99cceSTakashi Iwai 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2295d7a99cceSTakashi Iwai 	/* The multiple "Capture Source" controls confuse alsamixer
2296d7a99cceSTakashi Iwai 	 * So call somewhat different..
2297d7a99cceSTakashi Iwai 	 */
2298d7a99cceSTakashi Iwai 	/* .name = "Capture Source", */
2299d7a99cceSTakashi Iwai 	.name = "Input Source",
2300d7a99cceSTakashi Iwai 	.info = via_mux_enum_info,
2301d7a99cceSTakashi Iwai 	.get = via_mux_enum_get,
2302d7a99cceSTakashi Iwai 	.put = via_mux_enum_put,
2303d7a99cceSTakashi Iwai };
2304d7a99cceSTakashi Iwai 
2305a86a88eaSTakashi Iwai static int create_input_src_ctls(struct hda_codec *codec, int count)
2306a86a88eaSTakashi Iwai {
2307a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2308a86a88eaSTakashi Iwai 	struct snd_kcontrol_new *knew;
2309a86a88eaSTakashi Iwai 
2310a86a88eaSTakashi Iwai 	if (spec->num_inputs <= 1 || !count)
2311a86a88eaSTakashi Iwai 		return 0; /* no need for single src */
2312a86a88eaSTakashi Iwai 
2313a86a88eaSTakashi Iwai 	knew = via_clone_control(spec, &via_input_src_ctl);
2314a86a88eaSTakashi Iwai 	if (!knew)
2315a86a88eaSTakashi Iwai 		return -ENOMEM;
2316a86a88eaSTakashi Iwai 	knew->count = count;
2317a86a88eaSTakashi Iwai 	return 0;
2318a86a88eaSTakashi Iwai }
2319a86a88eaSTakashi Iwai 
2320a86a88eaSTakashi Iwai /* add the powersave loopback-list entry */
232113af8e77STakashi Iwai static void add_loopback_list(struct via_spec *spec, hda_nid_t mix, int idx)
232213af8e77STakashi Iwai {
232313af8e77STakashi Iwai 	struct hda_amp_list *list;
232413af8e77STakashi Iwai 
232513af8e77STakashi Iwai 	if (spec->num_loopbacks >= ARRAY_SIZE(spec->loopback_list) - 1)
232613af8e77STakashi Iwai 		return;
232713af8e77STakashi Iwai 	list = spec->loopback_list + spec->num_loopbacks;
232813af8e77STakashi Iwai 	list->nid = mix;
232913af8e77STakashi Iwai 	list->dir = HDA_INPUT;
233013af8e77STakashi Iwai 	list->idx = idx;
233113af8e77STakashi Iwai 	spec->num_loopbacks++;
233213af8e77STakashi Iwai 	spec->loopback.amplist = spec->loopback_list;
233313af8e77STakashi Iwai }
233413af8e77STakashi Iwai 
2335a86a88eaSTakashi Iwai static bool is_reachable_nid(struct hda_codec *codec, hda_nid_t src,
23368d087c76STakashi Iwai 			     hda_nid_t dst)
2337a86a88eaSTakashi Iwai {
23388d087c76STakashi Iwai 	return snd_hda_get_conn_index(codec, src, dst, 1) >= 0;
2339a86a88eaSTakashi Iwai }
2340a86a88eaSTakashi Iwai 
2341a86a88eaSTakashi Iwai /* add the input-route to the given pin */
2342a86a88eaSTakashi Iwai static bool add_input_route(struct hda_codec *codec, hda_nid_t pin)
2343c577b8a1SJoseph Chan {
234410a20af7STakashi Iwai 	struct via_spec *spec = codec->spec;
2345a86a88eaSTakashi Iwai 	int c, idx;
2346a86a88eaSTakashi Iwai 
2347a86a88eaSTakashi Iwai 	spec->inputs[spec->num_inputs].adc_idx = -1;
2348a86a88eaSTakashi Iwai 	spec->inputs[spec->num_inputs].pin = pin;
2349a86a88eaSTakashi Iwai 	for (c = 0; c < spec->num_adc_nids; c++) {
2350a86a88eaSTakashi Iwai 		if (spec->mux_nids[c]) {
2351a86a88eaSTakashi Iwai 			idx = get_connection_index(codec, spec->mux_nids[c],
2352a86a88eaSTakashi Iwai 						   pin);
2353a86a88eaSTakashi Iwai 			if (idx < 0)
2354a86a88eaSTakashi Iwai 				continue;
2355a86a88eaSTakashi Iwai 			spec->inputs[spec->num_inputs].mux_idx = idx;
2356a86a88eaSTakashi Iwai 		} else {
23578d087c76STakashi Iwai 			if (!is_reachable_nid(codec, spec->adc_nids[c], pin))
2358a86a88eaSTakashi Iwai 				continue;
2359a86a88eaSTakashi Iwai 		}
2360a86a88eaSTakashi Iwai 		spec->inputs[spec->num_inputs].adc_idx = c;
2361a86a88eaSTakashi Iwai 		/* Can primary ADC satisfy all inputs? */
2362a86a88eaSTakashi Iwai 		if (!spec->dyn_adc_switch &&
2363a86a88eaSTakashi Iwai 		    spec->num_inputs > 0 && spec->inputs[0].adc_idx != c) {
2364a86a88eaSTakashi Iwai 			snd_printd(KERN_INFO
2365a86a88eaSTakashi Iwai 				   "via: dynamic ADC switching enabled\n");
2366a86a88eaSTakashi Iwai 			spec->dyn_adc_switch = 1;
2367a86a88eaSTakashi Iwai 		}
2368a86a88eaSTakashi Iwai 		return true;
2369a86a88eaSTakashi Iwai 	}
2370a86a88eaSTakashi Iwai 	return false;
2371a86a88eaSTakashi Iwai }
2372a86a88eaSTakashi Iwai 
2373a86a88eaSTakashi Iwai static int get_mux_nids(struct hda_codec *codec);
2374a86a88eaSTakashi Iwai 
2375a86a88eaSTakashi Iwai /* parse input-routes; fill ADCs, MUXs and input-src entries */
2376a86a88eaSTakashi Iwai static int parse_analog_inputs(struct hda_codec *codec)
2377a86a88eaSTakashi Iwai {
2378a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2379a86a88eaSTakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
2380a86a88eaSTakashi Iwai 	int i, err;
2381a766d0d7STakashi Iwai 
2382a766d0d7STakashi Iwai 	err = via_fill_adcs(codec);
2383a766d0d7STakashi Iwai 	if (err < 0)
2384a766d0d7STakashi Iwai 		return err;
2385a766d0d7STakashi Iwai 	err = get_mux_nids(codec);
2386a766d0d7STakashi Iwai 	if (err < 0)
2387a766d0d7STakashi Iwai 		return err;
2388a766d0d7STakashi Iwai 
2389a86a88eaSTakashi Iwai 	/* fill all input-routes */
2390a86a88eaSTakashi Iwai 	for (i = 0; i < cfg->num_inputs; i++) {
2391a86a88eaSTakashi Iwai 		if (add_input_route(codec, cfg->inputs[i].pin))
2392a86a88eaSTakashi Iwai 			spec->inputs[spec->num_inputs++].label =
2393a86a88eaSTakashi Iwai 				hda_get_autocfg_input_label(codec, cfg, i);
2394a86a88eaSTakashi Iwai 	}
2395a86a88eaSTakashi Iwai 
2396a86a88eaSTakashi Iwai 	/* check for internal loopback recording */
2397a86a88eaSTakashi Iwai 	if (spec->aa_mix_nid &&
2398a86a88eaSTakashi Iwai 	    add_input_route(codec, spec->aa_mix_nid))
2399a86a88eaSTakashi Iwai 		spec->inputs[spec->num_inputs++].label = "Stereo Mixer";
2400a86a88eaSTakashi Iwai 
2401a86a88eaSTakashi Iwai 	return 0;
2402a86a88eaSTakashi Iwai }
2403a86a88eaSTakashi Iwai 
2404a86a88eaSTakashi Iwai /* create analog-loopback volume/switch controls */
2405a86a88eaSTakashi Iwai static int create_loopback_ctls(struct hda_codec *codec)
2406a86a88eaSTakashi Iwai {
2407a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2408a86a88eaSTakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
2409a86a88eaSTakashi Iwai 	const char *prev_label = NULL;
2410a86a88eaSTakashi Iwai 	int type_idx = 0;
2411a86a88eaSTakashi Iwai 	int i, j, err, idx;
2412a86a88eaSTakashi Iwai 
2413a86a88eaSTakashi Iwai 	if (!spec->aa_mix_nid)
2414a766d0d7STakashi Iwai 		return 0;
2415c577b8a1SJoseph Chan 
24167b315bb4STakashi Iwai 	for (i = 0; i < cfg->num_inputs; i++) {
2417a86a88eaSTakashi Iwai 		hda_nid_t pin = cfg->inputs[i].pin;
2418a86a88eaSTakashi Iwai 		const char *label = hda_get_autocfg_input_label(codec, cfg, i);
2419a86a88eaSTakashi Iwai 
24201e11cae1STakashi Iwai 		if (prev_label && !strcmp(label, prev_label))
24217b315bb4STakashi Iwai 			type_idx++;
24227b315bb4STakashi Iwai 		else
24237b315bb4STakashi Iwai 			type_idx = 0;
24241e11cae1STakashi Iwai 		prev_label = label;
2425a86a88eaSTakashi Iwai 		idx = get_connection_index(codec, spec->aa_mix_nid, pin);
2426a86a88eaSTakashi Iwai 		if (idx >= 0) {
242716922281SLydia Wang 			err = via_new_analog_input(spec, label, type_idx,
2428a86a88eaSTakashi Iwai 						   idx, spec->aa_mix_nid);
2429c577b8a1SJoseph Chan 			if (err < 0)
2430c577b8a1SJoseph Chan 				return err;
2431a86a88eaSTakashi Iwai 			add_loopback_list(spec, spec->aa_mix_nid, idx);
243213af8e77STakashi Iwai 		}
2433e3d7a143STakashi Iwai 
2434e3d7a143STakashi Iwai 		/* remember the label for smart51 control */
2435e3d7a143STakashi Iwai 		for (j = 0; j < spec->smart51_nums; j++) {
2436a86a88eaSTakashi Iwai 			if (spec->smart51_pins[j] == pin) {
2437e3d7a143STakashi Iwai 				spec->smart51_idxs[j] = idx;
2438e3d7a143STakashi Iwai 				spec->smart51_labels[j] = label;
2439e3d7a143STakashi Iwai 				break;
2440e3d7a143STakashi Iwai 			}
2441e3d7a143STakashi Iwai 		}
2442c577b8a1SJoseph Chan 	}
2443a86a88eaSTakashi Iwai 	return 0;
2444a86a88eaSTakashi Iwai }
2445a86a88eaSTakashi Iwai 
2446a86a88eaSTakashi Iwai /* create mic-boost controls (if present) */
2447a86a88eaSTakashi Iwai static int create_mic_boost_ctls(struct hda_codec *codec)
2448a86a88eaSTakashi Iwai {
2449a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2450a86a88eaSTakashi Iwai 	const struct auto_pin_cfg *cfg = &spec->autocfg;
24518d8bbc6fSTakashi Iwai 	const char *prev_label = NULL;
24528d8bbc6fSTakashi Iwai 	int type_idx = 0;
2453a86a88eaSTakashi Iwai 	int i, err;
2454a86a88eaSTakashi Iwai 
2455a86a88eaSTakashi Iwai 	for (i = 0; i < cfg->num_inputs; i++) {
2456a86a88eaSTakashi Iwai 		hda_nid_t pin = cfg->inputs[i].pin;
2457a86a88eaSTakashi Iwai 		unsigned int caps;
2458a86a88eaSTakashi Iwai 		const char *label;
2459a86a88eaSTakashi Iwai 		char name[32];
2460a86a88eaSTakashi Iwai 
2461a86a88eaSTakashi Iwai 		if (cfg->inputs[i].type != AUTO_PIN_MIC)
2462a86a88eaSTakashi Iwai 			continue;
2463a86a88eaSTakashi Iwai 		caps = query_amp_caps(codec, pin, HDA_INPUT);
2464a86a88eaSTakashi Iwai 		if (caps == -1 || !(caps & AC_AMPCAP_NUM_STEPS))
2465a86a88eaSTakashi Iwai 			continue;
2466a86a88eaSTakashi Iwai 		label = hda_get_autocfg_input_label(codec, cfg, i);
24678d8bbc6fSTakashi Iwai 		if (prev_label && !strcmp(label, prev_label))
24688d8bbc6fSTakashi Iwai 			type_idx++;
24698d8bbc6fSTakashi Iwai 		else
24708d8bbc6fSTakashi Iwai 			type_idx = 0;
24718d8bbc6fSTakashi Iwai 		prev_label = label;
2472a86a88eaSTakashi Iwai 		snprintf(name, sizeof(name), "%s Boost Volume", label);
24738d8bbc6fSTakashi Iwai 		err = __via_add_control(spec, VIA_CTL_WIDGET_VOL, name, type_idx,
2474a86a88eaSTakashi Iwai 			      HDA_COMPOSE_AMP_VAL(pin, 3, 0, HDA_INPUT));
2475a86a88eaSTakashi Iwai 		if (err < 0)
2476a86a88eaSTakashi Iwai 			return err;
2477a86a88eaSTakashi Iwai 	}
2478a86a88eaSTakashi Iwai 	return 0;
2479a86a88eaSTakashi Iwai }
2480a86a88eaSTakashi Iwai 
2481a86a88eaSTakashi Iwai /* create capture and input-src controls for multiple streams */
2482a86a88eaSTakashi Iwai static int create_multi_adc_ctls(struct hda_codec *codec)
2483a86a88eaSTakashi Iwai {
2484a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2485a86a88eaSTakashi Iwai 	int i, err;
2486d7a99cceSTakashi Iwai 
2487d7a99cceSTakashi Iwai 	/* create capture mixer elements */
2488d7a99cceSTakashi Iwai 	for (i = 0; i < spec->num_adc_nids; i++) {
2489d7a99cceSTakashi Iwai 		hda_nid_t adc = spec->adc_nids[i];
2490d7a99cceSTakashi Iwai 		err = __via_add_control(spec, VIA_CTL_WIDGET_VOL,
2491d7a99cceSTakashi Iwai 					"Capture Volume", i,
2492d7a99cceSTakashi Iwai 					HDA_COMPOSE_AMP_VAL(adc, 3, 0,
2493d7a99cceSTakashi Iwai 							    HDA_INPUT));
2494d7a99cceSTakashi Iwai 		if (err < 0)
2495d7a99cceSTakashi Iwai 			return err;
2496d7a99cceSTakashi Iwai 		err = __via_add_control(spec, VIA_CTL_WIDGET_MUTE,
2497d7a99cceSTakashi Iwai 					"Capture Switch", i,
2498d7a99cceSTakashi Iwai 					HDA_COMPOSE_AMP_VAL(adc, 3, 0,
2499d7a99cceSTakashi Iwai 							    HDA_INPUT));
2500d7a99cceSTakashi Iwai 		if (err < 0)
2501d7a99cceSTakashi Iwai 			return err;
2502d7a99cceSTakashi Iwai 	}
2503d7a99cceSTakashi Iwai 
2504d7a99cceSTakashi Iwai 	/* input-source control */
2505d7a99cceSTakashi Iwai 	for (i = 0; i < spec->num_adc_nids; i++)
2506d7a99cceSTakashi Iwai 		if (!spec->mux_nids[i])
2507d7a99cceSTakashi Iwai 			break;
2508a86a88eaSTakashi Iwai 	err = create_input_src_ctls(codec, i);
2509d7a99cceSTakashi Iwai 	if (err < 0)
2510d7a99cceSTakashi Iwai 		return err;
2511a86a88eaSTakashi Iwai 	return 0;
2512d7a99cceSTakashi Iwai }
2513d7a99cceSTakashi Iwai 
2514a86a88eaSTakashi Iwai /* bind capture volume/switch */
2515a86a88eaSTakashi Iwai static struct snd_kcontrol_new via_bind_cap_vol_ctl =
2516a86a88eaSTakashi Iwai 	HDA_BIND_VOL("Capture Volume", 0);
2517a86a88eaSTakashi Iwai static struct snd_kcontrol_new via_bind_cap_sw_ctl =
2518a86a88eaSTakashi Iwai 	HDA_BIND_SW("Capture Switch", 0);
2519a86a88eaSTakashi Iwai 
2520a86a88eaSTakashi Iwai static int init_bind_ctl(struct via_spec *spec, struct hda_bind_ctls **ctl_ret,
2521a86a88eaSTakashi Iwai 			 struct hda_ctl_ops *ops)
2522a86a88eaSTakashi Iwai {
2523a86a88eaSTakashi Iwai 	struct hda_bind_ctls *ctl;
2524a86a88eaSTakashi Iwai 	int i;
2525a86a88eaSTakashi Iwai 
2526a86a88eaSTakashi Iwai 	ctl = kzalloc(sizeof(*ctl) + sizeof(long) * 4, GFP_KERNEL);
2527a86a88eaSTakashi Iwai 	if (!ctl)
2528a86a88eaSTakashi Iwai 		return -ENOMEM;
2529a86a88eaSTakashi Iwai 	ctl->ops = ops;
2530a86a88eaSTakashi Iwai 	for (i = 0; i < spec->num_adc_nids; i++)
2531a86a88eaSTakashi Iwai 		ctl->values[i] =
2532a86a88eaSTakashi Iwai 			HDA_COMPOSE_AMP_VAL(spec->adc_nids[i], 3, 0, HDA_INPUT);
2533a86a88eaSTakashi Iwai 	*ctl_ret = ctl;
2534a86a88eaSTakashi Iwai 	return 0;
2535a86a88eaSTakashi Iwai }
2536a86a88eaSTakashi Iwai 
2537a86a88eaSTakashi Iwai /* create capture and input-src controls for dynamic ADC-switch case */
2538a86a88eaSTakashi Iwai static int create_dyn_adc_ctls(struct hda_codec *codec)
2539a86a88eaSTakashi Iwai {
2540a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2541a86a88eaSTakashi Iwai 	struct snd_kcontrol_new *knew;
2542a86a88eaSTakashi Iwai 	int err;
2543a86a88eaSTakashi Iwai 
2544a86a88eaSTakashi Iwai 	/* set up the bind capture ctls */
2545a86a88eaSTakashi Iwai 	err = init_bind_ctl(spec, &spec->bind_cap_vol, &snd_hda_bind_vol);
2546a86a88eaSTakashi Iwai 	if (err < 0)
2547a86a88eaSTakashi Iwai 		return err;
2548a86a88eaSTakashi Iwai 	err = init_bind_ctl(spec, &spec->bind_cap_sw, &snd_hda_bind_sw);
2549a86a88eaSTakashi Iwai 	if (err < 0)
2550a86a88eaSTakashi Iwai 		return err;
2551a86a88eaSTakashi Iwai 
2552a86a88eaSTakashi Iwai 	/* create capture mixer elements */
2553a86a88eaSTakashi Iwai 	knew = via_clone_control(spec, &via_bind_cap_vol_ctl);
2554a86a88eaSTakashi Iwai 	if (!knew)
2555a86a88eaSTakashi Iwai 		return -ENOMEM;
2556a86a88eaSTakashi Iwai 	knew->private_value = (long)spec->bind_cap_vol;
2557a86a88eaSTakashi Iwai 
2558a86a88eaSTakashi Iwai 	knew = via_clone_control(spec, &via_bind_cap_sw_ctl);
2559a86a88eaSTakashi Iwai 	if (!knew)
2560a86a88eaSTakashi Iwai 		return -ENOMEM;
2561a86a88eaSTakashi Iwai 	knew->private_value = (long)spec->bind_cap_sw;
2562a86a88eaSTakashi Iwai 
2563a86a88eaSTakashi Iwai 	/* input-source control */
2564a86a88eaSTakashi Iwai 	err = create_input_src_ctls(codec, 1);
2565a86a88eaSTakashi Iwai 	if (err < 0)
2566a86a88eaSTakashi Iwai 		return err;
2567a86a88eaSTakashi Iwai 	return 0;
2568a86a88eaSTakashi Iwai }
2569a86a88eaSTakashi Iwai 
2570a86a88eaSTakashi Iwai /* parse and create capture-related stuff */
2571a86a88eaSTakashi Iwai static int via_auto_create_analog_input_ctls(struct hda_codec *codec)
2572a86a88eaSTakashi Iwai {
2573a86a88eaSTakashi Iwai 	struct via_spec *spec = codec->spec;
2574a86a88eaSTakashi Iwai 	int err;
2575a86a88eaSTakashi Iwai 
2576a86a88eaSTakashi Iwai 	err = parse_analog_inputs(codec);
2577a86a88eaSTakashi Iwai 	if (err < 0)
2578a86a88eaSTakashi Iwai 		return err;
2579a86a88eaSTakashi Iwai 	if (spec->dyn_adc_switch)
2580a86a88eaSTakashi Iwai 		err = create_dyn_adc_ctls(codec);
2581a86a88eaSTakashi Iwai 	else
2582a86a88eaSTakashi Iwai 		err = create_multi_adc_ctls(codec);
2583a86a88eaSTakashi Iwai 	if (err < 0)
2584a86a88eaSTakashi Iwai 		return err;
2585a86a88eaSTakashi Iwai 	err = create_loopback_ctls(codec);
2586a86a88eaSTakashi Iwai 	if (err < 0)
2587a86a88eaSTakashi Iwai 		return err;
2588a86a88eaSTakashi Iwai 	err = create_mic_boost_ctls(codec);
2589a86a88eaSTakashi Iwai 	if (err < 0)
2590a86a88eaSTakashi Iwai 		return err;
2591c577b8a1SJoseph Chan 	return 0;
2592c577b8a1SJoseph Chan }
2593c577b8a1SJoseph Chan 
259476d9b0ddSHarald Welte static void vt1708_set_pinconfig_connect(struct hda_codec *codec, hda_nid_t nid)
259576d9b0ddSHarald Welte {
259676d9b0ddSHarald Welte 	unsigned int def_conf;
259776d9b0ddSHarald Welte 	unsigned char seqassoc;
259876d9b0ddSHarald Welte 
25992f334f92STakashi Iwai 	def_conf = snd_hda_codec_get_pincfg(codec, nid);
260076d9b0ddSHarald Welte 	seqassoc = (unsigned char) get_defcfg_association(def_conf);
260176d9b0ddSHarald Welte 	seqassoc = (seqassoc << 4) | get_defcfg_sequence(def_conf);
260282ef9e45SLydia Wang 	if (get_defcfg_connect(def_conf) == AC_JACK_PORT_NONE
260382ef9e45SLydia Wang 	    && (seqassoc == 0xf0 || seqassoc == 0xff)) {
260476d9b0ddSHarald Welte 		def_conf = def_conf & (~(AC_JACK_PORT_BOTH << 30));
26052f334f92STakashi Iwai 		snd_hda_codec_set_pincfg(codec, nid, def_conf);
260676d9b0ddSHarald Welte 	}
260776d9b0ddSHarald Welte 
260876d9b0ddSHarald Welte 	return;
260976d9b0ddSHarald Welte }
261076d9b0ddSHarald Welte 
2611e06e5a29STakashi Iwai static int vt1708_jack_detect_get(struct snd_kcontrol *kcontrol,
26121f2e99feSLydia Wang 				     struct snd_ctl_elem_value *ucontrol)
26131f2e99feSLydia Wang {
26141f2e99feSLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
26151f2e99feSLydia Wang 	struct via_spec *spec = codec->spec;
26161f2e99feSLydia Wang 
26171f2e99feSLydia Wang 	if (spec->codec_type != VT1708)
26181f2e99feSLydia Wang 		return 0;
2619e06e5a29STakashi Iwai 	ucontrol->value.integer.value[0] = spec->vt1708_jack_detect;
26201f2e99feSLydia Wang 	return 0;
26211f2e99feSLydia Wang }
26221f2e99feSLydia Wang 
2623e06e5a29STakashi Iwai static int vt1708_jack_detect_put(struct snd_kcontrol *kcontrol,
26241f2e99feSLydia Wang 				     struct snd_ctl_elem_value *ucontrol)
26251f2e99feSLydia Wang {
26261f2e99feSLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
26271f2e99feSLydia Wang 	struct via_spec *spec = codec->spec;
2628187d333eSTakashi Iwai 	int val;
26291f2e99feSLydia Wang 
26301f2e99feSLydia Wang 	if (spec->codec_type != VT1708)
26311f2e99feSLydia Wang 		return 0;
2632187d333eSTakashi Iwai 	val = !!ucontrol->value.integer.value[0];
2633187d333eSTakashi Iwai 	if (spec->vt1708_jack_detect == val)
2634187d333eSTakashi Iwai 		return 0;
2635187d333eSTakashi Iwai 	spec->vt1708_jack_detect = val;
2636187d333eSTakashi Iwai 	if (spec->vt1708_jack_detect &&
2637187d333eSTakashi Iwai 	    snd_hda_get_bool_hint(codec, "analog_loopback_hp_detect") != 1) {
26381f2e99feSLydia Wang 		mute_aa_path(codec, 1);
26391f2e99feSLydia Wang 		notify_aa_path_ctls(codec);
26401f2e99feSLydia Wang 	}
2641187d333eSTakashi Iwai 	via_hp_automute(codec);
2642187d333eSTakashi Iwai 	vt1708_update_hp_work(spec);
2643187d333eSTakashi Iwai 	return 1;
26441f2e99feSLydia Wang }
26451f2e99feSLydia Wang 
2646e06e5a29STakashi Iwai static const struct snd_kcontrol_new vt1708_jack_detect_ctl = {
26471f2e99feSLydia Wang 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
26481f2e99feSLydia Wang 	.name = "Jack Detect",
26491f2e99feSLydia Wang 	.count = 1,
26501f2e99feSLydia Wang 	.info = snd_ctl_boolean_mono_info,
2651e06e5a29STakashi Iwai 	.get = vt1708_jack_detect_get,
2652e06e5a29STakashi Iwai 	.put = vt1708_jack_detect_put,
26531f2e99feSLydia Wang };
26541f2e99feSLydia Wang 
265512daef65STakashi Iwai static void fill_dig_outs(struct hda_codec *codec);
265612daef65STakashi Iwai static void fill_dig_in(struct hda_codec *codec);
265712daef65STakashi Iwai 
265812daef65STakashi Iwai static int via_parse_auto_config(struct hda_codec *codec)
2659c577b8a1SJoseph Chan {
2660c577b8a1SJoseph Chan 	struct via_spec *spec = codec->spec;
2661c577b8a1SJoseph Chan 	int err;
2662c577b8a1SJoseph Chan 
2663c577b8a1SJoseph Chan 	err = snd_hda_parse_pin_def_config(codec, &spec->autocfg, NULL);
2664c577b8a1SJoseph Chan 	if (err < 0)
2665c577b8a1SJoseph Chan 		return err;
2666c577b8a1SJoseph Chan 	if (!spec->autocfg.line_outs && !spec->autocfg.hp_pins[0])
26677f0df88cSTakashi Iwai 		return -EINVAL;
2668c577b8a1SJoseph Chan 
26694a79616dSTakashi Iwai 	err = via_auto_create_multi_out_ctls(codec);
2670c577b8a1SJoseph Chan 	if (err < 0)
2671c577b8a1SJoseph Chan 		return err;
26724a79616dSTakashi Iwai 	err = via_auto_create_hp_ctls(codec, spec->autocfg.hp_pins[0]);
2673c577b8a1SJoseph Chan 	if (err < 0)
2674c577b8a1SJoseph Chan 		return err;
26754a918ffeSTakashi Iwai 	err = via_auto_create_speaker_ctls(codec);
26764a918ffeSTakashi Iwai 	if (err < 0)
26774a918ffeSTakashi Iwai 		return err;
26783214b966STakashi Iwai 	err = via_auto_create_loopback_switch(codec);
26793214b966STakashi Iwai 	if (err < 0)
26803214b966STakashi Iwai 		return err;
2681a86a88eaSTakashi Iwai 	err = via_auto_create_analog_input_ctls(codec);
2682c577b8a1SJoseph Chan 	if (err < 0)
2683c577b8a1SJoseph Chan 		return err;
2684c577b8a1SJoseph Chan 
2685c577b8a1SJoseph Chan 	spec->multiout.max_channels = spec->multiout.num_dacs * 2;
2686c577b8a1SJoseph Chan 
268712daef65STakashi Iwai 	fill_dig_outs(codec);
268812daef65STakashi Iwai 	fill_dig_in(codec);
2689c577b8a1SJoseph Chan 
2690603c4019STakashi Iwai 	if (spec->kctls.list)
2691603c4019STakashi Iwai 		spec->mixers[spec->num_mixers++] = spec->kctls.list;
2692c577b8a1SJoseph Chan 
2693c577b8a1SJoseph Chan 
26943214b966STakashi Iwai 	if (spec->hp_dac_nid && spec->hp_mix_path.depth) {
2695ece8d043STakashi Iwai 		err = via_hp_build(codec);
2696ece8d043STakashi Iwai 		if (err < 0)
2697ece8d043STakashi Iwai 			return err;
2698ece8d043STakashi Iwai 	}
2699c577b8a1SJoseph Chan 
2700f4a7828bSTakashi Iwai 	err = via_smart51_build(codec);
2701f4a7828bSTakashi Iwai 	if (err < 0)
2702f4a7828bSTakashi Iwai 		return err;
2703f4a7828bSTakashi Iwai 
27045d41762aSTakashi Iwai 	/* assign slave outs */
27055d41762aSTakashi Iwai 	if (spec->slave_dig_outs[0])
27065d41762aSTakashi Iwai 		codec->slave_dig_outs = spec->slave_dig_outs;
27075d41762aSTakashi Iwai 
2708c577b8a1SJoseph Chan 	return 1;
2709c577b8a1SJoseph Chan }
2710c577b8a1SJoseph Chan 
27115d41762aSTakashi Iwai static void via_auto_init_dig_outs(struct hda_codec *codec)
2712c577b8a1SJoseph Chan {
271325eaba2fSLydia Wang 	struct via_spec *spec = codec->spec;
27145d41762aSTakashi Iwai 	if (spec->multiout.dig_out_nid)
27155d41762aSTakashi Iwai 		init_output_pin(codec, spec->autocfg.dig_out_pins[0], PIN_OUT);
27165d41762aSTakashi Iwai 	if (spec->slave_dig_outs[0])
27175d41762aSTakashi Iwai 		init_output_pin(codec, spec->autocfg.dig_out_pins[1], PIN_OUT);
27185d41762aSTakashi Iwai }
271925eaba2fSLydia Wang 
27205d41762aSTakashi Iwai static void via_auto_init_dig_in(struct hda_codec *codec)
27215d41762aSTakashi Iwai {
27225d41762aSTakashi Iwai 	struct via_spec *spec = codec->spec;
27235d41762aSTakashi Iwai 	if (!spec->dig_in_nid)
27245d41762aSTakashi Iwai 		return;
2725cdd03cedSTakashi Iwai 	snd_hda_set_pin_ctl(codec, spec->autocfg.dig_in_pin, PIN_IN);
27265d41762aSTakashi Iwai }
27275d41762aSTakashi Iwai 
27284e2d16d3SDavid Henningsson static void via_jack_output_event(struct hda_codec *codec, struct hda_jack_tbl *tbl)
27294e2d16d3SDavid Henningsson {
27304e2d16d3SDavid Henningsson 	set_widgets_power_state(codec);
27314e2d16d3SDavid Henningsson 	via_hp_automute(codec);
27324e2d16d3SDavid Henningsson }
27334e2d16d3SDavid Henningsson 
27344e2d16d3SDavid Henningsson static void via_jack_powerstate_event(struct hda_codec *codec, struct hda_jack_tbl *tbl)
27354e2d16d3SDavid Henningsson {
27364e2d16d3SDavid Henningsson 	set_widgets_power_state(codec);
27374e2d16d3SDavid Henningsson }
27384e2d16d3SDavid Henningsson 
27394a918ffeSTakashi Iwai /* initialize the unsolicited events */
27404a918ffeSTakashi Iwai static void via_auto_init_unsol_event(struct hda_codec *codec)
27414a918ffeSTakashi Iwai {
27424a918ffeSTakashi Iwai 	struct via_spec *spec = codec->spec;
27434a918ffeSTakashi Iwai 	struct auto_pin_cfg *cfg = &spec->autocfg;
27444a918ffeSTakashi Iwai 	unsigned int ev;
27454a918ffeSTakashi Iwai 	int i;
27464e2d16d3SDavid Henningsson 	hda_jack_callback cb;
27474a918ffeSTakashi Iwai 
27484a918ffeSTakashi Iwai 	if (cfg->hp_pins[0] && is_jack_detectable(codec, cfg->hp_pins[0]))
27494e2d16d3SDavid Henningsson 		snd_hda_jack_detect_enable_callback(codec, cfg->hp_pins[0],
27504e2d16d3SDavid Henningsson 						    VIA_HP_EVENT | VIA_JACK_EVENT,
27514e2d16d3SDavid Henningsson 						    via_jack_output_event);
27524a918ffeSTakashi Iwai 
27534a918ffeSTakashi Iwai 	if (cfg->speaker_pins[0])
27544a918ffeSTakashi Iwai 		ev = VIA_LINE_EVENT;
27554a918ffeSTakashi Iwai 	else
27564a918ffeSTakashi Iwai 		ev = 0;
27574e2d16d3SDavid Henningsson 	cb = ev ? via_jack_output_event : via_jack_powerstate_event;
27584e2d16d3SDavid Henningsson 
27594a918ffeSTakashi Iwai 	for (i = 0; i < cfg->line_outs; i++) {
27604a918ffeSTakashi Iwai 		if (cfg->line_out_pins[i] &&
27614a918ffeSTakashi Iwai 		    is_jack_detectable(codec, cfg->line_out_pins[i]))
27624e2d16d3SDavid Henningsson 			snd_hda_jack_detect_enable_callback(codec, cfg->line_out_pins[i],
27634e2d16d3SDavid Henningsson 							    ev | VIA_JACK_EVENT, cb);
27644a918ffeSTakashi Iwai 	}
27654a918ffeSTakashi Iwai 
27664a918ffeSTakashi Iwai 	for (i = 0; i < cfg->num_inputs; i++) {
27674a918ffeSTakashi Iwai 		if (is_jack_detectable(codec, cfg->inputs[i].pin))
27684e2d16d3SDavid Henningsson 			snd_hda_jack_detect_enable_callback(codec, cfg->inputs[i].pin,
27694e2d16d3SDavid Henningsson 							    VIA_JACK_EVENT,
27704e2d16d3SDavid Henningsson 							    via_jack_powerstate_event);
27714a918ffeSTakashi Iwai 	}
27724a918ffeSTakashi Iwai }
27734a918ffeSTakashi Iwai 
27745d41762aSTakashi Iwai static int via_init(struct hda_codec *codec)
27755d41762aSTakashi Iwai {
27765d41762aSTakashi Iwai 	struct via_spec *spec = codec->spec;
27775d41762aSTakashi Iwai 	int i;
27785d41762aSTakashi Iwai 
27795d41762aSTakashi Iwai 	for (i = 0; i < spec->num_iverbs; i++)
27805d41762aSTakashi Iwai 		snd_hda_sequence_write(codec, spec->init_verbs[i]);
27815d41762aSTakashi Iwai 
2782e9d010c2STakashi Iwai 	/* init power states */
2783e9d010c2STakashi Iwai 	set_widgets_power_state(codec);
2784e9d010c2STakashi Iwai 	__analog_low_current_mode(codec, true);
2785e9d010c2STakashi Iwai 
2786c577b8a1SJoseph Chan 	via_auto_init_multi_out(codec);
2787c577b8a1SJoseph Chan 	via_auto_init_hp_out(codec);
27884a918ffeSTakashi Iwai 	via_auto_init_speaker_out(codec);
2789c577b8a1SJoseph Chan 	via_auto_init_analog_input(codec);
27905d41762aSTakashi Iwai 	via_auto_init_dig_outs(codec);
27915d41762aSTakashi Iwai 	via_auto_init_dig_in(codec);
279211890956SLydia Wang 
27934a918ffeSTakashi Iwai 	via_auto_init_unsol_event(codec);
27944a918ffeSTakashi Iwai 
279525eaba2fSLydia Wang 	via_hp_automute(codec);
2796187d333eSTakashi Iwai 	vt1708_update_hp_work(spec);
279725eaba2fSLydia Wang 
2798c577b8a1SJoseph Chan 	return 0;
2799c577b8a1SJoseph Chan }
2800c577b8a1SJoseph Chan 
28011f2e99feSLydia Wang static void vt1708_update_hp_jack_state(struct work_struct *work)
28021f2e99feSLydia Wang {
28031f2e99feSLydia Wang 	struct via_spec *spec = container_of(work, struct via_spec,
28041f2e99feSLydia Wang 					     vt1708_hp_work.work);
28051f2e99feSLydia Wang 	if (spec->codec_type != VT1708)
28061f2e99feSLydia Wang 		return;
28071835a0f9STakashi Iwai 	snd_hda_jack_set_dirty_all(spec->codec);
28081f2e99feSLydia Wang 	/* if jack state toggled */
28091f2e99feSLydia Wang 	if (spec->vt1708_hp_present
2810d56757abSTakashi Iwai 	    != snd_hda_jack_detect(spec->codec, spec->autocfg.hp_pins[0])) {
28111f2e99feSLydia Wang 		spec->vt1708_hp_present ^= 1;
28121f2e99feSLydia Wang 		via_hp_automute(spec->codec);
28131f2e99feSLydia Wang 	}
2814187d333eSTakashi Iwai 	if (spec->vt1708_jack_detect)
2815187d333eSTakashi Iwai 		schedule_delayed_work(&spec->vt1708_hp_work,
2816187d333eSTakashi Iwai 				      msecs_to_jiffies(100));
28171f2e99feSLydia Wang }
28181f2e99feSLydia Wang 
2819337b9d02STakashi Iwai static int get_mux_nids(struct hda_codec *codec)
2820337b9d02STakashi Iwai {
2821337b9d02STakashi Iwai 	struct via_spec *spec = codec->spec;
2822337b9d02STakashi Iwai 	hda_nid_t nid, conn[8];
2823337b9d02STakashi Iwai 	unsigned int type;
2824337b9d02STakashi Iwai 	int i, n;
2825337b9d02STakashi Iwai 
2826337b9d02STakashi Iwai 	for (i = 0; i < spec->num_adc_nids; i++) {
2827337b9d02STakashi Iwai 		nid = spec->adc_nids[i];
2828337b9d02STakashi Iwai 		while (nid) {
2829a22d543aSTakashi Iwai 			type = get_wcaps_type(get_wcaps(codec, nid));
28301c55d521STakashi Iwai 			if (type == AC_WID_PIN)
28311c55d521STakashi Iwai 				break;
2832337b9d02STakashi Iwai 			n = snd_hda_get_connections(codec, nid, conn,
2833337b9d02STakashi Iwai 						    ARRAY_SIZE(conn));
2834337b9d02STakashi Iwai 			if (n <= 0)
2835337b9d02STakashi Iwai 				break;
2836337b9d02STakashi Iwai 			if (n > 1) {
2837337b9d02STakashi Iwai 				spec->mux_nids[i] = nid;
2838337b9d02STakashi Iwai 				break;
2839337b9d02STakashi Iwai 			}
2840337b9d02STakashi Iwai 			nid = conn[0];
2841337b9d02STakashi Iwai 		}
2842337b9d02STakashi Iwai 	}
28431c55d521STakashi Iwai 	return 0;
2844337b9d02STakashi Iwai }
2845337b9d02STakashi Iwai 
2846c577b8a1SJoseph Chan static int patch_vt1708(struct hda_codec *codec)
2847c577b8a1SJoseph Chan {
2848c577b8a1SJoseph Chan 	struct via_spec *spec;
2849c577b8a1SJoseph Chan 	int err;
2850c577b8a1SJoseph Chan 
2851c577b8a1SJoseph Chan 	/* create a codec specific record */
28525b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
2853c577b8a1SJoseph Chan 	if (spec == NULL)
2854c577b8a1SJoseph Chan 		return -ENOMEM;
2855c577b8a1SJoseph Chan 
2856620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x17;
2857620e2b28STakashi Iwai 
285812daef65STakashi Iwai 	/* Add HP and CD pin config connect bit re-config action */
285912daef65STakashi Iwai 	vt1708_set_pinconfig_connect(codec, VT1708_HP_PIN_NID);
286012daef65STakashi Iwai 	vt1708_set_pinconfig_connect(codec, VT1708_CD_PIN_NID);
286112daef65STakashi Iwai 
2862c577b8a1SJoseph Chan 	/* automatic parse from the BIOS config */
286312daef65STakashi Iwai 	err = via_parse_auto_config(codec);
2864c577b8a1SJoseph Chan 	if (err < 0) {
2865c577b8a1SJoseph Chan 		via_free(codec);
2866c577b8a1SJoseph Chan 		return err;
2867c577b8a1SJoseph Chan 	}
2868c577b8a1SJoseph Chan 
286912daef65STakashi Iwai 	/* add jack detect on/off control */
287012daef65STakashi Iwai 	if (!via_clone_control(spec, &vt1708_jack_detect_ctl))
287112daef65STakashi Iwai 		return -ENOMEM;
287212daef65STakashi Iwai 
2873bc9b5623STakashi Iwai 	/* disable 32bit format on VT1708 */
2874bc9b5623STakashi Iwai 	if (codec->vendor_id == 0x11061708)
2875bc9b5623STakashi Iwai 		spec->stream_analog_playback = &vt1708_pcm_analog_s16_playback;
2876c577b8a1SJoseph Chan 
2877e322a36dSLydia Wang 	spec->init_verbs[spec->num_iverbs++] = vt1708_init_verbs;
2878e322a36dSLydia Wang 
2879c577b8a1SJoseph Chan 	codec->patch_ops = via_patch_ops;
2880c577b8a1SJoseph Chan 
28811f2e99feSLydia Wang 	INIT_DELAYED_WORK(&spec->vt1708_hp_work, vt1708_update_hp_jack_state);
2882c577b8a1SJoseph Chan 	return 0;
2883c577b8a1SJoseph Chan }
2884c577b8a1SJoseph Chan 
2885ddd304d8STakashi Iwai static int patch_vt1709(struct hda_codec *codec)
2886c577b8a1SJoseph Chan {
2887c577b8a1SJoseph Chan 	struct via_spec *spec;
2888c577b8a1SJoseph Chan 	int err;
2889c577b8a1SJoseph Chan 
2890c577b8a1SJoseph Chan 	/* create a codec specific record */
28915b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
2892c577b8a1SJoseph Chan 	if (spec == NULL)
2893c577b8a1SJoseph Chan 		return -ENOMEM;
2894c577b8a1SJoseph Chan 
2895620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x18;
2896620e2b28STakashi Iwai 
289712daef65STakashi Iwai 	err = via_parse_auto_config(codec);
2898c577b8a1SJoseph Chan 	if (err < 0) {
2899c577b8a1SJoseph Chan 		via_free(codec);
2900c577b8a1SJoseph Chan 		return err;
2901c577b8a1SJoseph Chan 	}
2902c577b8a1SJoseph Chan 
2903c577b8a1SJoseph Chan 	codec->patch_ops = via_patch_ops;
2904c577b8a1SJoseph Chan 
2905f7278fd0SJosepch Chan 	return 0;
2906f7278fd0SJosepch Chan }
2907f7278fd0SJosepch Chan 
29083e95b9abSLydia Wang static void set_widgets_power_state_vt1708B(struct hda_codec *codec)
29093e95b9abSLydia Wang {
29103e95b9abSLydia Wang 	struct via_spec *spec = codec->spec;
29113e95b9abSLydia Wang 	int imux_is_smixer;
29123e95b9abSLydia Wang 	unsigned int parm;
29133e95b9abSLydia Wang 	int is_8ch = 0;
2914bc92df7fSLydia Wang 	if ((spec->codec_type != VT1708B_4CH) &&
2915bc92df7fSLydia Wang 	    (codec->vendor_id != 0x11064397))
29163e95b9abSLydia Wang 		is_8ch = 1;
29173e95b9abSLydia Wang 
29183e95b9abSLydia Wang 	/* SW0 (17h) = stereo mixer */
29193e95b9abSLydia Wang 	imux_is_smixer =
29203e95b9abSLydia Wang 	(snd_hda_codec_read(codec, 0x17, 0, AC_VERB_GET_CONNECT_SEL, 0x00)
29213e95b9abSLydia Wang 	 == ((spec->codec_type == VT1708S) ? 5 : 0));
29223e95b9abSLydia Wang 	/* inputs */
29233e95b9abSLydia Wang 	/* PW 1/2/5 (1ah/1bh/1eh) */
29243e95b9abSLydia Wang 	parm = AC_PWRST_D3;
29253e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1a, &parm);
29263e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1b, &parm);
29273e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1e, &parm);
29283e95b9abSLydia Wang 	if (imux_is_smixer)
29293e95b9abSLydia Wang 		parm = AC_PWRST_D0;
29303e95b9abSLydia Wang 	/* SW0 (17h), AIW 0/1 (13h/14h) */
2931054d867eSTakashi Iwai 	update_power_state(codec, 0x17, parm);
2932054d867eSTakashi Iwai 	update_power_state(codec, 0x13, parm);
2933054d867eSTakashi Iwai 	update_power_state(codec, 0x14, parm);
29343e95b9abSLydia Wang 
29353e95b9abSLydia Wang 	/* outputs */
29363e95b9abSLydia Wang 	/* PW0 (19h), SW1 (18h), AOW1 (11h) */
29373e95b9abSLydia Wang 	parm = AC_PWRST_D3;
29383e95b9abSLydia Wang 	set_pin_power_state(codec, 0x19, &parm);
29393e95b9abSLydia Wang 	if (spec->smart51_enabled)
29403e95b9abSLydia Wang 		set_pin_power_state(codec, 0x1b, &parm);
2941054d867eSTakashi Iwai 	update_power_state(codec, 0x18, parm);
2942054d867eSTakashi Iwai 	update_power_state(codec, 0x11, parm);
29433e95b9abSLydia Wang 
29443e95b9abSLydia Wang 	/* PW6 (22h), SW2 (26h), AOW2 (24h) */
29453e95b9abSLydia Wang 	if (is_8ch) {
29463e95b9abSLydia Wang 		parm = AC_PWRST_D3;
29473e95b9abSLydia Wang 		set_pin_power_state(codec, 0x22, &parm);
29483e95b9abSLydia Wang 		if (spec->smart51_enabled)
29493e95b9abSLydia Wang 			set_pin_power_state(codec, 0x1a, &parm);
2950054d867eSTakashi Iwai 		update_power_state(codec, 0x26, parm);
2951054d867eSTakashi Iwai 		update_power_state(codec, 0x24, parm);
2952bc92df7fSLydia Wang 	} else if (codec->vendor_id == 0x11064397) {
2953bc92df7fSLydia Wang 		/* PW7(23h), SW2(27h), AOW2(25h) */
2954bc92df7fSLydia Wang 		parm = AC_PWRST_D3;
2955bc92df7fSLydia Wang 		set_pin_power_state(codec, 0x23, &parm);
2956bc92df7fSLydia Wang 		if (spec->smart51_enabled)
2957bc92df7fSLydia Wang 			set_pin_power_state(codec, 0x1a, &parm);
2958054d867eSTakashi Iwai 		update_power_state(codec, 0x27, parm);
2959054d867eSTakashi Iwai 		update_power_state(codec, 0x25, parm);
29603e95b9abSLydia Wang 	}
29613e95b9abSLydia Wang 
29623e95b9abSLydia Wang 	/* PW 3/4/7 (1ch/1dh/23h) */
29633e95b9abSLydia Wang 	parm = AC_PWRST_D3;
29643e95b9abSLydia Wang 	/* force to D0 for internal Speaker */
29653e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1c, &parm);
29663e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1d, &parm);
29673e95b9abSLydia Wang 	if (is_8ch)
29683e95b9abSLydia Wang 		set_pin_power_state(codec, 0x23, &parm);
29693e95b9abSLydia Wang 
29703e95b9abSLydia Wang 	/* MW0 (16h), Sw3 (27h), AOW 0/3 (10h/25h) */
2971054d867eSTakashi Iwai 	update_power_state(codec, 0x16, imux_is_smixer ? AC_PWRST_D0 : parm);
2972054d867eSTakashi Iwai 	update_power_state(codec, 0x10, parm);
29733e95b9abSLydia Wang 	if (is_8ch) {
2974054d867eSTakashi Iwai 		update_power_state(codec, 0x25, parm);
2975054d867eSTakashi Iwai 		update_power_state(codec, 0x27, parm);
2976bc92df7fSLydia Wang 	} else if (codec->vendor_id == 0x11064397 && spec->hp_independent_mode)
2977054d867eSTakashi Iwai 		update_power_state(codec, 0x25, parm);
29783e95b9abSLydia Wang }
29793e95b9abSLydia Wang 
2980518bf3baSLydia Wang static int patch_vt1708S(struct hda_codec *codec);
2981ddd304d8STakashi Iwai static int patch_vt1708B(struct hda_codec *codec)
2982f7278fd0SJosepch Chan {
2983f7278fd0SJosepch Chan 	struct via_spec *spec;
2984f7278fd0SJosepch Chan 	int err;
2985f7278fd0SJosepch Chan 
2986518bf3baSLydia Wang 	if (get_codec_type(codec) == VT1708BCE)
2987518bf3baSLydia Wang 		return patch_vt1708S(codec);
2988ddd304d8STakashi Iwai 
2989f7278fd0SJosepch Chan 	/* create a codec specific record */
29905b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
2991f7278fd0SJosepch Chan 	if (spec == NULL)
2992f7278fd0SJosepch Chan 		return -ENOMEM;
2993f7278fd0SJosepch Chan 
2994620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x16;
2995620e2b28STakashi Iwai 
2996f7278fd0SJosepch Chan 	/* automatic parse from the BIOS config */
299712daef65STakashi Iwai 	err = via_parse_auto_config(codec);
2998f7278fd0SJosepch Chan 	if (err < 0) {
2999f7278fd0SJosepch Chan 		via_free(codec);
3000f7278fd0SJosepch Chan 		return err;
3001f7278fd0SJosepch Chan 	}
3002f7278fd0SJosepch Chan 
3003f7278fd0SJosepch Chan 	codec->patch_ops = via_patch_ops;
3004f7278fd0SJosepch Chan 
30053e95b9abSLydia Wang 	spec->set_widgets_power_state =  set_widgets_power_state_vt1708B;
30063e95b9abSLydia Wang 
3007f7278fd0SJosepch Chan 	return 0;
3008f7278fd0SJosepch Chan }
3009f7278fd0SJosepch Chan 
3010d949cac1SHarald Welte /* Patch for VT1708S */
3011096a8854STakashi Iwai static const struct hda_verb vt1708S_init_verbs[] = {
3012d7426329SHarald Welte 	/* Enable Mic Boost Volume backdoor */
3013d7426329SHarald Welte 	{0x1, 0xf98, 0x1},
3014bc7e7e5cSLydia Wang 	/* don't bybass mixer */
3015bc7e7e5cSLydia Wang 	{0x1, 0xf88, 0xc0},
3016d949cac1SHarald Welte 	{ }
3017d949cac1SHarald Welte };
3018d949cac1SHarald Welte 
30199da29271STakashi Iwai /* fill out digital output widgets; one for master and one for slave outputs */
30209da29271STakashi Iwai static void fill_dig_outs(struct hda_codec *codec)
30219da29271STakashi Iwai {
30229da29271STakashi Iwai 	struct via_spec *spec = codec->spec;
30239da29271STakashi Iwai 	int i;
30249da29271STakashi Iwai 
30259da29271STakashi Iwai 	for (i = 0; i < spec->autocfg.dig_outs; i++) {
30269da29271STakashi Iwai 		hda_nid_t nid;
30279da29271STakashi Iwai 		int conn;
30289da29271STakashi Iwai 
30299da29271STakashi Iwai 		nid = spec->autocfg.dig_out_pins[i];
30309da29271STakashi Iwai 		if (!nid)
30319da29271STakashi Iwai 			continue;
30329da29271STakashi Iwai 		conn = snd_hda_get_connections(codec, nid, &nid, 1);
30339da29271STakashi Iwai 		if (conn < 1)
30349da29271STakashi Iwai 			continue;
30359da29271STakashi Iwai 		if (!spec->multiout.dig_out_nid)
30369da29271STakashi Iwai 			spec->multiout.dig_out_nid = nid;
30379da29271STakashi Iwai 		else {
30389da29271STakashi Iwai 			spec->slave_dig_outs[0] = nid;
30399da29271STakashi Iwai 			break; /* at most two dig outs */
30409da29271STakashi Iwai 		}
30419da29271STakashi Iwai 	}
30429da29271STakashi Iwai }
30439da29271STakashi Iwai 
304412daef65STakashi Iwai static void fill_dig_in(struct hda_codec *codec)
3045d949cac1SHarald Welte {
3046d949cac1SHarald Welte 	struct via_spec *spec = codec->spec;
304712daef65STakashi Iwai 	hda_nid_t dig_nid;
304812daef65STakashi Iwai 	int i, err;
3049d949cac1SHarald Welte 
305012daef65STakashi Iwai 	if (!spec->autocfg.dig_in_pin)
305112daef65STakashi Iwai 		return;
3052d949cac1SHarald Welte 
305312daef65STakashi Iwai 	dig_nid = codec->start_nid;
305412daef65STakashi Iwai 	for (i = 0; i < codec->num_nodes; i++, dig_nid++) {
305512daef65STakashi Iwai 		unsigned int wcaps = get_wcaps(codec, dig_nid);
305612daef65STakashi Iwai 		if (get_wcaps_type(wcaps) != AC_WID_AUD_IN)
305712daef65STakashi Iwai 			continue;
305812daef65STakashi Iwai 		if (!(wcaps & AC_WCAP_DIGITAL))
305912daef65STakashi Iwai 			continue;
306012daef65STakashi Iwai 		if (!(wcaps & AC_WCAP_CONN_LIST))
306112daef65STakashi Iwai 			continue;
306212daef65STakashi Iwai 		err = get_connection_index(codec, dig_nid,
306312daef65STakashi Iwai 					   spec->autocfg.dig_in_pin);
306412daef65STakashi Iwai 		if (err >= 0) {
306512daef65STakashi Iwai 			spec->dig_in_nid = dig_nid;
306612daef65STakashi Iwai 			break;
306712daef65STakashi Iwai 		}
306812daef65STakashi Iwai 	}
3069d949cac1SHarald Welte }
3070d949cac1SHarald Welte 
30716369bcfcSLydia Wang static void override_mic_boost(struct hda_codec *codec, hda_nid_t pin,
30726369bcfcSLydia Wang 			       int offset, int num_steps, int step_size)
30736369bcfcSLydia Wang {
30746369bcfcSLydia Wang 	snd_hda_override_amp_caps(codec, pin, HDA_INPUT,
30756369bcfcSLydia Wang 				  (offset << AC_AMPCAP_OFFSET_SHIFT) |
30766369bcfcSLydia Wang 				  (num_steps << AC_AMPCAP_NUM_STEPS_SHIFT) |
30776369bcfcSLydia Wang 				  (step_size << AC_AMPCAP_STEP_SIZE_SHIFT) |
30786369bcfcSLydia Wang 				  (0 << AC_AMPCAP_MUTE_SHIFT));
30796369bcfcSLydia Wang }
30806369bcfcSLydia Wang 
3081d949cac1SHarald Welte static int patch_vt1708S(struct hda_codec *codec)
3082d949cac1SHarald Welte {
3083d949cac1SHarald Welte 	struct via_spec *spec;
3084d949cac1SHarald Welte 	int err;
3085d949cac1SHarald Welte 
3086d949cac1SHarald Welte 	/* create a codec specific record */
30875b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
3088d949cac1SHarald Welte 	if (spec == NULL)
3089d949cac1SHarald Welte 		return -ENOMEM;
3090d949cac1SHarald Welte 
3091620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x16;
3092d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x1a, 0, 3, 40);
3093d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x1e, 0, 3, 40);
3094620e2b28STakashi Iwai 
3095d949cac1SHarald Welte 	/* automatic parse from the BIOS config */
309612daef65STakashi Iwai 	err = via_parse_auto_config(codec);
3097d949cac1SHarald Welte 	if (err < 0) {
3098d949cac1SHarald Welte 		via_free(codec);
3099d949cac1SHarald Welte 		return err;
3100d949cac1SHarald Welte 	}
3101d949cac1SHarald Welte 
3102096a8854STakashi Iwai 	spec->init_verbs[spec->num_iverbs++] = vt1708S_init_verbs;
3103d949cac1SHarald Welte 
3104d949cac1SHarald Welte 	codec->patch_ops = via_patch_ops;
3105d949cac1SHarald Welte 
3106518bf3baSLydia Wang 	/* correct names for VT1708BCE */
3107518bf3baSLydia Wang 	if (get_codec_type(codec) == VT1708BCE)	{
3108518bf3baSLydia Wang 		kfree(codec->chip_name);
3109518bf3baSLydia Wang 		codec->chip_name = kstrdup("VT1708BCE", GFP_KERNEL);
3110518bf3baSLydia Wang 		snprintf(codec->bus->card->mixername,
3111518bf3baSLydia Wang 			 sizeof(codec->bus->card->mixername),
3112518bf3baSLydia Wang 			 "%s %s", codec->vendor_name, codec->chip_name);
3113970f630fSLydia Wang 	}
3114bc92df7fSLydia Wang 	/* correct names for VT1705 */
3115bc92df7fSLydia Wang 	if (codec->vendor_id == 0x11064397)	{
3116bc92df7fSLydia Wang 		kfree(codec->chip_name);
3117bc92df7fSLydia Wang 		codec->chip_name = kstrdup("VT1705", GFP_KERNEL);
3118bc92df7fSLydia Wang 		snprintf(codec->bus->card->mixername,
3119bc92df7fSLydia Wang 			 sizeof(codec->bus->card->mixername),
3120bc92df7fSLydia Wang 			 "%s %s", codec->vendor_name, codec->chip_name);
3121bc92df7fSLydia Wang 	}
31223e95b9abSLydia Wang 	spec->set_widgets_power_state =  set_widgets_power_state_vt1708B;
3123d949cac1SHarald Welte 	return 0;
3124d949cac1SHarald Welte }
3125d949cac1SHarald Welte 
3126d949cac1SHarald Welte /* Patch for VT1702 */
3127d949cac1SHarald Welte 
3128096a8854STakashi Iwai static const struct hda_verb vt1702_init_verbs[] = {
3129bc7e7e5cSLydia Wang 	/* mixer enable */
3130bc7e7e5cSLydia Wang 	{0x1, 0xF88, 0x3},
3131bc7e7e5cSLydia Wang 	/* GPIO 0~2 */
3132bc7e7e5cSLydia Wang 	{0x1, 0xF82, 0x3F},
3133d949cac1SHarald Welte 	{ }
3134d949cac1SHarald Welte };
3135d949cac1SHarald Welte 
31363e95b9abSLydia Wang static void set_widgets_power_state_vt1702(struct hda_codec *codec)
31373e95b9abSLydia Wang {
31383e95b9abSLydia Wang 	int imux_is_smixer =
31393e95b9abSLydia Wang 	snd_hda_codec_read(codec, 0x13, 0, AC_VERB_GET_CONNECT_SEL, 0x00) == 3;
31403e95b9abSLydia Wang 	unsigned int parm;
31413e95b9abSLydia Wang 	/* inputs */
31423e95b9abSLydia Wang 	/* PW 1/2/5 (14h/15h/18h) */
31433e95b9abSLydia Wang 	parm = AC_PWRST_D3;
31443e95b9abSLydia Wang 	set_pin_power_state(codec, 0x14, &parm);
31453e95b9abSLydia Wang 	set_pin_power_state(codec, 0x15, &parm);
31463e95b9abSLydia Wang 	set_pin_power_state(codec, 0x18, &parm);
31473e95b9abSLydia Wang 	if (imux_is_smixer)
31483e95b9abSLydia Wang 		parm = AC_PWRST_D0; /* SW0 (13h) = stereo mixer (idx 3) */
31493e95b9abSLydia Wang 	/* SW0 (13h), AIW 0/1/2 (12h/1fh/20h) */
3150054d867eSTakashi Iwai 	update_power_state(codec, 0x13, parm);
3151054d867eSTakashi Iwai 	update_power_state(codec, 0x12, parm);
3152054d867eSTakashi Iwai 	update_power_state(codec, 0x1f, parm);
3153054d867eSTakashi Iwai 	update_power_state(codec, 0x20, parm);
31543e95b9abSLydia Wang 
31553e95b9abSLydia Wang 	/* outputs */
31563e95b9abSLydia Wang 	/* PW 3/4 (16h/17h) */
31573e95b9abSLydia Wang 	parm = AC_PWRST_D3;
31583e95b9abSLydia Wang 	set_pin_power_state(codec, 0x17, &parm);
31593e95b9abSLydia Wang 	set_pin_power_state(codec, 0x16, &parm);
31603e95b9abSLydia Wang 	/* MW0 (1ah), AOW 0/1 (10h/1dh) */
3161054d867eSTakashi Iwai 	update_power_state(codec, 0x1a, imux_is_smixer ? AC_PWRST_D0 : parm);
3162054d867eSTakashi Iwai 	update_power_state(codec, 0x10, parm);
3163054d867eSTakashi Iwai 	update_power_state(codec, 0x1d, parm);
31643e95b9abSLydia Wang }
31653e95b9abSLydia Wang 
3166d949cac1SHarald Welte static int patch_vt1702(struct hda_codec *codec)
3167d949cac1SHarald Welte {
3168d949cac1SHarald Welte 	struct via_spec *spec;
3169d949cac1SHarald Welte 	int err;
3170d949cac1SHarald Welte 
3171d949cac1SHarald Welte 	/* create a codec specific record */
31725b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
3173d949cac1SHarald Welte 	if (spec == NULL)
3174d949cac1SHarald Welte 		return -ENOMEM;
3175d949cac1SHarald Welte 
3176620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x1a;
3177620e2b28STakashi Iwai 
317812daef65STakashi Iwai 	/* limit AA path volume to 0 dB */
317912daef65STakashi Iwai 	snd_hda_override_amp_caps(codec, 0x1A, HDA_INPUT,
318012daef65STakashi Iwai 				  (0x17 << AC_AMPCAP_OFFSET_SHIFT) |
318112daef65STakashi Iwai 				  (0x17 << AC_AMPCAP_NUM_STEPS_SHIFT) |
318212daef65STakashi Iwai 				  (0x5 << AC_AMPCAP_STEP_SIZE_SHIFT) |
318312daef65STakashi Iwai 				  (1 << AC_AMPCAP_MUTE_SHIFT));
318412daef65STakashi Iwai 
3185d949cac1SHarald Welte 	/* automatic parse from the BIOS config */
318612daef65STakashi Iwai 	err = via_parse_auto_config(codec);
3187d949cac1SHarald Welte 	if (err < 0) {
3188d949cac1SHarald Welte 		via_free(codec);
3189d949cac1SHarald Welte 		return err;
3190d949cac1SHarald Welte 	}
3191d949cac1SHarald Welte 
3192096a8854STakashi Iwai 	spec->init_verbs[spec->num_iverbs++] = vt1702_init_verbs;
3193d949cac1SHarald Welte 
3194d949cac1SHarald Welte 	codec->patch_ops = via_patch_ops;
3195d949cac1SHarald Welte 
31963e95b9abSLydia Wang 	spec->set_widgets_power_state =  set_widgets_power_state_vt1702;
3197d949cac1SHarald Welte 	return 0;
3198d949cac1SHarald Welte }
3199d949cac1SHarald Welte 
3200eb7188caSLydia Wang /* Patch for VT1718S */
3201eb7188caSLydia Wang 
3202096a8854STakashi Iwai static const struct hda_verb vt1718S_init_verbs[] = {
32034ab2d53aSLydia Wang 	/* Enable MW0 adjust Gain 5 */
32044ab2d53aSLydia Wang 	{0x1, 0xfb2, 0x10},
3205eb7188caSLydia Wang 	/* Enable Boost Volume backdoor */
3206eb7188caSLydia Wang 	{0x1, 0xf88, 0x8},
32075d41762aSTakashi Iwai 
3208eb7188caSLydia Wang 	{ }
3209eb7188caSLydia Wang };
3210eb7188caSLydia Wang 
32113e95b9abSLydia Wang static void set_widgets_power_state_vt1718S(struct hda_codec *codec)
32123e95b9abSLydia Wang {
32133e95b9abSLydia Wang 	struct via_spec *spec = codec->spec;
32143e95b9abSLydia Wang 	int imux_is_smixer;
32156162552bSTakashi Iwai 	unsigned int parm, parm2;
32163e95b9abSLydia Wang 	/* MUX6 (1eh) = stereo mixer */
32173e95b9abSLydia Wang 	imux_is_smixer =
32183e95b9abSLydia Wang 	snd_hda_codec_read(codec, 0x1e, 0, AC_VERB_GET_CONNECT_SEL, 0x00) == 5;
32193e95b9abSLydia Wang 	/* inputs */
32203e95b9abSLydia Wang 	/* PW 5/6/7 (29h/2ah/2bh) */
32213e95b9abSLydia Wang 	parm = AC_PWRST_D3;
32223e95b9abSLydia Wang 	set_pin_power_state(codec, 0x29, &parm);
32233e95b9abSLydia Wang 	set_pin_power_state(codec, 0x2a, &parm);
32243e95b9abSLydia Wang 	set_pin_power_state(codec, 0x2b, &parm);
32253e95b9abSLydia Wang 	if (imux_is_smixer)
32263e95b9abSLydia Wang 		parm = AC_PWRST_D0;
32273e95b9abSLydia Wang 	/* MUX6/7 (1eh/1fh), AIW 0/1 (10h/11h) */
3228054d867eSTakashi Iwai 	update_power_state(codec, 0x1e, parm);
3229054d867eSTakashi Iwai 	update_power_state(codec, 0x1f, parm);
3230054d867eSTakashi Iwai 	update_power_state(codec, 0x10, parm);
3231054d867eSTakashi Iwai 	update_power_state(codec, 0x11, parm);
32323e95b9abSLydia Wang 
32333e95b9abSLydia Wang 	/* outputs */
32343e95b9abSLydia Wang 	/* PW3 (27h), MW2 (1ah), AOW3 (bh) */
32353e95b9abSLydia Wang 	parm = AC_PWRST_D3;
32363e95b9abSLydia Wang 	set_pin_power_state(codec, 0x27, &parm);
3237054d867eSTakashi Iwai 	update_power_state(codec, 0x1a, parm);
32386162552bSTakashi Iwai 	parm2 = parm; /* for pin 0x0b */
32393e95b9abSLydia Wang 
32403e95b9abSLydia Wang 	/* PW2 (26h), AOW2 (ah) */
32413e95b9abSLydia Wang 	parm = AC_PWRST_D3;
32423e95b9abSLydia Wang 	set_pin_power_state(codec, 0x26, &parm);
32433e95b9abSLydia Wang 	if (spec->smart51_enabled)
32443e95b9abSLydia Wang 		set_pin_power_state(codec, 0x2b, &parm);
3245054d867eSTakashi Iwai 	update_power_state(codec, 0xa, parm);
32463e95b9abSLydia Wang 
32473e95b9abSLydia Wang 	/* PW0 (24h), AOW0 (8h) */
32483e95b9abSLydia Wang 	parm = AC_PWRST_D3;
32493e95b9abSLydia Wang 	set_pin_power_state(codec, 0x24, &parm);
32503e95b9abSLydia Wang 	if (!spec->hp_independent_mode) /* check for redirected HP */
32513e95b9abSLydia Wang 		set_pin_power_state(codec, 0x28, &parm);
3252054d867eSTakashi Iwai 	update_power_state(codec, 0x8, parm);
32536162552bSTakashi Iwai 	if (!spec->hp_independent_mode && parm2 != AC_PWRST_D3)
32546162552bSTakashi Iwai 		parm = parm2;
32556162552bSTakashi Iwai 	update_power_state(codec, 0xb, parm);
32563e95b9abSLydia Wang 	/* MW9 (21h), Mw2 (1ah), AOW0 (8h) */
3257054d867eSTakashi Iwai 	update_power_state(codec, 0x21, imux_is_smixer ? AC_PWRST_D0 : parm);
32583e95b9abSLydia Wang 
32593e95b9abSLydia Wang 	/* PW1 (25h), AOW1 (9h) */
32603e95b9abSLydia Wang 	parm = AC_PWRST_D3;
32613e95b9abSLydia Wang 	set_pin_power_state(codec, 0x25, &parm);
32623e95b9abSLydia Wang 	if (spec->smart51_enabled)
32633e95b9abSLydia Wang 		set_pin_power_state(codec, 0x2a, &parm);
3264054d867eSTakashi Iwai 	update_power_state(codec, 0x9, parm);
32653e95b9abSLydia Wang 
32663e95b9abSLydia Wang 	if (spec->hp_independent_mode) {
32673e95b9abSLydia Wang 		/* PW4 (28h), MW3 (1bh), MUX1(34h), AOW4 (ch) */
32683e95b9abSLydia Wang 		parm = AC_PWRST_D3;
32693e95b9abSLydia Wang 		set_pin_power_state(codec, 0x28, &parm);
3270054d867eSTakashi Iwai 		update_power_state(codec, 0x1b, parm);
3271054d867eSTakashi Iwai 		update_power_state(codec, 0x34, parm);
3272054d867eSTakashi Iwai 		update_power_state(codec, 0xc, parm);
32733e95b9abSLydia Wang 	}
32743e95b9abSLydia Wang }
32753e95b9abSLydia Wang 
327630b45033STakashi Iwai /* Add a connection to the primary DAC from AA-mixer for some codecs
327730b45033STakashi Iwai  * This isn't listed from the raw info, but the chip has a secret connection.
327830b45033STakashi Iwai  */
327930b45033STakashi Iwai static int add_secret_dac_path(struct hda_codec *codec)
328030b45033STakashi Iwai {
328130b45033STakashi Iwai 	struct via_spec *spec = codec->spec;
328230b45033STakashi Iwai 	int i, nums;
328330b45033STakashi Iwai 	hda_nid_t conn[8];
328430b45033STakashi Iwai 	hda_nid_t nid;
328530b45033STakashi Iwai 
328630b45033STakashi Iwai 	if (!spec->aa_mix_nid)
328730b45033STakashi Iwai 		return 0;
328830b45033STakashi Iwai 	nums = snd_hda_get_connections(codec, spec->aa_mix_nid, conn,
328930b45033STakashi Iwai 				       ARRAY_SIZE(conn) - 1);
329030b45033STakashi Iwai 	for (i = 0; i < nums; i++) {
329130b45033STakashi Iwai 		if (get_wcaps_type(get_wcaps(codec, conn[i])) == AC_WID_AUD_OUT)
329230b45033STakashi Iwai 			return 0;
329330b45033STakashi Iwai 	}
329430b45033STakashi Iwai 
329530b45033STakashi Iwai 	/* find the primary DAC and add to the connection list */
329630b45033STakashi Iwai 	nid = codec->start_nid;
329730b45033STakashi Iwai 	for (i = 0; i < codec->num_nodes; i++, nid++) {
329830b45033STakashi Iwai 		unsigned int caps = get_wcaps(codec, nid);
329930b45033STakashi Iwai 		if (get_wcaps_type(caps) == AC_WID_AUD_OUT &&
330030b45033STakashi Iwai 		    !(caps & AC_WCAP_DIGITAL)) {
330130b45033STakashi Iwai 			conn[nums++] = nid;
330230b45033STakashi Iwai 			return snd_hda_override_conn_list(codec,
330330b45033STakashi Iwai 							  spec->aa_mix_nid,
330430b45033STakashi Iwai 							  nums, conn);
330530b45033STakashi Iwai 		}
330630b45033STakashi Iwai 	}
330730b45033STakashi Iwai 	return 0;
330830b45033STakashi Iwai }
330930b45033STakashi Iwai 
331030b45033STakashi Iwai 
3311eb7188caSLydia Wang static int patch_vt1718S(struct hda_codec *codec)
3312eb7188caSLydia Wang {
3313eb7188caSLydia Wang 	struct via_spec *spec;
3314eb7188caSLydia Wang 	int err;
3315eb7188caSLydia Wang 
3316eb7188caSLydia Wang 	/* create a codec specific record */
33175b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
3318eb7188caSLydia Wang 	if (spec == NULL)
3319eb7188caSLydia Wang 		return -ENOMEM;
3320eb7188caSLydia Wang 
3321620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x21;
3322d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x2b, 0, 3, 40);
3323d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x29, 0, 3, 40);
332430b45033STakashi Iwai 	add_secret_dac_path(codec);
3325620e2b28STakashi Iwai 
3326eb7188caSLydia Wang 	/* automatic parse from the BIOS config */
332712daef65STakashi Iwai 	err = via_parse_auto_config(codec);
3328eb7188caSLydia Wang 	if (err < 0) {
3329eb7188caSLydia Wang 		via_free(codec);
3330eb7188caSLydia Wang 		return err;
3331eb7188caSLydia Wang 	}
3332eb7188caSLydia Wang 
3333096a8854STakashi Iwai 	spec->init_verbs[spec->num_iverbs++] = vt1718S_init_verbs;
3334eb7188caSLydia Wang 
3335eb7188caSLydia Wang 	codec->patch_ops = via_patch_ops;
3336eb7188caSLydia Wang 
33373e95b9abSLydia Wang 	spec->set_widgets_power_state =  set_widgets_power_state_vt1718S;
33383e95b9abSLydia Wang 
3339eb7188caSLydia Wang 	return 0;
3340eb7188caSLydia Wang }
3341f3db423dSLydia Wang 
3342f3db423dSLydia Wang /* Patch for VT1716S */
3343f3db423dSLydia Wang 
3344f3db423dSLydia Wang static int vt1716s_dmic_info(struct snd_kcontrol *kcontrol,
3345f3db423dSLydia Wang 			    struct snd_ctl_elem_info *uinfo)
3346f3db423dSLydia Wang {
3347f3db423dSLydia Wang 	uinfo->type = SNDRV_CTL_ELEM_TYPE_BOOLEAN;
3348f3db423dSLydia Wang 	uinfo->count = 1;
3349f3db423dSLydia Wang 	uinfo->value.integer.min = 0;
3350f3db423dSLydia Wang 	uinfo->value.integer.max = 1;
3351f3db423dSLydia Wang 	return 0;
3352f3db423dSLydia Wang }
3353f3db423dSLydia Wang 
3354f3db423dSLydia Wang static int vt1716s_dmic_get(struct snd_kcontrol *kcontrol,
3355f3db423dSLydia Wang 			   struct snd_ctl_elem_value *ucontrol)
3356f3db423dSLydia Wang {
3357f3db423dSLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
3358f3db423dSLydia Wang 	int index = 0;
3359f3db423dSLydia Wang 
3360f3db423dSLydia Wang 	index = snd_hda_codec_read(codec, 0x26, 0,
3361f3db423dSLydia Wang 					       AC_VERB_GET_CONNECT_SEL, 0);
3362f3db423dSLydia Wang 	if (index != -1)
3363f3db423dSLydia Wang 		*ucontrol->value.integer.value = index;
3364f3db423dSLydia Wang 
3365f3db423dSLydia Wang 	return 0;
3366f3db423dSLydia Wang }
3367f3db423dSLydia Wang 
3368f3db423dSLydia Wang static int vt1716s_dmic_put(struct snd_kcontrol *kcontrol,
3369f3db423dSLydia Wang 			   struct snd_ctl_elem_value *ucontrol)
3370f3db423dSLydia Wang {
3371f3db423dSLydia Wang 	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
3372f3db423dSLydia Wang 	struct via_spec *spec = codec->spec;
3373f3db423dSLydia Wang 	int index = *ucontrol->value.integer.value;
3374f3db423dSLydia Wang 
3375f3db423dSLydia Wang 	snd_hda_codec_write(codec, 0x26, 0,
3376f3db423dSLydia Wang 					       AC_VERB_SET_CONNECT_SEL, index);
3377f3db423dSLydia Wang 	spec->dmic_enabled = index;
33783e95b9abSLydia Wang 	set_widgets_power_state(codec);
3379f3db423dSLydia Wang 	return 1;
3380f3db423dSLydia Wang }
3381f3db423dSLydia Wang 
338290dd48a1STakashi Iwai static const struct snd_kcontrol_new vt1716s_dmic_mixer[] = {
3383f3db423dSLydia Wang 	HDA_CODEC_VOLUME("Digital Mic Capture Volume", 0x22, 0x0, HDA_INPUT),
3384f3db423dSLydia Wang 	{
3385f3db423dSLydia Wang 	 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
3386f3db423dSLydia Wang 	 .name = "Digital Mic Capture Switch",
33875b0cb1d8SJaroslav Kysela 	 .subdevice = HDA_SUBDEV_NID_FLAG | 0x26,
3388f3db423dSLydia Wang 	 .count = 1,
3389f3db423dSLydia Wang 	 .info = vt1716s_dmic_info,
3390f3db423dSLydia Wang 	 .get = vt1716s_dmic_get,
3391f3db423dSLydia Wang 	 .put = vt1716s_dmic_put,
3392f3db423dSLydia Wang 	 },
3393f3db423dSLydia Wang 	{}			/* end */
3394f3db423dSLydia Wang };
3395f3db423dSLydia Wang 
3396f3db423dSLydia Wang 
3397f3db423dSLydia Wang /* mono-out mixer elements */
339890dd48a1STakashi Iwai static const struct snd_kcontrol_new vt1716S_mono_out_mixer[] = {
3399f3db423dSLydia Wang 	HDA_CODEC_MUTE("Mono Playback Switch", 0x2a, 0x0, HDA_OUTPUT),
3400f3db423dSLydia Wang 	{ } /* end */
3401f3db423dSLydia Wang };
3402f3db423dSLydia Wang 
3403096a8854STakashi Iwai static const struct hda_verb vt1716S_init_verbs[] = {
3404f3db423dSLydia Wang 	/* Enable Boost Volume backdoor */
3405f3db423dSLydia Wang 	{0x1, 0xf8a, 0x80},
3406f3db423dSLydia Wang 	/* don't bybass mixer */
3407f3db423dSLydia Wang 	{0x1, 0xf88, 0xc0},
3408f3db423dSLydia Wang 	/* Enable mono output */
3409f3db423dSLydia Wang 	{0x1, 0xf90, 0x08},
3410f3db423dSLydia Wang 	{ }
3411f3db423dSLydia Wang };
3412f3db423dSLydia Wang 
34133e95b9abSLydia Wang static void set_widgets_power_state_vt1716S(struct hda_codec *codec)
34143e95b9abSLydia Wang {
34153e95b9abSLydia Wang 	struct via_spec *spec = codec->spec;
34163e95b9abSLydia Wang 	int imux_is_smixer;
34173e95b9abSLydia Wang 	unsigned int parm;
34183e95b9abSLydia Wang 	unsigned int mono_out, present;
34193e95b9abSLydia Wang 	/* SW0 (17h) = stereo mixer */
34203e95b9abSLydia Wang 	imux_is_smixer =
34213e95b9abSLydia Wang 	(snd_hda_codec_read(codec, 0x17, 0,
34223e95b9abSLydia Wang 			    AC_VERB_GET_CONNECT_SEL, 0x00) ==  5);
34233e95b9abSLydia Wang 	/* inputs */
34243e95b9abSLydia Wang 	/* PW 1/2/5 (1ah/1bh/1eh) */
34253e95b9abSLydia Wang 	parm = AC_PWRST_D3;
34263e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1a, &parm);
34273e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1b, &parm);
34283e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1e, &parm);
34293e95b9abSLydia Wang 	if (imux_is_smixer)
34303e95b9abSLydia Wang 		parm = AC_PWRST_D0;
34313e95b9abSLydia Wang 	/* SW0 (17h), AIW0(13h) */
3432054d867eSTakashi Iwai 	update_power_state(codec, 0x17, parm);
3433054d867eSTakashi Iwai 	update_power_state(codec, 0x13, parm);
34343e95b9abSLydia Wang 
34353e95b9abSLydia Wang 	parm = AC_PWRST_D3;
34363e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1e, &parm);
34373e95b9abSLydia Wang 	/* PW11 (22h) */
34383e95b9abSLydia Wang 	if (spec->dmic_enabled)
34393e95b9abSLydia Wang 		set_pin_power_state(codec, 0x22, &parm);
34403e95b9abSLydia Wang 	else
3441054d867eSTakashi Iwai 		update_power_state(codec, 0x22, AC_PWRST_D3);
34423e95b9abSLydia Wang 
34433e95b9abSLydia Wang 	/* SW2(26h), AIW1(14h) */
3444054d867eSTakashi Iwai 	update_power_state(codec, 0x26, parm);
3445054d867eSTakashi Iwai 	update_power_state(codec, 0x14, parm);
34463e95b9abSLydia Wang 
34473e95b9abSLydia Wang 	/* outputs */
34483e95b9abSLydia Wang 	/* PW0 (19h), SW1 (18h), AOW1 (11h) */
34493e95b9abSLydia Wang 	parm = AC_PWRST_D3;
34503e95b9abSLydia Wang 	set_pin_power_state(codec, 0x19, &parm);
34513e95b9abSLydia Wang 	/* Smart 5.1 PW2(1bh) */
34523e95b9abSLydia Wang 	if (spec->smart51_enabled)
34533e95b9abSLydia Wang 		set_pin_power_state(codec, 0x1b, &parm);
3454054d867eSTakashi Iwai 	update_power_state(codec, 0x18, parm);
3455054d867eSTakashi Iwai 	update_power_state(codec, 0x11, parm);
34563e95b9abSLydia Wang 
34573e95b9abSLydia Wang 	/* PW7 (23h), SW3 (27h), AOW3 (25h) */
34583e95b9abSLydia Wang 	parm = AC_PWRST_D3;
34593e95b9abSLydia Wang 	set_pin_power_state(codec, 0x23, &parm);
34603e95b9abSLydia Wang 	/* Smart 5.1 PW1(1ah) */
34613e95b9abSLydia Wang 	if (spec->smart51_enabled)
34623e95b9abSLydia Wang 		set_pin_power_state(codec, 0x1a, &parm);
3463054d867eSTakashi Iwai 	update_power_state(codec, 0x27, parm);
34643e95b9abSLydia Wang 
34653e95b9abSLydia Wang 	/* Smart 5.1 PW5(1eh) */
34663e95b9abSLydia Wang 	if (spec->smart51_enabled)
34673e95b9abSLydia Wang 		set_pin_power_state(codec, 0x1e, &parm);
3468054d867eSTakashi Iwai 	update_power_state(codec, 0x25, parm);
34693e95b9abSLydia Wang 
34703e95b9abSLydia Wang 	/* Mono out */
34713e95b9abSLydia Wang 	/* SW4(28h)->MW1(29h)-> PW12 (2ah)*/
34723e95b9abSLydia Wang 	present = snd_hda_jack_detect(codec, 0x1c);
34733e95b9abSLydia Wang 
34743e95b9abSLydia Wang 	if (present)
34753e95b9abSLydia Wang 		mono_out = 0;
34763e95b9abSLydia Wang 	else {
34773e95b9abSLydia Wang 		present = snd_hda_jack_detect(codec, 0x1d);
34783e95b9abSLydia Wang 		if (!spec->hp_independent_mode && present)
34793e95b9abSLydia Wang 			mono_out = 0;
34803e95b9abSLydia Wang 		else
34813e95b9abSLydia Wang 			mono_out = 1;
34823e95b9abSLydia Wang 	}
34833e95b9abSLydia Wang 	parm = mono_out ? AC_PWRST_D0 : AC_PWRST_D3;
3484054d867eSTakashi Iwai 	update_power_state(codec, 0x28, parm);
3485054d867eSTakashi Iwai 	update_power_state(codec, 0x29, parm);
3486054d867eSTakashi Iwai 	update_power_state(codec, 0x2a, parm);
34873e95b9abSLydia Wang 
34883e95b9abSLydia Wang 	/* PW 3/4 (1ch/1dh) */
34893e95b9abSLydia Wang 	parm = AC_PWRST_D3;
34903e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1c, &parm);
34913e95b9abSLydia Wang 	set_pin_power_state(codec, 0x1d, &parm);
34923e95b9abSLydia Wang 	/* HP Independent Mode, power on AOW3 */
34933e95b9abSLydia Wang 	if (spec->hp_independent_mode)
3494054d867eSTakashi Iwai 		update_power_state(codec, 0x25, parm);
34953e95b9abSLydia Wang 
34963e95b9abSLydia Wang 	/* force to D0 for internal Speaker */
34973e95b9abSLydia Wang 	/* MW0 (16h), AOW0 (10h) */
3498054d867eSTakashi Iwai 	update_power_state(codec, 0x16, imux_is_smixer ? AC_PWRST_D0 : parm);
3499054d867eSTakashi Iwai 	update_power_state(codec, 0x10, mono_out ? AC_PWRST_D0 : parm);
35003e95b9abSLydia Wang }
35013e95b9abSLydia Wang 
3502f3db423dSLydia Wang static int patch_vt1716S(struct hda_codec *codec)
3503f3db423dSLydia Wang {
3504f3db423dSLydia Wang 	struct via_spec *spec;
3505f3db423dSLydia Wang 	int err;
3506f3db423dSLydia Wang 
3507f3db423dSLydia Wang 	/* create a codec specific record */
35085b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
3509f3db423dSLydia Wang 	if (spec == NULL)
3510f3db423dSLydia Wang 		return -ENOMEM;
3511f3db423dSLydia Wang 
3512620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x16;
3513d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x1a, 0, 3, 40);
3514d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x1e, 0, 3, 40);
3515620e2b28STakashi Iwai 
3516f3db423dSLydia Wang 	/* automatic parse from the BIOS config */
351712daef65STakashi Iwai 	err = via_parse_auto_config(codec);
3518f3db423dSLydia Wang 	if (err < 0) {
3519f3db423dSLydia Wang 		via_free(codec);
3520f3db423dSLydia Wang 		return err;
3521f3db423dSLydia Wang 	}
3522f3db423dSLydia Wang 
3523096a8854STakashi Iwai 	spec->init_verbs[spec->num_iverbs++]  = vt1716S_init_verbs;
3524f3db423dSLydia Wang 
3525f3db423dSLydia Wang 	spec->mixers[spec->num_mixers] = vt1716s_dmic_mixer;
3526f3db423dSLydia Wang 	spec->num_mixers++;
3527f3db423dSLydia Wang 
3528f3db423dSLydia Wang 	spec->mixers[spec->num_mixers++] = vt1716S_mono_out_mixer;
3529f3db423dSLydia Wang 
3530f3db423dSLydia Wang 	codec->patch_ops = via_patch_ops;
3531f3db423dSLydia Wang 
35323e95b9abSLydia Wang 	spec->set_widgets_power_state = set_widgets_power_state_vt1716S;
3533f3db423dSLydia Wang 	return 0;
3534f3db423dSLydia Wang }
353525eaba2fSLydia Wang 
353625eaba2fSLydia Wang /* for vt2002P */
353725eaba2fSLydia Wang 
3538096a8854STakashi Iwai static const struct hda_verb vt2002P_init_verbs[] = {
3539eadb9a80SLydia Wang 	/* Class-D speaker related verbs */
3540eadb9a80SLydia Wang 	{0x1, 0xfe0, 0x4},
3541eadb9a80SLydia Wang 	{0x1, 0xfe9, 0x80},
3542eadb9a80SLydia Wang 	{0x1, 0xfe2, 0x22},
354325eaba2fSLydia Wang 	/* Enable Boost Volume backdoor */
354425eaba2fSLydia Wang 	{0x1, 0xfb9, 0x24},
354525eaba2fSLydia Wang 	/* Enable AOW0 to MW9 */
354625eaba2fSLydia Wang 	{0x1, 0xfb8, 0x88},
354725eaba2fSLydia Wang 	{ }
354825eaba2fSLydia Wang };
35494a918ffeSTakashi Iwai 
3550096a8854STakashi Iwai static const struct hda_verb vt1802_init_verbs[] = {
355111890956SLydia Wang 	/* Enable Boost Volume backdoor */
355211890956SLydia Wang 	{0x1, 0xfb9, 0x24},
355311890956SLydia Wang 	/* Enable AOW0 to MW9 */
355411890956SLydia Wang 	{0x1, 0xfb8, 0x88},
355511890956SLydia Wang 	{ }
355611890956SLydia Wang };
355725eaba2fSLydia Wang 
35583e95b9abSLydia Wang static void set_widgets_power_state_vt2002P(struct hda_codec *codec)
35593e95b9abSLydia Wang {
35603e95b9abSLydia Wang 	struct via_spec *spec = codec->spec;
35613e95b9abSLydia Wang 	int imux_is_smixer;
35623e95b9abSLydia Wang 	unsigned int parm;
35633e95b9abSLydia Wang 	unsigned int present;
35643e95b9abSLydia Wang 	/* MUX9 (1eh) = stereo mixer */
35653e95b9abSLydia Wang 	imux_is_smixer =
35663e95b9abSLydia Wang 	snd_hda_codec_read(codec, 0x1e, 0, AC_VERB_GET_CONNECT_SEL, 0x00) == 3;
35673e95b9abSLydia Wang 	/* inputs */
35683e95b9abSLydia Wang 	/* PW 5/6/7 (29h/2ah/2bh) */
35693e95b9abSLydia Wang 	parm = AC_PWRST_D3;
35703e95b9abSLydia Wang 	set_pin_power_state(codec, 0x29, &parm);
35713e95b9abSLydia Wang 	set_pin_power_state(codec, 0x2a, &parm);
35723e95b9abSLydia Wang 	set_pin_power_state(codec, 0x2b, &parm);
35733e95b9abSLydia Wang 	parm = AC_PWRST_D0;
35743e95b9abSLydia Wang 	/* MUX9/10 (1eh/1fh), AIW 0/1 (10h/11h) */
3575054d867eSTakashi Iwai 	update_power_state(codec, 0x1e, parm);
3576054d867eSTakashi Iwai 	update_power_state(codec, 0x1f, parm);
3577054d867eSTakashi Iwai 	update_power_state(codec, 0x10, parm);
3578054d867eSTakashi Iwai 	update_power_state(codec, 0x11, parm);
35793e95b9abSLydia Wang 
35803e95b9abSLydia Wang 	/* outputs */
35813e95b9abSLydia Wang 	/* AOW0 (8h)*/
3582054d867eSTakashi Iwai 	update_power_state(codec, 0x8, parm);
35833e95b9abSLydia Wang 
358411890956SLydia Wang 	if (spec->codec_type == VT1802) {
358511890956SLydia Wang 		/* PW4 (28h), MW4 (18h), MUX4(38h) */
358611890956SLydia Wang 		parm = AC_PWRST_D3;
358711890956SLydia Wang 		set_pin_power_state(codec, 0x28, &parm);
3588054d867eSTakashi Iwai 		update_power_state(codec, 0x18, parm);
3589054d867eSTakashi Iwai 		update_power_state(codec, 0x38, parm);
359011890956SLydia Wang 	} else {
35913e95b9abSLydia Wang 		/* PW4 (26h), MW4 (1ch), MUX4(37h) */
35923e95b9abSLydia Wang 		parm = AC_PWRST_D3;
35933e95b9abSLydia Wang 		set_pin_power_state(codec, 0x26, &parm);
3594054d867eSTakashi Iwai 		update_power_state(codec, 0x1c, parm);
3595054d867eSTakashi Iwai 		update_power_state(codec, 0x37, parm);
359611890956SLydia Wang 	}
35973e95b9abSLydia Wang 
359811890956SLydia Wang 	if (spec->codec_type == VT1802) {
359911890956SLydia Wang 		/* PW1 (25h), MW1 (15h), MUX1(35h), AOW1 (9h) */
360011890956SLydia Wang 		parm = AC_PWRST_D3;
360111890956SLydia Wang 		set_pin_power_state(codec, 0x25, &parm);
3602054d867eSTakashi Iwai 		update_power_state(codec, 0x15, parm);
3603054d867eSTakashi Iwai 		update_power_state(codec, 0x35, parm);
360411890956SLydia Wang 	} else {
36053e95b9abSLydia Wang 		/* PW1 (25h), MW1 (19h), MUX1(35h), AOW1 (9h) */
36063e95b9abSLydia Wang 		parm = AC_PWRST_D3;
36073e95b9abSLydia Wang 		set_pin_power_state(codec, 0x25, &parm);
3608054d867eSTakashi Iwai 		update_power_state(codec, 0x19, parm);
3609054d867eSTakashi Iwai 		update_power_state(codec, 0x35, parm);
361011890956SLydia Wang 	}
36113e95b9abSLydia Wang 
36123e95b9abSLydia Wang 	if (spec->hp_independent_mode)
3613054d867eSTakashi Iwai 		update_power_state(codec, 0x9, AC_PWRST_D0);
36143e95b9abSLydia Wang 
36153e95b9abSLydia Wang 	/* Class-D */
36163e95b9abSLydia Wang 	/* PW0 (24h), MW0(18h/14h), MUX0(34h) */
36173e95b9abSLydia Wang 	present = snd_hda_jack_detect(codec, 0x25);
36183e95b9abSLydia Wang 
36193e95b9abSLydia Wang 	parm = AC_PWRST_D3;
36203e95b9abSLydia Wang 	set_pin_power_state(codec, 0x24, &parm);
36213e95b9abSLydia Wang 	parm = present ? AC_PWRST_D3 : AC_PWRST_D0;
362211890956SLydia Wang 	if (spec->codec_type == VT1802)
3623054d867eSTakashi Iwai 		update_power_state(codec, 0x14, parm);
362411890956SLydia Wang 	else
3625054d867eSTakashi Iwai 		update_power_state(codec, 0x18, parm);
3626054d867eSTakashi Iwai 	update_power_state(codec, 0x34, parm);
36273e95b9abSLydia Wang 
36283e95b9abSLydia Wang 	/* Mono Out */
36293e95b9abSLydia Wang 	present = snd_hda_jack_detect(codec, 0x26);
36303e95b9abSLydia Wang 
36313e95b9abSLydia Wang 	parm = present ? AC_PWRST_D3 : AC_PWRST_D0;
363211890956SLydia Wang 	if (spec->codec_type == VT1802) {
363311890956SLydia Wang 		/* PW15 (33h), MW8(1ch), MUX8(3ch) */
3634054d867eSTakashi Iwai 		update_power_state(codec, 0x33, parm);
3635054d867eSTakashi Iwai 		update_power_state(codec, 0x1c, parm);
3636054d867eSTakashi Iwai 		update_power_state(codec, 0x3c, parm);
363711890956SLydia Wang 	} else {
36383e95b9abSLydia Wang 		/* PW15 (31h), MW8(17h), MUX8(3bh) */
3639054d867eSTakashi Iwai 		update_power_state(codec, 0x31, parm);
3640054d867eSTakashi Iwai 		update_power_state(codec, 0x17, parm);
3641054d867eSTakashi Iwai 		update_power_state(codec, 0x3b, parm);
364211890956SLydia Wang 	}
36433e95b9abSLydia Wang 	/* MW9 (21h) */
36443e95b9abSLydia Wang 	if (imux_is_smixer || !is_aa_path_mute(codec))
3645054d867eSTakashi Iwai 		update_power_state(codec, 0x21, AC_PWRST_D0);
36463e95b9abSLydia Wang 	else
3647054d867eSTakashi Iwai 		update_power_state(codec, 0x21, AC_PWRST_D3);
36483e95b9abSLydia Wang }
364925eaba2fSLydia Wang 
36504b527b65SDavid Henningsson /*
36514b527b65SDavid Henningsson  * pin fix-up
36524b527b65SDavid Henningsson  */
36534b527b65SDavid Henningsson enum {
36544b527b65SDavid Henningsson 	VIA_FIXUP_INTMIC_BOOST,
3655d5266125STakashi Iwai 	VIA_FIXUP_ASUS_G75,
36564b527b65SDavid Henningsson };
36574b527b65SDavid Henningsson 
36584b527b65SDavid Henningsson static void via_fixup_intmic_boost(struct hda_codec *codec,
36594b527b65SDavid Henningsson 				  const struct hda_fixup *fix, int action)
36604b527b65SDavid Henningsson {
36614b527b65SDavid Henningsson 	if (action == HDA_FIXUP_ACT_PRE_PROBE)
36624b527b65SDavid Henningsson 		override_mic_boost(codec, 0x30, 0, 2, 40);
36634b527b65SDavid Henningsson }
36644b527b65SDavid Henningsson 
36654b527b65SDavid Henningsson static const struct hda_fixup via_fixups[] = {
36664b527b65SDavid Henningsson 	[VIA_FIXUP_INTMIC_BOOST] = {
36674b527b65SDavid Henningsson 		.type = HDA_FIXUP_FUNC,
36684b527b65SDavid Henningsson 		.v.func = via_fixup_intmic_boost,
36694b527b65SDavid Henningsson 	},
3670d5266125STakashi Iwai 	[VIA_FIXUP_ASUS_G75] = {
3671d5266125STakashi Iwai 		.type = HDA_FIXUP_PINS,
3672d5266125STakashi Iwai 		.v.pins = (const struct hda_pintbl[]) {
3673d5266125STakashi Iwai 			/* set 0x24 and 0x33 as speakers */
3674d5266125STakashi Iwai 			{ 0x24, 0x991301f0 },
3675d5266125STakashi Iwai 			{ 0x33, 0x991301f1 }, /* subwoofer */
3676d5266125STakashi Iwai 			{ }
3677d5266125STakashi Iwai 		}
3678d5266125STakashi Iwai 	},
36794b527b65SDavid Henningsson };
36804b527b65SDavid Henningsson 
36814b527b65SDavid Henningsson static const struct snd_pci_quirk vt2002p_fixups[] = {
3682d5266125STakashi Iwai 	SND_PCI_QUIRK(0x1043, 0x1487, "Asus G75", VIA_FIXUP_ASUS_G75),
36834b527b65SDavid Henningsson 	SND_PCI_QUIRK(0x1043, 0x8532, "Asus X202E", VIA_FIXUP_INTMIC_BOOST),
36844b527b65SDavid Henningsson 	{}
36854b527b65SDavid Henningsson };
36864b527b65SDavid Henningsson 
3687ef4da458STakashi Iwai /* NIDs 0x24 and 0x33 on VT1802 have connections to non-existing NID 0x3e
3688ef4da458STakashi Iwai  * Replace this with mixer NID 0x1c
3689ef4da458STakashi Iwai  */
3690ef4da458STakashi Iwai static void fix_vt1802_connections(struct hda_codec *codec)
3691ef4da458STakashi Iwai {
3692ef4da458STakashi Iwai 	static hda_nid_t conn_24[] = { 0x14, 0x1c };
3693ef4da458STakashi Iwai 	static hda_nid_t conn_33[] = { 0x1c };
3694ef4da458STakashi Iwai 
3695ef4da458STakashi Iwai 	snd_hda_override_conn_list(codec, 0x24, ARRAY_SIZE(conn_24), conn_24);
3696ef4da458STakashi Iwai 	snd_hda_override_conn_list(codec, 0x33, ARRAY_SIZE(conn_33), conn_33);
3697ef4da458STakashi Iwai }
3698ef4da458STakashi Iwai 
369925eaba2fSLydia Wang /* patch for vt2002P */
370025eaba2fSLydia Wang static int patch_vt2002P(struct hda_codec *codec)
370125eaba2fSLydia Wang {
370225eaba2fSLydia Wang 	struct via_spec *spec;
370325eaba2fSLydia Wang 	int err;
370425eaba2fSLydia Wang 
370525eaba2fSLydia Wang 	/* create a codec specific record */
37065b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
370725eaba2fSLydia Wang 	if (spec == NULL)
370825eaba2fSLydia Wang 		return -ENOMEM;
370925eaba2fSLydia Wang 
3710620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x21;
3711d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x2b, 0, 3, 40);
3712d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x29, 0, 3, 40);
3713ef4da458STakashi Iwai 	if (spec->codec_type == VT1802)
3714ef4da458STakashi Iwai 		fix_vt1802_connections(codec);
371530b45033STakashi Iwai 	add_secret_dac_path(codec);
3716620e2b28STakashi Iwai 
37174b527b65SDavid Henningsson 	snd_hda_pick_fixup(codec, NULL, vt2002p_fixups, via_fixups);
37184b527b65SDavid Henningsson 	snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
37194b527b65SDavid Henningsson 
372025eaba2fSLydia Wang 	/* automatic parse from the BIOS config */
372112daef65STakashi Iwai 	err = via_parse_auto_config(codec);
372225eaba2fSLydia Wang 	if (err < 0) {
372325eaba2fSLydia Wang 		via_free(codec);
372425eaba2fSLydia Wang 		return err;
372525eaba2fSLydia Wang 	}
372625eaba2fSLydia Wang 
372711890956SLydia Wang 	if (spec->codec_type == VT1802)
37284a918ffeSTakashi Iwai 		spec->init_verbs[spec->num_iverbs++] = vt1802_init_verbs;
372911890956SLydia Wang 	else
37304a918ffeSTakashi Iwai 		spec->init_verbs[spec->num_iverbs++] = vt2002P_init_verbs;
373111890956SLydia Wang 
373225eaba2fSLydia Wang 	codec->patch_ops = via_patch_ops;
373325eaba2fSLydia Wang 
37343e95b9abSLydia Wang 	spec->set_widgets_power_state =  set_widgets_power_state_vt2002P;
373525eaba2fSLydia Wang 	return 0;
373625eaba2fSLydia Wang }
3737ab6734e7SLydia Wang 
3738ab6734e7SLydia Wang /* for vt1812 */
3739ab6734e7SLydia Wang 
3740096a8854STakashi Iwai static const struct hda_verb vt1812_init_verbs[] = {
3741ab6734e7SLydia Wang 	/* Enable Boost Volume backdoor */
3742ab6734e7SLydia Wang 	{0x1, 0xfb9, 0x24},
3743ab6734e7SLydia Wang 	/* Enable AOW0 to MW9 */
3744ab6734e7SLydia Wang 	{0x1, 0xfb8, 0xa8},
3745ab6734e7SLydia Wang 	{ }
3746ab6734e7SLydia Wang };
3747ab6734e7SLydia Wang 
37483e95b9abSLydia Wang static void set_widgets_power_state_vt1812(struct hda_codec *codec)
37493e95b9abSLydia Wang {
37503e95b9abSLydia Wang 	struct via_spec *spec = codec->spec;
37513e95b9abSLydia Wang 	unsigned int parm;
37523e95b9abSLydia Wang 	unsigned int present;
37533e95b9abSLydia Wang 	/* inputs */
37543e95b9abSLydia Wang 	/* PW 5/6/7 (29h/2ah/2bh) */
37553e95b9abSLydia Wang 	parm = AC_PWRST_D3;
37563e95b9abSLydia Wang 	set_pin_power_state(codec, 0x29, &parm);
37573e95b9abSLydia Wang 	set_pin_power_state(codec, 0x2a, &parm);
37583e95b9abSLydia Wang 	set_pin_power_state(codec, 0x2b, &parm);
37593e95b9abSLydia Wang 	parm = AC_PWRST_D0;
37603e95b9abSLydia Wang 	/* MUX10/11 (1eh/1fh), AIW 0/1 (10h/11h) */
3761054d867eSTakashi Iwai 	update_power_state(codec, 0x1e, parm);
3762054d867eSTakashi Iwai 	update_power_state(codec, 0x1f, parm);
3763054d867eSTakashi Iwai 	update_power_state(codec, 0x10, parm);
3764054d867eSTakashi Iwai 	update_power_state(codec, 0x11, parm);
37653e95b9abSLydia Wang 
37663e95b9abSLydia Wang 	/* outputs */
37673e95b9abSLydia Wang 	/* AOW0 (8h)*/
3768054d867eSTakashi Iwai 	update_power_state(codec, 0x8, AC_PWRST_D0);
37693e95b9abSLydia Wang 
37703e95b9abSLydia Wang 	/* PW4 (28h), MW4 (18h), MUX4(38h) */
37713e95b9abSLydia Wang 	parm = AC_PWRST_D3;
37723e95b9abSLydia Wang 	set_pin_power_state(codec, 0x28, &parm);
3773054d867eSTakashi Iwai 	update_power_state(codec, 0x18, parm);
3774054d867eSTakashi Iwai 	update_power_state(codec, 0x38, parm);
37753e95b9abSLydia Wang 
37763e95b9abSLydia Wang 	/* PW1 (25h), MW1 (15h), MUX1(35h), AOW1 (9h) */
37773e95b9abSLydia Wang 	parm = AC_PWRST_D3;
37783e95b9abSLydia Wang 	set_pin_power_state(codec, 0x25, &parm);
3779054d867eSTakashi Iwai 	update_power_state(codec, 0x15, parm);
3780054d867eSTakashi Iwai 	update_power_state(codec, 0x35, parm);
37813e95b9abSLydia Wang 	if (spec->hp_independent_mode)
3782054d867eSTakashi Iwai 		update_power_state(codec, 0x9, AC_PWRST_D0);
37833e95b9abSLydia Wang 
37843e95b9abSLydia Wang 	/* Internal Speaker */
37853e95b9abSLydia Wang 	/* PW0 (24h), MW0(14h), MUX0(34h) */
37863e95b9abSLydia Wang 	present = snd_hda_jack_detect(codec, 0x25);
37873e95b9abSLydia Wang 
37883e95b9abSLydia Wang 	parm = AC_PWRST_D3;
37893e95b9abSLydia Wang 	set_pin_power_state(codec, 0x24, &parm);
37903e95b9abSLydia Wang 	if (present) {
3791054d867eSTakashi Iwai 		update_power_state(codec, 0x14, AC_PWRST_D3);
3792054d867eSTakashi Iwai 		update_power_state(codec, 0x34, AC_PWRST_D3);
37933e95b9abSLydia Wang 	} else {
3794054d867eSTakashi Iwai 		update_power_state(codec, 0x14, AC_PWRST_D0);
3795054d867eSTakashi Iwai 		update_power_state(codec, 0x34, AC_PWRST_D0);
37963e95b9abSLydia Wang 	}
37973e95b9abSLydia Wang 
37983e95b9abSLydia Wang 
37993e95b9abSLydia Wang 	/* Mono Out */
38003e95b9abSLydia Wang 	/* PW13 (31h), MW13(1ch), MUX13(3ch), MW14(3eh) */
38013e95b9abSLydia Wang 	present = snd_hda_jack_detect(codec, 0x28);
38023e95b9abSLydia Wang 
38033e95b9abSLydia Wang 	parm = AC_PWRST_D3;
38043e95b9abSLydia Wang 	set_pin_power_state(codec, 0x31, &parm);
38053e95b9abSLydia Wang 	if (present) {
3806054d867eSTakashi Iwai 		update_power_state(codec, 0x1c, AC_PWRST_D3);
3807054d867eSTakashi Iwai 		update_power_state(codec, 0x3c, AC_PWRST_D3);
3808054d867eSTakashi Iwai 		update_power_state(codec, 0x3e, AC_PWRST_D3);
38093e95b9abSLydia Wang 	} else {
3810054d867eSTakashi Iwai 		update_power_state(codec, 0x1c, AC_PWRST_D0);
3811054d867eSTakashi Iwai 		update_power_state(codec, 0x3c, AC_PWRST_D0);
3812054d867eSTakashi Iwai 		update_power_state(codec, 0x3e, AC_PWRST_D0);
38133e95b9abSLydia Wang 	}
38143e95b9abSLydia Wang 
38153e95b9abSLydia Wang 	/* PW15 (33h), MW15 (1dh), MUX15(3dh) */
38163e95b9abSLydia Wang 	parm = AC_PWRST_D3;
38173e95b9abSLydia Wang 	set_pin_power_state(codec, 0x33, &parm);
3818054d867eSTakashi Iwai 	update_power_state(codec, 0x1d, parm);
3819054d867eSTakashi Iwai 	update_power_state(codec, 0x3d, parm);
38203e95b9abSLydia Wang 
38213e95b9abSLydia Wang }
3822ab6734e7SLydia Wang 
3823ab6734e7SLydia Wang /* patch for vt1812 */
3824ab6734e7SLydia Wang static int patch_vt1812(struct hda_codec *codec)
3825ab6734e7SLydia Wang {
3826ab6734e7SLydia Wang 	struct via_spec *spec;
3827ab6734e7SLydia Wang 	int err;
3828ab6734e7SLydia Wang 
3829ab6734e7SLydia Wang 	/* create a codec specific record */
38305b0cb1d8SJaroslav Kysela 	spec = via_new_spec(codec);
3831ab6734e7SLydia Wang 	if (spec == NULL)
3832ab6734e7SLydia Wang 		return -ENOMEM;
3833ab6734e7SLydia Wang 
3834620e2b28STakashi Iwai 	spec->aa_mix_nid = 0x21;
3835d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x2b, 0, 3, 40);
3836d7a99cceSTakashi Iwai 	override_mic_boost(codec, 0x29, 0, 3, 40);
383730b45033STakashi Iwai 	add_secret_dac_path(codec);
3838620e2b28STakashi Iwai 
3839ab6734e7SLydia Wang 	/* automatic parse from the BIOS config */
384012daef65STakashi Iwai 	err = via_parse_auto_config(codec);
3841ab6734e7SLydia Wang 	if (err < 0) {
3842ab6734e7SLydia Wang 		via_free(codec);
3843ab6734e7SLydia Wang 		return err;
3844ab6734e7SLydia Wang 	}
3845ab6734e7SLydia Wang 
3846096a8854STakashi Iwai 	spec->init_verbs[spec->num_iverbs++]  = vt1812_init_verbs;
3847ab6734e7SLydia Wang 
3848ab6734e7SLydia Wang 	codec->patch_ops = via_patch_ops;
3849ab6734e7SLydia Wang 
38503e95b9abSLydia Wang 	spec->set_widgets_power_state =  set_widgets_power_state_vt1812;
3851ab6734e7SLydia Wang 	return 0;
3852ab6734e7SLydia Wang }
3853ab6734e7SLydia Wang 
385443737e0aSLydia Wang /* patch for vt3476 */
385543737e0aSLydia Wang 
385643737e0aSLydia Wang static const struct hda_verb vt3476_init_verbs[] = {
385743737e0aSLydia Wang 	/* Enable DMic 8/16/32K */
385843737e0aSLydia Wang 	{0x1, 0xF7B, 0x30},
385943737e0aSLydia Wang 	/* Enable Boost Volume backdoor */
386043737e0aSLydia Wang 	{0x1, 0xFB9, 0x20},
386143737e0aSLydia Wang 	/* Enable AOW-MW9 path */
386243737e0aSLydia Wang 	{0x1, 0xFB8, 0x10},
386343737e0aSLydia Wang 	{ }
386443737e0aSLydia Wang };
386543737e0aSLydia Wang 
386643737e0aSLydia Wang static void set_widgets_power_state_vt3476(struct hda_codec *codec)
386743737e0aSLydia Wang {
386843737e0aSLydia Wang 	struct via_spec *spec = codec->spec;
386943737e0aSLydia Wang 	int imux_is_smixer;
387043737e0aSLydia Wang 	unsigned int parm, parm2;
387143737e0aSLydia Wang 	/* MUX10 (1eh) = stereo mixer */
387243737e0aSLydia Wang 	imux_is_smixer =
387343737e0aSLydia Wang 	snd_hda_codec_read(codec, 0x1e, 0, AC_VERB_GET_CONNECT_SEL, 0x00) == 4;
387443737e0aSLydia Wang 	/* inputs */
387543737e0aSLydia Wang 	/* PW 5/6/7 (29h/2ah/2bh) */
387643737e0aSLydia Wang 	parm = AC_PWRST_D3;
387743737e0aSLydia Wang 	set_pin_power_state(codec, 0x29, &parm);
387843737e0aSLydia Wang 	set_pin_power_state(codec, 0x2a, &parm);
387943737e0aSLydia Wang 	set_pin_power_state(codec, 0x2b, &parm);
388043737e0aSLydia Wang 	if (imux_is_smixer)
388143737e0aSLydia Wang 		parm = AC_PWRST_D0;
388243737e0aSLydia Wang 	/* MUX10/11 (1eh/1fh), AIW 0/1 (10h/11h) */
388343737e0aSLydia Wang 	update_power_state(codec, 0x1e, parm);
388443737e0aSLydia Wang 	update_power_state(codec, 0x1f, parm);
388543737e0aSLydia Wang 	update_power_state(codec, 0x10, parm);
388643737e0aSLydia Wang 	update_power_state(codec, 0x11, parm);
388743737e0aSLydia Wang 
388843737e0aSLydia Wang 	/* outputs */
388943737e0aSLydia Wang 	/* PW3 (27h), MW3(37h), AOW3 (bh) */
389043737e0aSLydia Wang 	if (spec->codec_type == VT1705CF) {
389143737e0aSLydia Wang 		parm = AC_PWRST_D3;
389243737e0aSLydia Wang 		update_power_state(codec, 0x27, parm);
389343737e0aSLydia Wang 		update_power_state(codec, 0x37, parm);
389443737e0aSLydia Wang 	}	else {
389543737e0aSLydia Wang 		parm = AC_PWRST_D3;
389643737e0aSLydia Wang 		set_pin_power_state(codec, 0x27, &parm);
389743737e0aSLydia Wang 		update_power_state(codec, 0x37, parm);
389843737e0aSLydia Wang 	}
389943737e0aSLydia Wang 
390043737e0aSLydia Wang 	/* PW2 (26h), MW2(36h), AOW2 (ah) */
390143737e0aSLydia Wang 	parm = AC_PWRST_D3;
390243737e0aSLydia Wang 	set_pin_power_state(codec, 0x26, &parm);
390343737e0aSLydia Wang 	update_power_state(codec, 0x36, parm);
390443737e0aSLydia Wang 	if (spec->smart51_enabled) {
390543737e0aSLydia Wang 		/* PW7(2bh), MW7(3bh), MUX7(1Bh) */
390643737e0aSLydia Wang 		set_pin_power_state(codec, 0x2b, &parm);
390743737e0aSLydia Wang 		update_power_state(codec, 0x3b, parm);
390843737e0aSLydia Wang 		update_power_state(codec, 0x1b, parm);
390943737e0aSLydia Wang 	}
391043737e0aSLydia Wang 	update_conv_power_state(codec, 0xa, parm, 2);
391143737e0aSLydia Wang 
391243737e0aSLydia Wang 	/* PW1 (25h), MW1(35h), AOW1 (9h) */
391343737e0aSLydia Wang 	parm = AC_PWRST_D3;
391443737e0aSLydia Wang 	set_pin_power_state(codec, 0x25, &parm);
391543737e0aSLydia Wang 	update_power_state(codec, 0x35, parm);
391643737e0aSLydia Wang 	if (spec->smart51_enabled) {
391743737e0aSLydia Wang 		/* PW6(2ah), MW6(3ah), MUX6(1ah) */
391843737e0aSLydia Wang 		set_pin_power_state(codec, 0x2a, &parm);
391943737e0aSLydia Wang 		update_power_state(codec, 0x3a, parm);
392043737e0aSLydia Wang 		update_power_state(codec, 0x1a, parm);
392143737e0aSLydia Wang 	}
392243737e0aSLydia Wang 	update_conv_power_state(codec, 0x9, parm, 1);
392343737e0aSLydia Wang 
392443737e0aSLydia Wang 	/* PW4 (28h), MW4 (38h), MUX4(18h), AOW3(bh)/AOW0(8h) */
392543737e0aSLydia Wang 	parm = AC_PWRST_D3;
392643737e0aSLydia Wang 	set_pin_power_state(codec, 0x28, &parm);
392743737e0aSLydia Wang 	update_power_state(codec, 0x38, parm);
392843737e0aSLydia Wang 	update_power_state(codec, 0x18, parm);
392943737e0aSLydia Wang 	if (spec->hp_independent_mode)
393043737e0aSLydia Wang 		update_conv_power_state(codec, 0xb, parm, 3);
393143737e0aSLydia Wang 	parm2 = parm; /* for pin 0x0b */
393243737e0aSLydia Wang 
393343737e0aSLydia Wang 	/* PW0 (24h), MW0(34h), MW9(3fh), AOW0 (8h) */
393443737e0aSLydia Wang 	parm = AC_PWRST_D3;
393543737e0aSLydia Wang 	set_pin_power_state(codec, 0x24, &parm);
393643737e0aSLydia Wang 	update_power_state(codec, 0x34, parm);
393743737e0aSLydia Wang 	if (!spec->hp_independent_mode && parm2 != AC_PWRST_D3)
393843737e0aSLydia Wang 		parm = parm2;
393943737e0aSLydia Wang 	update_conv_power_state(codec, 0x8, parm, 0);
394043737e0aSLydia Wang 	/* MW9 (21h), Mw2 (1ah), AOW0 (8h) */
394143737e0aSLydia Wang 	update_power_state(codec, 0x3f, imux_is_smixer ? AC_PWRST_D0 : parm);
394243737e0aSLydia Wang }
394343737e0aSLydia Wang 
394443737e0aSLydia Wang static int patch_vt3476(struct hda_codec *codec)
394543737e0aSLydia Wang {
394643737e0aSLydia Wang 	struct via_spec *spec;
394743737e0aSLydia Wang 	int err;
394843737e0aSLydia Wang 
394943737e0aSLydia Wang 	/* create a codec specific record */
395043737e0aSLydia Wang 	spec = via_new_spec(codec);
395143737e0aSLydia Wang 	if (spec == NULL)
395243737e0aSLydia Wang 		return -ENOMEM;
395343737e0aSLydia Wang 
395443737e0aSLydia Wang 	spec->aa_mix_nid = 0x3f;
395543737e0aSLydia Wang 	add_secret_dac_path(codec);
395643737e0aSLydia Wang 
395743737e0aSLydia Wang 	/* automatic parse from the BIOS config */
395843737e0aSLydia Wang 	err = via_parse_auto_config(codec);
395943737e0aSLydia Wang 	if (err < 0) {
396043737e0aSLydia Wang 		via_free(codec);
396143737e0aSLydia Wang 		return err;
396243737e0aSLydia Wang 	}
396343737e0aSLydia Wang 
396443737e0aSLydia Wang 	spec->init_verbs[spec->num_iverbs++] = vt3476_init_verbs;
396543737e0aSLydia Wang 
396643737e0aSLydia Wang 	codec->patch_ops = via_patch_ops;
396743737e0aSLydia Wang 
396843737e0aSLydia Wang 	spec->set_widgets_power_state = set_widgets_power_state_vt3476;
396943737e0aSLydia Wang 
397043737e0aSLydia Wang 	return 0;
397143737e0aSLydia Wang }
397243737e0aSLydia Wang 
3973c577b8a1SJoseph Chan /*
3974c577b8a1SJoseph Chan  * patch entries
3975c577b8a1SJoseph Chan  */
397690dd48a1STakashi Iwai static const struct hda_codec_preset snd_hda_preset_via[] = {
39773218c178STakashi Iwai 	{ .id = 0x11061708, .name = "VT1708", .patch = patch_vt1708},
39783218c178STakashi Iwai 	{ .id = 0x11061709, .name = "VT1708", .patch = patch_vt1708},
39793218c178STakashi Iwai 	{ .id = 0x1106170a, .name = "VT1708", .patch = patch_vt1708},
39803218c178STakashi Iwai 	{ .id = 0x1106170b, .name = "VT1708", .patch = patch_vt1708},
39813218c178STakashi Iwai 	{ .id = 0x1106e710, .name = "VT1709 10-Ch",
3982ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
39833218c178STakashi Iwai 	{ .id = 0x1106e711, .name = "VT1709 10-Ch",
3984ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
39853218c178STakashi Iwai 	{ .id = 0x1106e712, .name = "VT1709 10-Ch",
3986ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
39873218c178STakashi Iwai 	{ .id = 0x1106e713, .name = "VT1709 10-Ch",
3988ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
39893218c178STakashi Iwai 	{ .id = 0x1106e714, .name = "VT1709 6-Ch",
3990ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
39913218c178STakashi Iwai 	{ .id = 0x1106e715, .name = "VT1709 6-Ch",
3992ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
39933218c178STakashi Iwai 	{ .id = 0x1106e716, .name = "VT1709 6-Ch",
3994ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
39953218c178STakashi Iwai 	{ .id = 0x1106e717, .name = "VT1709 6-Ch",
3996ddd304d8STakashi Iwai 	  .patch = patch_vt1709},
39973218c178STakashi Iwai 	{ .id = 0x1106e720, .name = "VT1708B 8-Ch",
3998ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
39993218c178STakashi Iwai 	{ .id = 0x1106e721, .name = "VT1708B 8-Ch",
4000ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
40013218c178STakashi Iwai 	{ .id = 0x1106e722, .name = "VT1708B 8-Ch",
4002ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
40033218c178STakashi Iwai 	{ .id = 0x1106e723, .name = "VT1708B 8-Ch",
4004ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
40053218c178STakashi Iwai 	{ .id = 0x1106e724, .name = "VT1708B 4-Ch",
4006ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
40073218c178STakashi Iwai 	{ .id = 0x1106e725, .name = "VT1708B 4-Ch",
4008ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
40093218c178STakashi Iwai 	{ .id = 0x1106e726, .name = "VT1708B 4-Ch",
4010ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
40113218c178STakashi Iwai 	{ .id = 0x1106e727, .name = "VT1708B 4-Ch",
4012ddd304d8STakashi Iwai 	  .patch = patch_vt1708B},
40133218c178STakashi Iwai 	{ .id = 0x11060397, .name = "VT1708S",
4014d949cac1SHarald Welte 	  .patch = patch_vt1708S},
40153218c178STakashi Iwai 	{ .id = 0x11061397, .name = "VT1708S",
4016d949cac1SHarald Welte 	  .patch = patch_vt1708S},
40173218c178STakashi Iwai 	{ .id = 0x11062397, .name = "VT1708S",
4018d949cac1SHarald Welte 	  .patch = patch_vt1708S},
40193218c178STakashi Iwai 	{ .id = 0x11063397, .name = "VT1708S",
4020d949cac1SHarald Welte 	  .patch = patch_vt1708S},
4021bc92df7fSLydia Wang 	{ .id = 0x11064397, .name = "VT1705",
4022d949cac1SHarald Welte 	  .patch = patch_vt1708S},
40233218c178STakashi Iwai 	{ .id = 0x11065397, .name = "VT1708S",
4024d949cac1SHarald Welte 	  .patch = patch_vt1708S},
40253218c178STakashi Iwai 	{ .id = 0x11066397, .name = "VT1708S",
4026d949cac1SHarald Welte 	  .patch = patch_vt1708S},
40273218c178STakashi Iwai 	{ .id = 0x11067397, .name = "VT1708S",
4028d949cac1SHarald Welte 	  .patch = patch_vt1708S},
40293218c178STakashi Iwai 	{ .id = 0x11060398, .name = "VT1702",
4030d949cac1SHarald Welte 	  .patch = patch_vt1702},
40313218c178STakashi Iwai 	{ .id = 0x11061398, .name = "VT1702",
4032d949cac1SHarald Welte 	  .patch = patch_vt1702},
40333218c178STakashi Iwai 	{ .id = 0x11062398, .name = "VT1702",
4034d949cac1SHarald Welte 	  .patch = patch_vt1702},
40353218c178STakashi Iwai 	{ .id = 0x11063398, .name = "VT1702",
4036d949cac1SHarald Welte 	  .patch = patch_vt1702},
40373218c178STakashi Iwai 	{ .id = 0x11064398, .name = "VT1702",
4038d949cac1SHarald Welte 	  .patch = patch_vt1702},
40393218c178STakashi Iwai 	{ .id = 0x11065398, .name = "VT1702",
4040d949cac1SHarald Welte 	  .patch = patch_vt1702},
40413218c178STakashi Iwai 	{ .id = 0x11066398, .name = "VT1702",
4042d949cac1SHarald Welte 	  .patch = patch_vt1702},
40433218c178STakashi Iwai 	{ .id = 0x11067398, .name = "VT1702",
4044d949cac1SHarald Welte 	  .patch = patch_vt1702},
4045eb7188caSLydia Wang 	{ .id = 0x11060428, .name = "VT1718S",
4046eb7188caSLydia Wang 	  .patch = patch_vt1718S},
4047eb7188caSLydia Wang 	{ .id = 0x11064428, .name = "VT1718S",
4048eb7188caSLydia Wang 	  .patch = patch_vt1718S},
4049bb3c6bfcSLydia Wang 	{ .id = 0x11060441, .name = "VT2020",
4050bb3c6bfcSLydia Wang 	  .patch = patch_vt1718S},
4051bb3c6bfcSLydia Wang 	{ .id = 0x11064441, .name = "VT1828S",
4052bb3c6bfcSLydia Wang 	  .patch = patch_vt1718S},
4053f3db423dSLydia Wang 	{ .id = 0x11060433, .name = "VT1716S",
4054f3db423dSLydia Wang 	  .patch = patch_vt1716S},
4055f3db423dSLydia Wang 	{ .id = 0x1106a721, .name = "VT1716S",
4056f3db423dSLydia Wang 	  .patch = patch_vt1716S},
405725eaba2fSLydia Wang 	{ .id = 0x11060438, .name = "VT2002P", .patch = patch_vt2002P},
405825eaba2fSLydia Wang 	{ .id = 0x11064438, .name = "VT2002P", .patch = patch_vt2002P},
4059ab6734e7SLydia Wang 	{ .id = 0x11060448, .name = "VT1812", .patch = patch_vt1812},
406036dd5c4aSLydia Wang 	{ .id = 0x11060440, .name = "VT1818S",
406136dd5c4aSLydia Wang 	  .patch = patch_vt1708S},
406211890956SLydia Wang 	{ .id = 0x11060446, .name = "VT1802",
406311890956SLydia Wang 		.patch = patch_vt2002P},
406411890956SLydia Wang 	{ .id = 0x11068446, .name = "VT1802",
406511890956SLydia Wang 		.patch = patch_vt2002P},
406643737e0aSLydia Wang 	{ .id = 0x11064760, .name = "VT1705CF",
406743737e0aSLydia Wang 		.patch = patch_vt3476},
4068*6121b84aSLydia Wang 	{ .id = 0x11064761, .name = "VT1708SCE",
4069*6121b84aSLydia Wang 		.patch = patch_vt3476},
4070*6121b84aSLydia Wang 	{ .id = 0x11064762, .name = "VT1808",
4071*6121b84aSLydia Wang 		.patch = patch_vt3476},
4072c577b8a1SJoseph Chan 	{} /* terminator */
4073c577b8a1SJoseph Chan };
40741289e9e8STakashi Iwai 
40751289e9e8STakashi Iwai MODULE_ALIAS("snd-hda-codec-id:1106*");
40761289e9e8STakashi Iwai 
40771289e9e8STakashi Iwai static struct hda_codec_preset_list via_list = {
40781289e9e8STakashi Iwai 	.preset = snd_hda_preset_via,
40791289e9e8STakashi Iwai 	.owner = THIS_MODULE,
40801289e9e8STakashi Iwai };
40811289e9e8STakashi Iwai 
40821289e9e8STakashi Iwai MODULE_LICENSE("GPL");
40831289e9e8STakashi Iwai MODULE_DESCRIPTION("VIA HD-audio codec");
40841289e9e8STakashi Iwai 
40851289e9e8STakashi Iwai static int __init patch_via_init(void)
40861289e9e8STakashi Iwai {
40871289e9e8STakashi Iwai 	return snd_hda_add_codec_preset(&via_list);
40881289e9e8STakashi Iwai }
40891289e9e8STakashi Iwai 
40901289e9e8STakashi Iwai static void __exit patch_via_exit(void)
40911289e9e8STakashi Iwai {
40921289e9e8STakashi Iwai 	snd_hda_delete_codec_preset(&via_list);
40931289e9e8STakashi Iwai }
40941289e9e8STakashi Iwai 
40951289e9e8STakashi Iwai module_init(patch_via_init)
40961289e9e8STakashi Iwai module_exit(patch_via_exit)
4097