blob: 71a058fcf884a8aaa114bfbd9f3d709121beedd5 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * Universal Interface for Intel High Definition Audio Codec
3 *
Takashi Iwai1d045db2011-07-07 18:23:21 +02004 * HD audio interface patch for Realtek ALC codecs
Linus Torvalds1da177e2005-04-16 15:20:36 -07005 *
Kailang Yangdf694da2005-12-05 19:42:22 +01006 * Copyright (c) 2004 Kailang Yang <kailang@realtek.com.tw>
7 * PeiSen Hou <pshou@realtek.com.tw>
Linus Torvalds1da177e2005-04-16 15:20:36 -07008 * Takashi Iwai <tiwai@suse.de>
Jonathan Woithe409a3e92012-03-27 13:01:01 +10309 * Jonathan Woithe <jwoithe@just42.net>
Linus Torvalds1da177e2005-04-16 15:20:36 -070010 *
11 * This driver is free software; you can redistribute it and/or modify
12 * it under the terms of the GNU General Public License as published by
13 * the Free Software Foundation; either version 2 of the License, or
14 * (at your option) any later version.
15 *
16 * This driver is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 * GNU General Public License for more details.
20 *
21 * You should have received a copy of the GNU General Public License
22 * along with this program; if not, write to the Free Software
23 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
24 */
25
Linus Torvalds1da177e2005-04-16 15:20:36 -070026#include <linux/init.h>
27#include <linux/delay.h>
28#include <linux/slab.h>
29#include <linux/pci.h>
Takashi Iwai08fb0d02013-01-10 17:33:58 +010030#include <linux/dmi.h>
Paul Gortmakerda155d52011-07-15 12:38:28 -040031#include <linux/module.h>
David Henningsson33f4acd2015-01-07 15:50:13 +010032#include <linux/input.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070033#include <sound/core.h>
Kailang Yang9ad0e492010-09-14 23:22:00 +020034#include <sound/jack.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070035#include "hda_codec.h"
36#include "hda_local.h"
Takashi Iwai23d30f22012-05-07 17:17:32 +020037#include "hda_auto_parser.h"
Takashi Iwai1835a0f2011-10-27 22:12:46 +020038#include "hda_jack.h"
Takashi Iwai08c189f2012-12-19 15:22:24 +010039#include "hda_generic.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070040
Takashi Iwaicd63a5f2013-07-05 12:13:59 +020041/* keep halting ALC5505 DSP, for power saving */
42#define HALT_REALTEK_ALC5505
43
Kailang Yangdf694da2005-12-05 19:42:22 +010044/* for GPIO Poll */
45#define GPIO_MASK 0x03
46
Takashi Iwai4a79ba32009-04-22 16:31:35 +020047/* extra amp-initialization sequence types */
48enum {
49 ALC_INIT_NONE,
50 ALC_INIT_DEFAULT,
51 ALC_INIT_GPIO1,
52 ALC_INIT_GPIO2,
53 ALC_INIT_GPIO3,
54};
55
David Henningsson73bdd592013-04-15 15:44:14 +020056enum {
57 ALC_HEADSET_MODE_UNKNOWN,
58 ALC_HEADSET_MODE_UNPLUGGED,
59 ALC_HEADSET_MODE_HEADSET,
60 ALC_HEADSET_MODE_MIC,
61 ALC_HEADSET_MODE_HEADPHONE,
62};
63
64enum {
65 ALC_HEADSET_TYPE_UNKNOWN,
66 ALC_HEADSET_TYPE_CTIA,
67 ALC_HEADSET_TYPE_OMTP,
68};
69
Hui Wangc7b60a82015-12-28 11:35:25 +080070enum {
71 ALC_KEY_MICMUTE_INDEX,
72};
73
Kailang Yangda00c242010-03-19 11:23:45 +010074struct alc_customize_define {
75 unsigned int sku_cfg;
76 unsigned char port_connectivity;
77 unsigned char check_sum;
78 unsigned char customization;
79 unsigned char external_amp;
80 unsigned int enable_pcbeep:1;
81 unsigned int platform_type:1;
82 unsigned int swap:1;
83 unsigned int override:1;
David Henningsson90622912010-10-14 14:50:18 +020084 unsigned int fixup:1; /* Means that this sku is set by driver, not read from hw */
Kailang Yangda00c242010-03-19 11:23:45 +010085};
86
Linus Torvalds1da177e2005-04-16 15:20:36 -070087struct alc_spec {
Takashi Iwai08c189f2012-12-19 15:22:24 +010088 struct hda_gen_spec gen; /* must be at head */
Takashi Iwai23d30f22012-05-07 17:17:32 +020089
Linus Torvalds1da177e2005-04-16 15:20:36 -070090 /* codec parameterization */
Takashi Iwaia9111322011-05-02 11:30:18 +020091 const struct snd_kcontrol_new *mixers[5]; /* mixer arrays */
Linus Torvalds1da177e2005-04-16 15:20:36 -070092 unsigned int num_mixers;
Takashi Iwai45bdd1c2009-02-06 16:11:25 +010093 unsigned int beep_amp; /* beep amp value, set via set_beep_amp() */
Linus Torvalds1da177e2005-04-16 15:20:36 -070094
Kailang Yangda00c242010-03-19 11:23:45 +010095 struct alc_customize_define cdefine;
Takashi Iwai08c189f2012-12-19 15:22:24 +010096 unsigned int parse_flags; /* flag for snd_hda_parse_pin_defcfg() */
97
Takashi Iwai08fb0d02013-01-10 17:33:58 +010098 /* mute LED for HP laptops, see alc269_fixup_mic_mute_hook() */
99 int mute_led_polarity;
100 hda_nid_t mute_led_nid;
Kailang Yang9c5dc3b2014-06-19 16:11:36 +0800101 hda_nid_t cap_mute_led_nid;
Takashi Iwai08fb0d02013-01-10 17:33:58 +0100102
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +0100103 unsigned int gpio_led; /* used for alc269_fixup_hp_gpio_led() */
Takashi Iwai0f32fd192014-11-19 12:16:14 +0100104 unsigned int gpio_mute_led_mask;
105 unsigned int gpio_mic_led_mask;
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +0100106
David Henningsson73bdd592013-04-15 15:44:14 +0200107 hda_nid_t headset_mic_pin;
108 hda_nid_t headphone_mic_pin;
109 int current_headset_mode;
110 int current_headset_type;
111
Takashi Iwaiae6b8132006-03-03 16:47:17 +0100112 /* hooks */
113 void (*init_hook)(struct hda_codec *codec);
Takashi Iwai83012a72012-08-24 18:38:08 +0200114#ifdef CONFIG_PM
Daniel T Chenc97259d2009-12-27 18:52:08 -0500115 void (*power_hook)(struct hda_codec *codec);
Hector Martinf5de24b2009-12-20 22:51:31 +0100116#endif
Takashi Iwai1c716152011-04-07 10:37:16 +0200117 void (*shutup)(struct hda_codec *codec);
Takashi Iwai70a09762015-12-15 14:59:58 +0100118 void (*reboot_notify)(struct hda_codec *codec);
Takashi Iwaid922b512011-04-28 12:18:53 +0200119
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200120 int init_amp;
Takashi Iwaid433a672010-09-20 15:11:54 +0200121 int codec_variant; /* flag for other variants */
Kailang Yang97a26572013-11-29 00:35:26 -0500122 unsigned int has_alc5505_dsp:1;
123 unsigned int no_depop_delay:1;
Takashi Iwaie64f14f2009-01-20 18:32:55 +0100124
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +0200125 /* for PLL fix */
126 hda_nid_t pll_nid;
127 unsigned int pll_coef_idx, pll_coef_bit;
Takashi Iwai1bb7e432011-10-17 16:50:59 +0200128 unsigned int coef0;
David Henningsson33f4acd2015-01-07 15:50:13 +0100129 struct input_dev *kb_dev;
Hui Wangc7b60a82015-12-28 11:35:25 +0800130 u8 alc_mute_keycode_map[1];
Kailang Yangdf694da2005-12-05 19:42:22 +0100131};
132
Takashi Iwai23f0c042009-02-26 13:03:58 +0100133/*
Takashi Iwaif2a227c2014-08-18 13:35:22 +0200134 * COEF access helper functions
135 */
136
137static int alc_read_coefex_idx(struct hda_codec *codec, hda_nid_t nid,
138 unsigned int coef_idx)
139{
140 unsigned int val;
141
142 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_COEF_INDEX, coef_idx);
143 val = snd_hda_codec_read(codec, nid, 0, AC_VERB_GET_PROC_COEF, 0);
144 return val;
145}
146
147#define alc_read_coef_idx(codec, coef_idx) \
148 alc_read_coefex_idx(codec, 0x20, coef_idx)
149
150static void alc_write_coefex_idx(struct hda_codec *codec, hda_nid_t nid,
151 unsigned int coef_idx, unsigned int coef_val)
152{
153 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_COEF_INDEX, coef_idx);
154 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_PROC_COEF, coef_val);
155}
156
157#define alc_write_coef_idx(codec, coef_idx, coef_val) \
158 alc_write_coefex_idx(codec, 0x20, coef_idx, coef_val)
159
Takashi Iwai98b24882014-08-18 13:47:50 +0200160static void alc_update_coefex_idx(struct hda_codec *codec, hda_nid_t nid,
161 unsigned int coef_idx, unsigned int mask,
162 unsigned int bits_set)
163{
164 unsigned int val = alc_read_coefex_idx(codec, nid, coef_idx);
165
166 if (val != -1)
167 alc_write_coefex_idx(codec, nid, coef_idx,
168 (val & ~mask) | bits_set);
169}
170
171#define alc_update_coef_idx(codec, coef_idx, mask, bits_set) \
172 alc_update_coefex_idx(codec, 0x20, coef_idx, mask, bits_set)
173
Takashi Iwaif2a227c2014-08-18 13:35:22 +0200174/* a special bypass for COEF 0; read the cached value at the second time */
175static unsigned int alc_get_coef0(struct hda_codec *codec)
176{
177 struct alc_spec *spec = codec->spec;
178
179 if (!spec->coef0)
180 spec->coef0 = alc_read_coef_idx(codec, 0);
181 return spec->coef0;
182}
183
Takashi Iwai54db6c32014-08-18 15:11:19 +0200184/* coef writes/updates batch */
185struct coef_fw {
186 unsigned char nid;
187 unsigned char idx;
188 unsigned short mask;
189 unsigned short val;
190};
191
192#define UPDATE_COEFEX(_nid, _idx, _mask, _val) \
193 { .nid = (_nid), .idx = (_idx), .mask = (_mask), .val = (_val) }
194#define WRITE_COEFEX(_nid, _idx, _val) UPDATE_COEFEX(_nid, _idx, -1, _val)
195#define WRITE_COEF(_idx, _val) WRITE_COEFEX(0x20, _idx, _val)
196#define UPDATE_COEF(_idx, _mask, _val) UPDATE_COEFEX(0x20, _idx, _mask, _val)
197
198static void alc_process_coef_fw(struct hda_codec *codec,
199 const struct coef_fw *fw)
200{
201 for (; fw->nid; fw++) {
202 if (fw->mask == (unsigned short)-1)
203 alc_write_coefex_idx(codec, fw->nid, fw->idx, fw->val);
204 else
205 alc_update_coefex_idx(codec, fw->nid, fw->idx,
206 fw->mask, fw->val);
207 }
208}
209
Takashi Iwaif2a227c2014-08-18 13:35:22 +0200210/*
Takashi Iwai1d045db2011-07-07 18:23:21 +0200211 * Append the given mixer and verb elements for the later use
212 * The mixer array is referred in build_controls(), and init_verbs are
213 * called in init().
Takashi Iwaid88897e2008-10-31 15:01:37 +0100214 */
Takashi Iwaia9111322011-05-02 11:30:18 +0200215static void add_mixer(struct alc_spec *spec, const struct snd_kcontrol_new *mix)
Takashi Iwaid88897e2008-10-31 15:01:37 +0100216{
217 if (snd_BUG_ON(spec->num_mixers >= ARRAY_SIZE(spec->mixers)))
218 return;
219 spec->mixers[spec->num_mixers++] = mix;
220}
221
Takashi Iwaid88897e2008-10-31 15:01:37 +0100222/*
Takashi Iwai1d045db2011-07-07 18:23:21 +0200223 * GPIO setup tables, used in initialization
Kailang Yangdf694da2005-12-05 19:42:22 +0100224 */
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200225/* Enable GPIO mask and set output */
Takashi Iwaia9111322011-05-02 11:30:18 +0200226static const struct hda_verb alc_gpio1_init_verbs[] = {
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200227 {0x01, AC_VERB_SET_GPIO_MASK, 0x01},
228 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x01},
229 {0x01, AC_VERB_SET_GPIO_DATA, 0x01},
230 { }
231};
232
Takashi Iwaia9111322011-05-02 11:30:18 +0200233static const struct hda_verb alc_gpio2_init_verbs[] = {
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200234 {0x01, AC_VERB_SET_GPIO_MASK, 0x02},
235 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x02},
236 {0x01, AC_VERB_SET_GPIO_DATA, 0x02},
237 { }
238};
239
Takashi Iwaia9111322011-05-02 11:30:18 +0200240static const struct hda_verb alc_gpio3_init_verbs[] = {
Kailang Yangbdd148a2007-05-08 15:19:08 +0200241 {0x01, AC_VERB_SET_GPIO_MASK, 0x03},
242 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x03},
243 {0x01, AC_VERB_SET_GPIO_DATA, 0x03},
244 { }
245};
246
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +0200247/*
248 * Fix hardware PLL issue
249 * On some codecs, the analog PLL gating control must be off while
250 * the default value is 1.
251 */
252static void alc_fix_pll(struct hda_codec *codec)
253{
254 struct alc_spec *spec = codec->spec;
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +0200255
Takashi Iwai98b24882014-08-18 13:47:50 +0200256 if (spec->pll_nid)
257 alc_update_coefex_idx(codec, spec->pll_nid, spec->pll_coef_idx,
258 1 << spec->pll_coef_bit, 0);
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +0200259}
260
261static void alc_fix_pll_init(struct hda_codec *codec, hda_nid_t nid,
262 unsigned int coef_idx, unsigned int coef_bit)
263{
264 struct alc_spec *spec = codec->spec;
265 spec->pll_nid = nid;
266 spec->pll_coef_idx = coef_idx;
267 spec->pll_coef_bit = coef_bit;
268 alc_fix_pll(codec);
269}
270
Takashi Iwaicf5a2272012-02-20 16:31:07 +0100271/* update the master volume per volume-knob's unsol event */
Takashi Iwai1a4f69d2014-09-11 15:22:46 +0200272static void alc_update_knob_master(struct hda_codec *codec,
273 struct hda_jack_callback *jack)
Takashi Iwaicf5a2272012-02-20 16:31:07 +0100274{
275 unsigned int val;
276 struct snd_kcontrol *kctl;
277 struct snd_ctl_elem_value *uctl;
278
279 kctl = snd_hda_find_mixer_ctl(codec, "Master Playback Volume");
280 if (!kctl)
281 return;
282 uctl = kzalloc(sizeof(*uctl), GFP_KERNEL);
283 if (!uctl)
284 return;
Takashi Iwai2ebab402016-02-09 10:23:52 +0100285 val = snd_hda_codec_read(codec, jack->nid, 0,
Takashi Iwaicf5a2272012-02-20 16:31:07 +0100286 AC_VERB_GET_VOLUME_KNOB_CONTROL, 0);
287 val &= HDA_AMP_VOLMASK;
288 uctl->value.integer.value[0] = val;
289 uctl->value.integer.value[1] = val;
290 kctl->put(kctl, uctl);
291 kfree(uctl);
292}
293
David Henningsson29adc4b2012-09-25 11:31:00 +0200294static void alc880_unsol_event(struct hda_codec *codec, unsigned int res)
Takashi Iwaif21d78e2012-01-19 12:10:29 +0100295{
David Henningsson29adc4b2012-09-25 11:31:00 +0200296 /* For some reason, the res given from ALC880 is broken.
297 Here we adjust it properly. */
298 snd_hda_jack_unsol_event(codec, res >> 2);
Takashi Iwaif21d78e2012-01-19 12:10:29 +0100299}
300
Kailang Yang394c97f2014-11-12 17:38:08 +0800301/* Change EAPD to verb control */
302static void alc_fill_eapd_coef(struct hda_codec *codec)
303{
304 int coef;
305
306 coef = alc_get_coef0(codec);
307
Takashi Iwai7639a062015-03-03 10:07:24 +0100308 switch (codec->core.vendor_id) {
Kailang Yang394c97f2014-11-12 17:38:08 +0800309 case 0x10ec0262:
310 alc_update_coef_idx(codec, 0x7, 0, 1<<5);
311 break;
312 case 0x10ec0267:
313 case 0x10ec0268:
314 alc_update_coef_idx(codec, 0x7, 0, 1<<13);
315 break;
316 case 0x10ec0269:
317 if ((coef & 0x00f0) == 0x0010)
318 alc_update_coef_idx(codec, 0xd, 0, 1<<14);
319 if ((coef & 0x00f0) == 0x0020)
320 alc_update_coef_idx(codec, 0x4, 1<<15, 0);
321 if ((coef & 0x00f0) == 0x0030)
322 alc_update_coef_idx(codec, 0x10, 1<<9, 0);
323 break;
324 case 0x10ec0280:
325 case 0x10ec0284:
326 case 0x10ec0290:
327 case 0x10ec0292:
328 alc_update_coef_idx(codec, 0x4, 1<<15, 0);
329 break;
Kailang Yang42314302016-02-03 15:03:50 +0800330 case 0x10ec0225:
Kailang Yang394c97f2014-11-12 17:38:08 +0800331 case 0x10ec0233:
Kailang Yang61ae3fb2017-10-20 15:06:34 +0800332 case 0x10ec0236:
Kailang Yang394c97f2014-11-12 17:38:08 +0800333 case 0x10ec0255:
Kailang Yang4344aec2014-12-17 17:39:05 +0800334 case 0x10ec0256:
Kailang Yang394c97f2014-11-12 17:38:08 +0800335 case 0x10ec0282:
336 case 0x10ec0283:
337 case 0x10ec0286:
338 case 0x10ec0288:
Kailang Yang7d727862016-05-24 16:46:07 +0800339 case 0x10ec0295:
Kailang Yang506b62c2014-12-18 17:07:44 +0800340 case 0x10ec0298:
Kailang Yangf6e94c22017-01-04 14:49:07 +0800341 case 0x10ec0299:
Kailang Yang394c97f2014-11-12 17:38:08 +0800342 alc_update_coef_idx(codec, 0x10, 1<<9, 0);
343 break;
344 case 0x10ec0285:
345 case 0x10ec0293:
346 alc_update_coef_idx(codec, 0xa, 1<<13, 0);
347 break;
Kailang Yangdcd4f0d2016-05-04 15:50:18 +0800348 case 0x10ec0234:
349 case 0x10ec0274:
350 case 0x10ec0294:
Kailang Yang6fbae352016-05-30 16:44:20 +0800351 case 0x10ec0700:
352 case 0x10ec0701:
353 case 0x10ec0703:
Kailang Yangdcd4f0d2016-05-04 15:50:18 +0800354 alc_update_coef_idx(codec, 0x10, 1<<15, 0);
355 break;
Kailang Yang394c97f2014-11-12 17:38:08 +0800356 case 0x10ec0662:
357 if ((coef & 0x00f0) == 0x0030)
358 alc_update_coef_idx(codec, 0x4, 1<<10, 0); /* EAPD Ctrl */
359 break;
360 case 0x10ec0272:
361 case 0x10ec0273:
362 case 0x10ec0663:
363 case 0x10ec0665:
364 case 0x10ec0670:
365 case 0x10ec0671:
366 case 0x10ec0672:
367 alc_update_coef_idx(codec, 0xd, 0, 1<<14); /* EAPD Ctrl */
368 break;
369 case 0x10ec0668:
370 alc_update_coef_idx(codec, 0x7, 3<<13, 0);
371 break;
372 case 0x10ec0867:
373 alc_update_coef_idx(codec, 0x4, 1<<10, 0);
374 break;
375 case 0x10ec0888:
376 if ((coef & 0x00f0) == 0x0020 || (coef & 0x00f0) == 0x0030)
377 alc_update_coef_idx(codec, 0x7, 1<<5, 0);
378 break;
379 case 0x10ec0892:
380 alc_update_coef_idx(codec, 0x7, 1<<5, 0);
381 break;
382 case 0x10ec0899:
383 case 0x10ec0900:
384 alc_update_coef_idx(codec, 0x7, 1<<1, 0);
385 break;
386 }
387}
388
Kailang Yangf9423e72008-05-27 12:32:25 +0200389/* additional initialization for ALC888 variants */
390static void alc888_coef_init(struct hda_codec *codec)
391{
Kailang Yang1df88742014-10-29 16:10:13 +0800392 switch (alc_get_coef0(codec) & 0x00f0) {
393 /* alc888-VA */
394 case 0x00:
395 /* alc888-VB */
396 case 0x10:
397 alc_update_coef_idx(codec, 7, 0, 0x2030); /* Turn EAPD to High */
398 break;
399 }
Jaroslav Kysela87a8c372009-07-23 10:58:29 +0200400}
401
Takashi Iwai3fb4a502010-01-19 15:46:37 +0100402/* turn on/off EAPD control (only if available) */
403static void set_eapd(struct hda_codec *codec, hda_nid_t nid, int on)
404{
405 if (get_wcaps_type(get_wcaps(codec, nid)) != AC_WID_PIN)
406 return;
407 if (snd_hda_query_pin_caps(codec, nid) & AC_PINCAP_EAPD)
408 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_EAPD_BTLENABLE,
409 on ? 2 : 0);
410}
411
Takashi Iwai691f1fc2011-04-07 10:31:43 +0200412/* turn on/off EAPD controls of the codec */
413static void alc_auto_setup_eapd(struct hda_codec *codec, bool on)
414{
415 /* We currently only handle front, HP */
Takashi Iwai39fa84e2011-06-27 15:28:57 +0200416 static hda_nid_t pins[] = {
Hui Wangaf95b412015-03-26 17:14:55 +0800417 0x0f, 0x10, 0x14, 0x15, 0x17, 0
Takashi Iwai39fa84e2011-06-27 15:28:57 +0200418 };
419 hda_nid_t *p;
420 for (p = pins; *p; p++)
421 set_eapd(codec, *p, on);
Takashi Iwai691f1fc2011-04-07 10:31:43 +0200422}
423
Takashi Iwai1c716152011-04-07 10:37:16 +0200424/* generic shutup callback;
425 * just turning off EPAD and a little pause for avoiding pop-noise
426 */
427static void alc_eapd_shutup(struct hda_codec *codec)
428{
Kailang Yang97a26572013-11-29 00:35:26 -0500429 struct alc_spec *spec = codec->spec;
430
Takashi Iwai1c716152011-04-07 10:37:16 +0200431 alc_auto_setup_eapd(codec, false);
Kailang Yang97a26572013-11-29 00:35:26 -0500432 if (!spec->no_depop_delay)
433 msleep(200);
Takashi Iwai9bfb2842013-07-24 14:31:50 +0200434 snd_hda_shutup_pins(codec);
Takashi Iwai1c716152011-04-07 10:37:16 +0200435}
436
Takashi Iwai1d045db2011-07-07 18:23:21 +0200437/* generic EAPD initialization */
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200438static void alc_auto_init_amp(struct hda_codec *codec, int type)
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200439{
Kailang Yang394c97f2014-11-12 17:38:08 +0800440 alc_fill_eapd_coef(codec);
Takashi Iwai39fa84e2011-06-27 15:28:57 +0200441 alc_auto_setup_eapd(codec, true);
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200442 switch (type) {
443 case ALC_INIT_GPIO1:
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200444 snd_hda_sequence_write(codec, alc_gpio1_init_verbs);
445 break;
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200446 case ALC_INIT_GPIO2:
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200447 snd_hda_sequence_write(codec, alc_gpio2_init_verbs);
448 break;
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200449 case ALC_INIT_GPIO3:
Kailang Yangbdd148a2007-05-08 15:19:08 +0200450 snd_hda_sequence_write(codec, alc_gpio3_init_verbs);
451 break;
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200452 case ALC_INIT_DEFAULT:
Takashi Iwai7639a062015-03-03 10:07:24 +0100453 switch (codec->core.vendor_id) {
Kailang Yangc9b58002007-10-16 14:30:01 +0200454 case 0x10ec0260:
Takashi Iwai98b24882014-08-18 13:47:50 +0200455 alc_update_coefex_idx(codec, 0x1a, 7, 0, 0x2010);
Kailang Yangc9b58002007-10-16 14:30:01 +0200456 break;
Kailang Yangc9b58002007-10-16 14:30:01 +0200457 case 0x10ec0880:
458 case 0x10ec0882:
459 case 0x10ec0883:
460 case 0x10ec0885:
Kailang Yang1df88742014-10-29 16:10:13 +0800461 alc_update_coef_idx(codec, 7, 0, 0x2030);
Kailang Yangc9b58002007-10-16 14:30:01 +0200462 break;
Kailang Yangf9423e72008-05-27 12:32:25 +0200463 case 0x10ec0888:
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200464 alc888_coef_init(codec);
Kailang Yangf9423e72008-05-27 12:32:25 +0200465 break;
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200466 }
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200467 break;
468 }
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200469}
Kailang Yangea1fb292008-08-26 12:58:38 +0200470
Takashi Iwai1d045db2011-07-07 18:23:21 +0200471
472/*
473 * Realtek SSID verification
474 */
475
David Henningsson90622912010-10-14 14:50:18 +0200476/* Could be any non-zero and even value. When used as fixup, tells
477 * the driver to ignore any present sku defines.
478 */
479#define ALC_FIXUP_SKU_IGNORE (2)
480
Takashi Iwai23d30f22012-05-07 17:17:32 +0200481static void alc_fixup_sku_ignore(struct hda_codec *codec,
482 const struct hda_fixup *fix, int action)
483{
484 struct alc_spec *spec = codec->spec;
485 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
486 spec->cdefine.fixup = 1;
487 spec->cdefine.sku_cfg = ALC_FIXUP_SKU_IGNORE;
488 }
489}
490
Mengdong Linb5c66112013-11-29 00:35:35 -0500491static void alc_fixup_no_depop_delay(struct hda_codec *codec,
492 const struct hda_fixup *fix, int action)
493{
494 struct alc_spec *spec = codec->spec;
495
Mengdong Lin84d2dc32013-12-02 22:26:03 -0500496 if (action == HDA_FIXUP_ACT_PROBE) {
Mengdong Linb5c66112013-11-29 00:35:35 -0500497 spec->no_depop_delay = 1;
Mengdong Lin84d2dc32013-12-02 22:26:03 -0500498 codec->depop_delay = 0;
499 }
Mengdong Linb5c66112013-11-29 00:35:35 -0500500}
501
Kailang Yangda00c242010-03-19 11:23:45 +0100502static int alc_auto_parse_customize_define(struct hda_codec *codec)
503{
504 unsigned int ass, tmp, i;
Takashi Iwai7fb56222010-03-22 17:09:47 +0100505 unsigned nid = 0;
Kailang Yangda00c242010-03-19 11:23:45 +0100506 struct alc_spec *spec = codec->spec;
507
Takashi Iwaib6cbe512010-07-28 17:43:36 +0200508 spec->cdefine.enable_pcbeep = 1; /* assume always enabled */
509
David Henningsson90622912010-10-14 14:50:18 +0200510 if (spec->cdefine.fixup) {
511 ass = spec->cdefine.sku_cfg;
512 if (ass == ALC_FIXUP_SKU_IGNORE)
513 return -1;
514 goto do_sku;
515 }
516
Takashi Iwai5100cd02014-02-15 10:03:19 +0100517 if (!codec->bus->pci)
518 return -1;
Takashi Iwai7639a062015-03-03 10:07:24 +0100519 ass = codec->core.subsystem_id & 0xffff;
Takashi Iwaib6cbe512010-07-28 17:43:36 +0200520 if (ass != codec->bus->pci->subsystem_device && (ass & 1))
Kailang Yangda00c242010-03-19 11:23:45 +0100521 goto do_sku;
522
523 nid = 0x1d;
Takashi Iwai7639a062015-03-03 10:07:24 +0100524 if (codec->core.vendor_id == 0x10ec0260)
Kailang Yangda00c242010-03-19 11:23:45 +0100525 nid = 0x17;
526 ass = snd_hda_codec_get_pincfg(codec, nid);
527
528 if (!(ass & 1)) {
Takashi Iwai4e76a882014-02-25 12:21:03 +0100529 codec_info(codec, "%s: SKU not ready 0x%08x\n",
Takashi Iwai7639a062015-03-03 10:07:24 +0100530 codec->core.chip_name, ass);
Kailang Yangda00c242010-03-19 11:23:45 +0100531 return -1;
532 }
533
534 /* check sum */
535 tmp = 0;
536 for (i = 1; i < 16; i++) {
537 if ((ass >> i) & 1)
538 tmp++;
539 }
540 if (((ass >> 16) & 0xf) != tmp)
541 return -1;
542
543 spec->cdefine.port_connectivity = ass >> 30;
544 spec->cdefine.enable_pcbeep = (ass & 0x100000) >> 20;
545 spec->cdefine.check_sum = (ass >> 16) & 0xf;
546 spec->cdefine.customization = ass >> 8;
547do_sku:
548 spec->cdefine.sku_cfg = ass;
549 spec->cdefine.external_amp = (ass & 0x38) >> 3;
550 spec->cdefine.platform_type = (ass & 0x4) >> 2;
551 spec->cdefine.swap = (ass & 0x2) >> 1;
552 spec->cdefine.override = ass & 0x1;
553
Takashi Iwai4e76a882014-02-25 12:21:03 +0100554 codec_dbg(codec, "SKU: Nid=0x%x sku_cfg=0x%08x\n",
Kailang Yangda00c242010-03-19 11:23:45 +0100555 nid, spec->cdefine.sku_cfg);
Takashi Iwai4e76a882014-02-25 12:21:03 +0100556 codec_dbg(codec, "SKU: port_connectivity=0x%x\n",
Kailang Yangda00c242010-03-19 11:23:45 +0100557 spec->cdefine.port_connectivity);
Takashi Iwai4e76a882014-02-25 12:21:03 +0100558 codec_dbg(codec, "SKU: enable_pcbeep=0x%x\n", spec->cdefine.enable_pcbeep);
559 codec_dbg(codec, "SKU: check_sum=0x%08x\n", spec->cdefine.check_sum);
560 codec_dbg(codec, "SKU: customization=0x%08x\n", spec->cdefine.customization);
561 codec_dbg(codec, "SKU: external_amp=0x%x\n", spec->cdefine.external_amp);
562 codec_dbg(codec, "SKU: platform_type=0x%x\n", spec->cdefine.platform_type);
563 codec_dbg(codec, "SKU: swap=0x%x\n", spec->cdefine.swap);
564 codec_dbg(codec, "SKU: override=0x%x\n", spec->cdefine.override);
Kailang Yangda00c242010-03-19 11:23:45 +0100565
566 return 0;
567}
568
Takashi Iwai08c189f2012-12-19 15:22:24 +0100569/* return the position of NID in the list, or -1 if not found */
570static int find_idx_in_nid_list(hda_nid_t nid, const hda_nid_t *list, int nums)
571{
572 int i;
573 for (i = 0; i < nums; i++)
574 if (list[i] == nid)
575 return i;
576 return -1;
577}
Takashi Iwai1d045db2011-07-07 18:23:21 +0200578/* return true if the given NID is found in the list */
Takashi Iwai3af9ee62011-06-27 12:34:01 +0200579static bool found_in_nid_list(hda_nid_t nid, const hda_nid_t *list, int nums)
580{
Takashi Iwai21268962011-07-07 15:01:13 +0200581 return find_idx_in_nid_list(nid, list, nums) >= 0;
Takashi Iwai3af9ee62011-06-27 12:34:01 +0200582}
583
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200584/* check subsystem ID and set up device-specific initialization;
585 * return 1 if initialized, 0 if invalid SSID
586 */
587/* 32-bit subsystem ID for BIOS loading in HD Audio codec.
588 * 31 ~ 16 : Manufacture ID
589 * 15 ~ 8 : SKU ID
590 * 7 ~ 0 : Assembly ID
591 * port-A --> pin 39/41, port-E --> pin 14/15, port-D --> pin 35/36
592 */
Takashi Iwai58c57cf2014-01-07 18:22:49 +0100593static int alc_subsystem_id(struct hda_codec *codec, const hda_nid_t *ports)
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200594{
595 unsigned int ass, tmp, i;
596 unsigned nid;
597 struct alc_spec *spec = codec->spec;
598
David Henningsson90622912010-10-14 14:50:18 +0200599 if (spec->cdefine.fixup) {
600 ass = spec->cdefine.sku_cfg;
601 if (ass == ALC_FIXUP_SKU_IGNORE)
602 return 0;
603 goto do_sku;
604 }
605
Takashi Iwai7639a062015-03-03 10:07:24 +0100606 ass = codec->core.subsystem_id & 0xffff;
Takashi Iwai5100cd02014-02-15 10:03:19 +0100607 if (codec->bus->pci &&
608 ass != codec->bus->pci->subsystem_device && (ass & 1))
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200609 goto do_sku;
610
611 /* invalid SSID, check the special NID pin defcfg instead */
612 /*
Sasha Alexandrdef319f2009-06-16 16:00:15 -0400613 * 31~30 : port connectivity
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200614 * 29~21 : reserve
615 * 20 : PCBEEP input
616 * 19~16 : Check sum (15:1)
617 * 15~1 : Custom
618 * 0 : override
619 */
620 nid = 0x1d;
Takashi Iwai7639a062015-03-03 10:07:24 +0100621 if (codec->core.vendor_id == 0x10ec0260)
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200622 nid = 0x17;
623 ass = snd_hda_codec_get_pincfg(codec, nid);
Takashi Iwai4e76a882014-02-25 12:21:03 +0100624 codec_dbg(codec,
625 "realtek: No valid SSID, checking pincfg 0x%08x for NID 0x%x\n",
Takashi Iwaicb6605c2009-04-28 13:03:19 +0200626 ass, nid);
Kailang Yang6227cdc2010-02-25 08:36:52 +0100627 if (!(ass & 1))
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200628 return 0;
629 if ((ass >> 30) != 1) /* no physical connection */
630 return 0;
631
632 /* check sum */
633 tmp = 0;
634 for (i = 1; i < 16; i++) {
635 if ((ass >> i) & 1)
636 tmp++;
637 }
638 if (((ass >> 16) & 0xf) != tmp)
639 return 0;
640do_sku:
Takashi Iwai4e76a882014-02-25 12:21:03 +0100641 codec_dbg(codec, "realtek: Enabling init ASM_ID=0x%04x CODEC_ID=%08x\n",
Takashi Iwai7639a062015-03-03 10:07:24 +0100642 ass & 0xffff, codec->core.vendor_id);
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200643 /*
644 * 0 : override
645 * 1 : Swap Jack
646 * 2 : 0 --> Desktop, 1 --> Laptop
647 * 3~5 : External Amplifier control
648 * 7~6 : Reserved
649 */
650 tmp = (ass & 0x38) >> 3; /* external Amp control */
651 switch (tmp) {
652 case 1:
653 spec->init_amp = ALC_INIT_GPIO1;
654 break;
655 case 3:
656 spec->init_amp = ALC_INIT_GPIO2;
657 break;
658 case 7:
659 spec->init_amp = ALC_INIT_GPIO3;
660 break;
661 case 5:
Takashi Iwai5a8cfb42010-11-26 17:11:18 +0100662 default:
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200663 spec->init_amp = ALC_INIT_DEFAULT;
664 break;
665 }
666
667 /* is laptop or Desktop and enable the function "Mute internal speaker
668 * when the external headphone out jack is plugged"
669 */
670 if (!(ass & 0x8000))
671 return 1;
672 /*
673 * 10~8 : Jack location
674 * 12~11: Headphone out -> 00: PortA, 01: PortE, 02: PortD, 03: Resvered
675 * 14~13: Resvered
676 * 15 : 1 --> enable the function "Mute internal speaker
677 * when the external headphone out jack is plugged"
678 */
Takashi Iwai08c189f2012-12-19 15:22:24 +0100679 if (!spec->gen.autocfg.hp_pins[0] &&
680 !(spec->gen.autocfg.line_out_pins[0] &&
681 spec->gen.autocfg.line_out_type == AUTO_PIN_HP_OUT)) {
Takashi Iwai01d48252009-10-06 13:21:54 +0200682 hda_nid_t nid;
Kailang Yangc9b58002007-10-16 14:30:01 +0200683 tmp = (ass >> 11) & 0x3; /* HP to chassis */
Takashi Iwai58c57cf2014-01-07 18:22:49 +0100684 nid = ports[tmp];
Takashi Iwai08c189f2012-12-19 15:22:24 +0100685 if (found_in_nid_list(nid, spec->gen.autocfg.line_out_pins,
686 spec->gen.autocfg.line_outs))
Takashi Iwai3af9ee62011-06-27 12:34:01 +0200687 return 1;
Takashi Iwai08c189f2012-12-19 15:22:24 +0100688 spec->gen.autocfg.hp_pins[0] = nid;
Kailang Yangc9b58002007-10-16 14:30:01 +0200689 }
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200690 return 1;
691}
Kailang Yangea1fb292008-08-26 12:58:38 +0200692
Takashi Iwai3e6179b2011-07-08 16:55:13 +0200693/* Check the validity of ALC subsystem-id
694 * ports contains an array of 4 pin NIDs for port-A, E, D and I */
695static void alc_ssid_check(struct hda_codec *codec, const hda_nid_t *ports)
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200696{
Takashi Iwai58c57cf2014-01-07 18:22:49 +0100697 if (!alc_subsystem_id(codec, ports)) {
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200698 struct alc_spec *spec = codec->spec;
Takashi Iwai4e76a882014-02-25 12:21:03 +0100699 codec_dbg(codec,
700 "realtek: Enable default setup for auto mode as fallback\n");
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200701 spec->init_amp = ALC_INIT_DEFAULT;
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200702 }
Takashi Iwai21268962011-07-07 15:01:13 +0200703}
Takashi Iwai1a1455d2011-04-28 17:36:18 +0200704
Takashi Iwai41e41f12005-06-08 14:48:49 +0200705/*
Takashi Iwai1d045db2011-07-07 18:23:21 +0200706 */
707
David Henningsson9d36a7d2014-10-07 10:18:42 +0200708static void alc_fixup_inv_dmic(struct hda_codec *codec,
709 const struct hda_fixup *fix, int action)
Takashi Iwai125821a2012-06-22 14:30:29 +0200710{
711 struct alc_spec *spec = codec->spec;
Takashi Iwai668d1e92012-11-29 14:10:17 +0100712
David Henningsson9d36a7d2014-10-07 10:18:42 +0200713 spec->gen.inv_dmic_split = 1;
Takashi Iwai6e72aa52012-06-25 10:52:25 +0200714}
715
Takashi Iwai603c4012008-07-30 15:01:44 +0200716
Takashi Iwai67d634c2009-11-16 15:35:59 +0100717#ifdef CONFIG_SND_HDA_INPUT_BEEP
Takashi Iwai45bdd1c2009-02-06 16:11:25 +0100718/* additional beep mixers; the actual parameters are overwritten at build */
Takashi Iwaia9111322011-05-02 11:30:18 +0200719static const struct snd_kcontrol_new alc_beep_mixer[] = {
Takashi Iwai45bdd1c2009-02-06 16:11:25 +0100720 HDA_CODEC_VOLUME("Beep Playback Volume", 0, 0, HDA_INPUT),
Jaroslav Kysela123c07a2009-10-21 14:48:23 +0200721 HDA_CODEC_MUTE_BEEP("Beep Playback Switch", 0, 0, HDA_INPUT),
Takashi Iwai45bdd1c2009-02-06 16:11:25 +0100722 { } /* end */
723};
Takashi Iwai67d634c2009-11-16 15:35:59 +0100724#endif
Takashi Iwai45bdd1c2009-02-06 16:11:25 +0100725
Takashi Iwai2eab6942012-12-18 15:30:41 +0100726static int alc_build_controls(struct hda_codec *codec)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700727{
728 struct alc_spec *spec = codec->spec;
Takashi Iwai666a70d2012-12-17 20:29:29 +0100729 int i, err;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700730
Takashi Iwai08c189f2012-12-19 15:22:24 +0100731 err = snd_hda_gen_build_controls(codec);
732 if (err < 0)
733 return err;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700734
735 for (i = 0; i < spec->num_mixers; i++) {
736 err = snd_hda_add_new_ctls(codec, spec->mixers[i]);
737 if (err < 0)
738 return err;
739 }
Takashi Iwai2134ea42008-01-10 16:53:55 +0100740
Takashi Iwai67d634c2009-11-16 15:35:59 +0100741#ifdef CONFIG_SND_HDA_INPUT_BEEP
Takashi Iwai45bdd1c2009-02-06 16:11:25 +0100742 /* create beep controls if needed */
743 if (spec->beep_amp) {
Takashi Iwaia9111322011-05-02 11:30:18 +0200744 const struct snd_kcontrol_new *knew;
Takashi Iwai45bdd1c2009-02-06 16:11:25 +0100745 for (knew = alc_beep_mixer; knew->name; knew++) {
746 struct snd_kcontrol *kctl;
747 kctl = snd_ctl_new1(knew, codec);
748 if (!kctl)
749 return -ENOMEM;
750 kctl->private_value = spec->beep_amp;
Jaroslav Kysela5e26dfd2009-12-10 13:57:01 +0100751 err = snd_hda_ctl_add(codec, 0, kctl);
Takashi Iwai45bdd1c2009-02-06 16:11:25 +0100752 if (err < 0)
753 return err;
754 }
755 }
Takashi Iwai67d634c2009-11-16 15:35:59 +0100756#endif
Takashi Iwai45bdd1c2009-02-06 16:11:25 +0100757
Takashi Iwai1727a772013-01-10 09:52:52 +0100758 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_BUILD);
Takashi Iwai420b0fe2012-03-12 12:35:27 +0100759 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700760}
761
Takashi Iwaie9edcee2005-06-13 14:16:38 +0200762
Linus Torvalds1da177e2005-04-16 15:20:36 -0700763/*
Takashi Iwaiae6b8132006-03-03 16:47:17 +0100764 * Common callbacks
Takashi Iwaie9edcee2005-06-13 14:16:38 +0200765 */
Takashi Iwai16ded522005-06-10 19:58:24 +0200766
Linus Torvalds1da177e2005-04-16 15:20:36 -0700767static int alc_init(struct hda_codec *codec)
768{
769 struct alc_spec *spec = codec->spec;
Takashi Iwaie9edcee2005-06-13 14:16:38 +0200770
Takashi Iwai546bb672012-03-07 08:37:19 +0100771 if (spec->init_hook)
772 spec->init_hook(codec);
Kailang Yang526af6e2012-03-07 08:25:20 +0100773
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +0200774 alc_fix_pll(codec);
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200775 alc_auto_init_amp(codec, spec->init_amp);
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +0200776
Takashi Iwai08c189f2012-12-19 15:22:24 +0100777 snd_hda_gen_init(codec);
Takashi Iwaiae6b8132006-03-03 16:47:17 +0100778
Takashi Iwai1727a772013-01-10 09:52:52 +0100779 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_INIT);
Takashi Iwaie08a0072006-09-07 17:52:14 +0200780
Linus Torvalds1da177e2005-04-16 15:20:36 -0700781 return 0;
782}
783
Takashi Iwaia4e09aa2009-12-27 11:22:24 +0100784static inline void alc_shutup(struct hda_codec *codec)
785{
Takashi Iwai1c716152011-04-07 10:37:16 +0200786 struct alc_spec *spec = codec->spec;
787
788 if (spec && spec->shutup)
789 spec->shutup(codec);
Takashi Iwai9bfb2842013-07-24 14:31:50 +0200790 else
791 snd_hda_shutup_pins(codec);
Takashi Iwaia4e09aa2009-12-27 11:22:24 +0100792}
793
Takashi Iwai70a09762015-12-15 14:59:58 +0100794static void alc_reboot_notify(struct hda_codec *codec)
795{
796 struct alc_spec *spec = codec->spec;
797
798 if (spec && spec->reboot_notify)
799 spec->reboot_notify(codec);
800 else
801 alc_shutup(codec);
802}
803
804/* power down codec to D3 at reboot/shutdown; set as reboot_notify ops */
805static void alc_d3_at_reboot(struct hda_codec *codec)
806{
807 snd_hda_codec_set_power_to_all(codec, codec->core.afg, AC_PWRST_D3);
808 snd_hda_codec_write(codec, codec->core.afg, 0,
809 AC_VERB_SET_POWER_STATE, AC_PWRST_D3);
810 msleep(10);
811}
812
Takashi Iwai8a02c0c2014-02-10 18:09:45 +0100813#define alc_free snd_hda_gen_free
Linus Torvalds1da177e2005-04-16 15:20:36 -0700814
Takashi Iwai83012a72012-08-24 18:38:08 +0200815#ifdef CONFIG_PM
Daniel T Chenc97259d2009-12-27 18:52:08 -0500816static void alc_power_eapd(struct hda_codec *codec)
817{
Takashi Iwai691f1fc2011-04-07 10:31:43 +0200818 alc_auto_setup_eapd(codec, false);
Daniel T Chenc97259d2009-12-27 18:52:08 -0500819}
820
Takashi Iwai68cb2b52012-07-02 15:20:37 +0200821static int alc_suspend(struct hda_codec *codec)
Hector Martinf5de24b2009-12-20 22:51:31 +0100822{
823 struct alc_spec *spec = codec->spec;
Takashi Iwaia4e09aa2009-12-27 11:22:24 +0100824 alc_shutup(codec);
Hector Martinf5de24b2009-12-20 22:51:31 +0100825 if (spec && spec->power_hook)
Daniel T Chenc97259d2009-12-27 18:52:08 -0500826 spec->power_hook(codec);
Hector Martinf5de24b2009-12-20 22:51:31 +0100827 return 0;
828}
829#endif
830
Takashi Iwai2a439522011-07-26 09:52:50 +0200831#ifdef CONFIG_PM
Takashi Iwaie044c392008-10-27 16:56:24 +0100832static int alc_resume(struct hda_codec *codec)
833{
Kailang Yang97a26572013-11-29 00:35:26 -0500834 struct alc_spec *spec = codec->spec;
835
836 if (!spec->no_depop_delay)
837 msleep(150); /* to avoid pop noise */
Takashi Iwaie044c392008-10-27 16:56:24 +0100838 codec->patch_ops.init(codec);
Takashi Iwaieeecd9d2015-02-25 15:18:50 +0100839 regcache_sync(codec->core.regmap);
Takashi Iwai9e5341b2010-09-21 09:57:06 +0200840 hda_call_check_power_status(codec, 0x01);
Takashi Iwaie044c392008-10-27 16:56:24 +0100841 return 0;
842}
Takashi Iwaie044c392008-10-27 16:56:24 +0100843#endif
844
Linus Torvalds1da177e2005-04-16 15:20:36 -0700845/*
846 */
Takashi Iwaia9111322011-05-02 11:30:18 +0200847static const struct hda_codec_ops alc_patch_ops = {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700848 .build_controls = alc_build_controls,
Takashi Iwai08c189f2012-12-19 15:22:24 +0100849 .build_pcms = snd_hda_gen_build_pcms,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700850 .init = alc_init,
851 .free = alc_free,
David Henningsson29adc4b2012-09-25 11:31:00 +0200852 .unsol_event = snd_hda_jack_unsol_event,
Takashi Iwai2a439522011-07-26 09:52:50 +0200853#ifdef CONFIG_PM
Takashi Iwaie044c392008-10-27 16:56:24 +0100854 .resume = alc_resume,
Hector Martinf5de24b2009-12-20 22:51:31 +0100855 .suspend = alc_suspend,
Takashi Iwaifce52a32013-01-07 12:42:48 +0100856 .check_power_status = snd_hda_gen_check_power_status,
Takashi Iwaicb53c622007-08-10 17:21:45 +0200857#endif
Takashi Iwai70a09762015-12-15 14:59:58 +0100858 .reboot_notify = alc_reboot_notify,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700859};
860
David Henningsson29adc4b2012-09-25 11:31:00 +0200861
Takashi Iwaided255b2015-10-01 17:59:43 +0200862#define alc_codec_rename(codec, name) snd_hda_codec_set_name(codec, name)
Kailang Yangc027ddc2010-03-19 11:33:06 +0100863
Takashi Iwai2fa522b2005-05-12 14:51:12 +0200864/*
Kailang Yang4b016932013-11-28 11:55:09 +0100865 * Rename codecs appropriately from COEF value or subvendor id
Takashi Iwaie16fb6d2011-10-17 16:39:09 +0200866 */
867struct alc_codec_rename_table {
868 unsigned int vendor_id;
869 unsigned short coef_mask;
870 unsigned short coef_bits;
871 const char *name;
872};
873
Kailang Yang4b016932013-11-28 11:55:09 +0100874struct alc_codec_rename_pci_table {
875 unsigned int codec_vendor_id;
876 unsigned short pci_subvendor;
877 unsigned short pci_subdevice;
878 const char *name;
879};
880
Takashi Iwaie16fb6d2011-10-17 16:39:09 +0200881static struct alc_codec_rename_table rename_tbl[] = {
Kailang Yange6e5f7a2014-06-06 16:46:43 +0800882 { 0x10ec0221, 0xf00f, 0x1003, "ALC231" },
Takashi Iwaie16fb6d2011-10-17 16:39:09 +0200883 { 0x10ec0269, 0xfff0, 0x3010, "ALC277" },
884 { 0x10ec0269, 0xf0f0, 0x2010, "ALC259" },
885 { 0x10ec0269, 0xf0f0, 0x3010, "ALC258" },
886 { 0x10ec0269, 0x00f0, 0x0010, "ALC269VB" },
887 { 0x10ec0269, 0xffff, 0xa023, "ALC259" },
888 { 0x10ec0269, 0xffff, 0x6023, "ALC281X" },
889 { 0x10ec0269, 0x00f0, 0x0020, "ALC269VC" },
Kailang Yangadcc70b2012-05-25 08:08:38 +0200890 { 0x10ec0269, 0x00f0, 0x0030, "ALC269VD" },
Kailang Yange6e5f7a2014-06-06 16:46:43 +0800891 { 0x10ec0662, 0xffff, 0x4020, "ALC656" },
Takashi Iwaie16fb6d2011-10-17 16:39:09 +0200892 { 0x10ec0887, 0x00f0, 0x0030, "ALC887-VD" },
893 { 0x10ec0888, 0x00f0, 0x0030, "ALC888-VD" },
894 { 0x10ec0888, 0xf0f0, 0x3020, "ALC886" },
895 { 0x10ec0899, 0x2000, 0x2000, "ALC899" },
896 { 0x10ec0892, 0xffff, 0x8020, "ALC661" },
897 { 0x10ec0892, 0xffff, 0x8011, "ALC661" },
898 { 0x10ec0892, 0xffff, 0x4011, "ALC656" },
899 { } /* terminator */
900};
901
Kailang Yang4b016932013-11-28 11:55:09 +0100902static struct alc_codec_rename_pci_table rename_pci_tbl[] = {
903 { 0x10ec0280, 0x1028, 0, "ALC3220" },
904 { 0x10ec0282, 0x1028, 0, "ALC3221" },
905 { 0x10ec0283, 0x1028, 0, "ALC3223" },
Kailang Yang193177d2014-04-23 16:06:13 +0800906 { 0x10ec0288, 0x1028, 0, "ALC3263" },
Kailang Yang4b016932013-11-28 11:55:09 +0100907 { 0x10ec0292, 0x1028, 0, "ALC3226" },
Kailang Yang193177d2014-04-23 16:06:13 +0800908 { 0x10ec0293, 0x1028, 0, "ALC3235" },
Kailang Yang4b016932013-11-28 11:55:09 +0100909 { 0x10ec0255, 0x1028, 0, "ALC3234" },
910 { 0x10ec0668, 0x1028, 0, "ALC3661" },
Kailang Yange6e5f7a2014-06-06 16:46:43 +0800911 { 0x10ec0275, 0x1028, 0, "ALC3260" },
912 { 0x10ec0899, 0x1028, 0, "ALC3861" },
Kailang Yang2c674fa2015-05-05 15:02:42 +0800913 { 0x10ec0298, 0x1028, 0, "ALC3266" },
Kailang Yang61ae3fb2017-10-20 15:06:34 +0800914 { 0x10ec0236, 0x1028, 0, "ALC3204" },
Kailang Yang82324502015-05-25 17:16:49 +0800915 { 0x10ec0256, 0x1028, 0, "ALC3246" },
Kailang Yang42314302016-02-03 15:03:50 +0800916 { 0x10ec0225, 0x1028, 0, "ALC3253" },
Kailang Yang7d727862016-05-24 16:46:07 +0800917 { 0x10ec0295, 0x1028, 0, "ALC3254" },
Kailang Yangf6e94c22017-01-04 14:49:07 +0800918 { 0x10ec0299, 0x1028, 0, "ALC3271" },
Kailang Yange6e5f7a2014-06-06 16:46:43 +0800919 { 0x10ec0670, 0x1025, 0, "ALC669X" },
920 { 0x10ec0676, 0x1025, 0, "ALC679X" },
921 { 0x10ec0282, 0x1043, 0, "ALC3229" },
922 { 0x10ec0233, 0x1043, 0, "ALC3236" },
923 { 0x10ec0280, 0x103c, 0, "ALC3228" },
924 { 0x10ec0282, 0x103c, 0, "ALC3227" },
925 { 0x10ec0286, 0x103c, 0, "ALC3242" },
926 { 0x10ec0290, 0x103c, 0, "ALC3241" },
927 { 0x10ec0668, 0x103c, 0, "ALC3662" },
928 { 0x10ec0283, 0x17aa, 0, "ALC3239" },
929 { 0x10ec0292, 0x17aa, 0, "ALC3232" },
Kailang Yang4b016932013-11-28 11:55:09 +0100930 { } /* terminator */
931};
932
Takashi Iwaie16fb6d2011-10-17 16:39:09 +0200933static int alc_codec_rename_from_preset(struct hda_codec *codec)
934{
935 const struct alc_codec_rename_table *p;
Kailang Yang4b016932013-11-28 11:55:09 +0100936 const struct alc_codec_rename_pci_table *q;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +0200937
938 for (p = rename_tbl; p->vendor_id; p++) {
Takashi Iwai7639a062015-03-03 10:07:24 +0100939 if (p->vendor_id != codec->core.vendor_id)
Takashi Iwaie16fb6d2011-10-17 16:39:09 +0200940 continue;
Takashi Iwai1bb7e432011-10-17 16:50:59 +0200941 if ((alc_get_coef0(codec) & p->coef_mask) == p->coef_bits)
Takashi Iwaie16fb6d2011-10-17 16:39:09 +0200942 return alc_codec_rename(codec, p->name);
943 }
Kailang Yang4b016932013-11-28 11:55:09 +0100944
Takashi Iwai5100cd02014-02-15 10:03:19 +0100945 if (!codec->bus->pci)
946 return 0;
Kailang Yang4b016932013-11-28 11:55:09 +0100947 for (q = rename_pci_tbl; q->codec_vendor_id; q++) {
Takashi Iwai7639a062015-03-03 10:07:24 +0100948 if (q->codec_vendor_id != codec->core.vendor_id)
Kailang Yang4b016932013-11-28 11:55:09 +0100949 continue;
950 if (q->pci_subvendor != codec->bus->pci->subsystem_vendor)
951 continue;
952 if (!q->pci_subdevice ||
953 q->pci_subdevice == codec->bus->pci->subsystem_device)
954 return alc_codec_rename(codec, q->name);
955 }
956
Takashi Iwaie16fb6d2011-10-17 16:39:09 +0200957 return 0;
958}
959
Takashi Iwaie4770622011-07-08 11:11:35 +0200960
961/*
Takashi Iwai1d045db2011-07-07 18:23:21 +0200962 * Digital-beep handlers
963 */
964#ifdef CONFIG_SND_HDA_INPUT_BEEP
965#define set_beep_amp(spec, nid, idx, dir) \
966 ((spec)->beep_amp = HDA_COMPOSE_AMP_VAL(nid, 3, idx, dir))
967
968static const struct snd_pci_quirk beep_white_list[] = {
Duncan Roe71100052012-10-10 14:19:50 +0200969 SND_PCI_QUIRK(0x1043, 0x103c, "ASUS", 1),
W. Trevor Kinga4b7f212014-03-29 17:47:24 -0700970 SND_PCI_QUIRK(0x1043, 0x115d, "ASUS", 1),
Takashi Iwai1d045db2011-07-07 18:23:21 +0200971 SND_PCI_QUIRK(0x1043, 0x829f, "ASUS", 1),
Takashi Iwai8554ee42013-02-25 09:54:43 +0100972 SND_PCI_QUIRK(0x1043, 0x8376, "EeePC", 1),
Takashi Iwai1d045db2011-07-07 18:23:21 +0200973 SND_PCI_QUIRK(0x1043, 0x83ce, "EeePC", 1),
974 SND_PCI_QUIRK(0x1043, 0x831a, "EeePC", 1),
975 SND_PCI_QUIRK(0x1043, 0x834a, "EeePC", 1),
Takashi Iwai78f8baf2012-03-06 14:02:32 +0100976 SND_PCI_QUIRK(0x1458, 0xa002, "GA-MA790X", 1),
Takashi Iwai1d045db2011-07-07 18:23:21 +0200977 SND_PCI_QUIRK(0x8086, 0xd613, "Intel", 1),
978 {}
979};
980
981static inline int has_cdefine_beep(struct hda_codec *codec)
982{
983 struct alc_spec *spec = codec->spec;
984 const struct snd_pci_quirk *q;
985 q = snd_pci_quirk_lookup(codec->bus->pci, beep_white_list);
986 if (q)
987 return q->value;
988 return spec->cdefine.enable_pcbeep;
989}
990#else
991#define set_beep_amp(spec, nid, idx, dir) /* NOP */
992#define has_cdefine_beep(codec) 0
993#endif
994
995/* parse the BIOS configuration and set up the alc_spec */
996/* return 1 if successful, 0 if the proper config is not found,
997 * or a negative error code
998 */
Takashi Iwai3e6179b2011-07-08 16:55:13 +0200999static int alc_parse_auto_config(struct hda_codec *codec,
1000 const hda_nid_t *ignore_nids,
1001 const hda_nid_t *ssid_nids)
Takashi Iwai1d045db2011-07-07 18:23:21 +02001002{
1003 struct alc_spec *spec = codec->spec;
Takashi Iwai08c189f2012-12-19 15:22:24 +01001004 struct auto_pin_cfg *cfg = &spec->gen.autocfg;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001005 int err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001006
Takashi Iwai53c334a2011-08-23 18:27:14 +02001007 err = snd_hda_parse_pin_defcfg(codec, cfg, ignore_nids,
1008 spec->parse_flags);
Takashi Iwai1d045db2011-07-07 18:23:21 +02001009 if (err < 0)
1010 return err;
Takashi Iwai3e6179b2011-07-08 16:55:13 +02001011
1012 if (ssid_nids)
1013 alc_ssid_check(codec, ssid_nids);
1014
Takashi Iwai08c189f2012-12-19 15:22:24 +01001015 err = snd_hda_gen_parse_auto_config(codec, cfg);
1016 if (err < 0)
1017 return err;
Takashi Iwai070cff42012-02-21 12:54:17 +01001018
Takashi Iwai1d045db2011-07-07 18:23:21 +02001019 return 1;
1020}
1021
Takashi Iwai3de95172012-05-07 18:03:15 +02001022/* common preparation job for alc_spec */
1023static int alc_alloc_spec(struct hda_codec *codec, hda_nid_t mixer_nid)
1024{
1025 struct alc_spec *spec = kzalloc(sizeof(*spec), GFP_KERNEL);
1026 int err;
1027
1028 if (!spec)
1029 return -ENOMEM;
1030 codec->spec = spec;
Takashi Iwai08c189f2012-12-19 15:22:24 +01001031 snd_hda_gen_spec_init(&spec->gen);
1032 spec->gen.mixer_nid = mixer_nid;
1033 spec->gen.own_eapd_ctl = 1;
Takashi Iwai1098b7c2012-12-17 20:03:15 +01001034 codec->single_adc_amp = 1;
Takashi Iwai08c189f2012-12-19 15:22:24 +01001035 /* FIXME: do we need this for all Realtek codec models? */
1036 codec->spdif_status_reset = 1;
Takashi Iwai225068a2015-05-29 10:42:14 +02001037 codec->patch_ops = alc_patch_ops;
Takashi Iwai3de95172012-05-07 18:03:15 +02001038
1039 err = alc_codec_rename_from_preset(codec);
1040 if (err < 0) {
1041 kfree(spec);
1042 return err;
1043 }
1044 return 0;
1045}
1046
Takashi Iwai3e6179b2011-07-08 16:55:13 +02001047static int alc880_parse_auto_config(struct hda_codec *codec)
1048{
1049 static const hda_nid_t alc880_ignore[] = { 0x1d, 0 };
Jesper Juhl7d7eb9e2012-04-12 22:11:25 +02001050 static const hda_nid_t alc880_ssids[] = { 0x15, 0x1b, 0x14, 0 };
Takashi Iwai3e6179b2011-07-08 16:55:13 +02001051 return alc_parse_auto_config(codec, alc880_ignore, alc880_ssids);
1052}
1053
Takashi Iwai1d045db2011-07-07 18:23:21 +02001054/*
Takashi Iwaiee3b2962011-11-15 14:26:54 +01001055 * ALC880 fix-ups
1056 */
1057enum {
Takashi Iwai411225a2012-02-20 17:48:19 +01001058 ALC880_FIXUP_GPIO1,
Takashi Iwaiee3b2962011-11-15 14:26:54 +01001059 ALC880_FIXUP_GPIO2,
1060 ALC880_FIXUP_MEDION_RIM,
Takashi Iwaidc6af522012-02-17 16:18:59 +01001061 ALC880_FIXUP_LG,
Takashi Iwaidb8a38e2013-08-09 12:34:42 +02001062 ALC880_FIXUP_LG_LW25,
Takashi Iwaif02aab52012-02-17 16:33:56 +01001063 ALC880_FIXUP_W810,
Takashi Iwai27e917f2012-02-17 17:49:54 +01001064 ALC880_FIXUP_EAPD_COEF,
Takashi Iwaib9368f52012-02-17 17:54:44 +01001065 ALC880_FIXUP_TCL_S700,
Takashi Iwaicf5a2272012-02-20 16:31:07 +01001066 ALC880_FIXUP_VOL_KNOB,
1067 ALC880_FIXUP_FUJITSU,
Takashi Iwaiba533812012-02-20 16:36:52 +01001068 ALC880_FIXUP_F1734,
Takashi Iwai817de922012-02-20 17:20:48 +01001069 ALC880_FIXUP_UNIWILL,
Takashi Iwai967b88c2012-02-20 17:31:02 +01001070 ALC880_FIXUP_UNIWILL_DIG,
Takashi Iwai96e225f2012-02-20 17:41:51 +01001071 ALC880_FIXUP_Z71V,
Takashi Iwai487a5882013-11-07 07:29:30 +01001072 ALC880_FIXUP_ASUS_W5A,
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001073 ALC880_FIXUP_3ST_BASE,
1074 ALC880_FIXUP_3ST,
1075 ALC880_FIXUP_3ST_DIG,
1076 ALC880_FIXUP_5ST_BASE,
1077 ALC880_FIXUP_5ST,
1078 ALC880_FIXUP_5ST_DIG,
1079 ALC880_FIXUP_6ST_BASE,
1080 ALC880_FIXUP_6ST,
1081 ALC880_FIXUP_6ST_DIG,
Takashi Iwai53971452013-01-23 18:21:37 +01001082 ALC880_FIXUP_6ST_AUTOMUTE,
Takashi Iwaiee3b2962011-11-15 14:26:54 +01001083};
1084
Takashi Iwaicf5a2272012-02-20 16:31:07 +01001085/* enable the volume-knob widget support on NID 0x21 */
1086static void alc880_fixup_vol_knob(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01001087 const struct hda_fixup *fix, int action)
Takashi Iwaicf5a2272012-02-20 16:31:07 +01001088{
Takashi Iwai1727a772013-01-10 09:52:52 +01001089 if (action == HDA_FIXUP_ACT_PROBE)
Takashi Iwai62f949b2014-09-11 14:06:53 +02001090 snd_hda_jack_detect_enable_callback(codec, 0x21,
1091 alc_update_knob_master);
Takashi Iwaicf5a2272012-02-20 16:31:07 +01001092}
1093
Takashi Iwai1727a772013-01-10 09:52:52 +01001094static const struct hda_fixup alc880_fixups[] = {
Takashi Iwai411225a2012-02-20 17:48:19 +01001095 [ALC880_FIXUP_GPIO1] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001096 .type = HDA_FIXUP_VERBS,
Takashi Iwai411225a2012-02-20 17:48:19 +01001097 .v.verbs = alc_gpio1_init_verbs,
1098 },
Takashi Iwaiee3b2962011-11-15 14:26:54 +01001099 [ALC880_FIXUP_GPIO2] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001100 .type = HDA_FIXUP_VERBS,
Takashi Iwaiee3b2962011-11-15 14:26:54 +01001101 .v.verbs = alc_gpio2_init_verbs,
1102 },
1103 [ALC880_FIXUP_MEDION_RIM] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001104 .type = HDA_FIXUP_VERBS,
Takashi Iwaiee3b2962011-11-15 14:26:54 +01001105 .v.verbs = (const struct hda_verb[]) {
1106 { 0x20, AC_VERB_SET_COEF_INDEX, 0x07 },
1107 { 0x20, AC_VERB_SET_PROC_COEF, 0x3060 },
1108 { }
1109 },
1110 .chained = true,
1111 .chain_id = ALC880_FIXUP_GPIO2,
1112 },
Takashi Iwaidc6af522012-02-17 16:18:59 +01001113 [ALC880_FIXUP_LG] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001114 .type = HDA_FIXUP_PINS,
1115 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaidc6af522012-02-17 16:18:59 +01001116 /* disable bogus unused pins */
1117 { 0x16, 0x411111f0 },
1118 { 0x18, 0x411111f0 },
1119 { 0x1a, 0x411111f0 },
1120 { }
1121 }
1122 },
Takashi Iwaidb8a38e2013-08-09 12:34:42 +02001123 [ALC880_FIXUP_LG_LW25] = {
1124 .type = HDA_FIXUP_PINS,
1125 .v.pins = (const struct hda_pintbl[]) {
1126 { 0x1a, 0x0181344f }, /* line-in */
1127 { 0x1b, 0x0321403f }, /* headphone */
1128 { }
1129 }
1130 },
Takashi Iwaif02aab52012-02-17 16:33:56 +01001131 [ALC880_FIXUP_W810] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001132 .type = HDA_FIXUP_PINS,
1133 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaif02aab52012-02-17 16:33:56 +01001134 /* disable bogus unused pins */
1135 { 0x17, 0x411111f0 },
1136 { }
1137 },
1138 .chained = true,
1139 .chain_id = ALC880_FIXUP_GPIO2,
1140 },
Takashi Iwai27e917f2012-02-17 17:49:54 +01001141 [ALC880_FIXUP_EAPD_COEF] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001142 .type = HDA_FIXUP_VERBS,
Takashi Iwai27e917f2012-02-17 17:49:54 +01001143 .v.verbs = (const struct hda_verb[]) {
1144 /* change to EAPD mode */
1145 { 0x20, AC_VERB_SET_COEF_INDEX, 0x07 },
1146 { 0x20, AC_VERB_SET_PROC_COEF, 0x3060 },
1147 {}
1148 },
1149 },
Takashi Iwaib9368f52012-02-17 17:54:44 +01001150 [ALC880_FIXUP_TCL_S700] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001151 .type = HDA_FIXUP_VERBS,
Takashi Iwaib9368f52012-02-17 17:54:44 +01001152 .v.verbs = (const struct hda_verb[]) {
1153 /* change to EAPD mode */
1154 { 0x20, AC_VERB_SET_COEF_INDEX, 0x07 },
1155 { 0x20, AC_VERB_SET_PROC_COEF, 0x3070 },
1156 {}
1157 },
1158 .chained = true,
1159 .chain_id = ALC880_FIXUP_GPIO2,
1160 },
Takashi Iwaicf5a2272012-02-20 16:31:07 +01001161 [ALC880_FIXUP_VOL_KNOB] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001162 .type = HDA_FIXUP_FUNC,
Takashi Iwaicf5a2272012-02-20 16:31:07 +01001163 .v.func = alc880_fixup_vol_knob,
1164 },
1165 [ALC880_FIXUP_FUJITSU] = {
1166 /* override all pins as BIOS on old Amilo is broken */
Takashi Iwai1727a772013-01-10 09:52:52 +01001167 .type = HDA_FIXUP_PINS,
1168 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaibb148bd2015-08-13 18:02:39 +02001169 { 0x14, 0x0121401f }, /* HP */
Takashi Iwaicf5a2272012-02-20 16:31:07 +01001170 { 0x15, 0x99030120 }, /* speaker */
1171 { 0x16, 0x99030130 }, /* bass speaker */
1172 { 0x17, 0x411111f0 }, /* N/A */
1173 { 0x18, 0x411111f0 }, /* N/A */
1174 { 0x19, 0x01a19950 }, /* mic-in */
1175 { 0x1a, 0x411111f0 }, /* N/A */
1176 { 0x1b, 0x411111f0 }, /* N/A */
1177 { 0x1c, 0x411111f0 }, /* N/A */
1178 { 0x1d, 0x411111f0 }, /* N/A */
1179 { 0x1e, 0x01454140 }, /* SPDIF out */
1180 { }
1181 },
1182 .chained = true,
1183 .chain_id = ALC880_FIXUP_VOL_KNOB,
1184 },
Takashi Iwaiba533812012-02-20 16:36:52 +01001185 [ALC880_FIXUP_F1734] = {
1186 /* almost compatible with FUJITSU, but no bass and SPDIF */
Takashi Iwai1727a772013-01-10 09:52:52 +01001187 .type = HDA_FIXUP_PINS,
1188 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaibb148bd2015-08-13 18:02:39 +02001189 { 0x14, 0x0121401f }, /* HP */
Takashi Iwaiba533812012-02-20 16:36:52 +01001190 { 0x15, 0x99030120 }, /* speaker */
1191 { 0x16, 0x411111f0 }, /* N/A */
1192 { 0x17, 0x411111f0 }, /* N/A */
1193 { 0x18, 0x411111f0 }, /* N/A */
1194 { 0x19, 0x01a19950 }, /* mic-in */
1195 { 0x1a, 0x411111f0 }, /* N/A */
1196 { 0x1b, 0x411111f0 }, /* N/A */
1197 { 0x1c, 0x411111f0 }, /* N/A */
1198 { 0x1d, 0x411111f0 }, /* N/A */
1199 { 0x1e, 0x411111f0 }, /* N/A */
1200 { }
1201 },
1202 .chained = true,
1203 .chain_id = ALC880_FIXUP_VOL_KNOB,
1204 },
Takashi Iwai817de922012-02-20 17:20:48 +01001205 [ALC880_FIXUP_UNIWILL] = {
1206 /* need to fix HP and speaker pins to be parsed correctly */
Takashi Iwai1727a772013-01-10 09:52:52 +01001207 .type = HDA_FIXUP_PINS,
1208 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai817de922012-02-20 17:20:48 +01001209 { 0x14, 0x0121411f }, /* HP */
1210 { 0x15, 0x99030120 }, /* speaker */
1211 { 0x16, 0x99030130 }, /* bass speaker */
1212 { }
1213 },
1214 },
Takashi Iwai967b88c2012-02-20 17:31:02 +01001215 [ALC880_FIXUP_UNIWILL_DIG] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001216 .type = HDA_FIXUP_PINS,
1217 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai967b88c2012-02-20 17:31:02 +01001218 /* disable bogus unused pins */
1219 { 0x17, 0x411111f0 },
1220 { 0x19, 0x411111f0 },
1221 { 0x1b, 0x411111f0 },
1222 { 0x1f, 0x411111f0 },
1223 { }
1224 }
1225 },
Takashi Iwai96e225f2012-02-20 17:41:51 +01001226 [ALC880_FIXUP_Z71V] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001227 .type = HDA_FIXUP_PINS,
1228 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai96e225f2012-02-20 17:41:51 +01001229 /* set up the whole pins as BIOS is utterly broken */
1230 { 0x14, 0x99030120 }, /* speaker */
1231 { 0x15, 0x0121411f }, /* HP */
1232 { 0x16, 0x411111f0 }, /* N/A */
1233 { 0x17, 0x411111f0 }, /* N/A */
1234 { 0x18, 0x01a19950 }, /* mic-in */
1235 { 0x19, 0x411111f0 }, /* N/A */
1236 { 0x1a, 0x01813031 }, /* line-in */
1237 { 0x1b, 0x411111f0 }, /* N/A */
1238 { 0x1c, 0x411111f0 }, /* N/A */
1239 { 0x1d, 0x411111f0 }, /* N/A */
1240 { 0x1e, 0x0144111e }, /* SPDIF */
1241 { }
1242 }
1243 },
Takashi Iwai487a5882013-11-07 07:29:30 +01001244 [ALC880_FIXUP_ASUS_W5A] = {
1245 .type = HDA_FIXUP_PINS,
1246 .v.pins = (const struct hda_pintbl[]) {
1247 /* set up the whole pins as BIOS is utterly broken */
1248 { 0x14, 0x0121411f }, /* HP */
1249 { 0x15, 0x411111f0 }, /* N/A */
1250 { 0x16, 0x411111f0 }, /* N/A */
1251 { 0x17, 0x411111f0 }, /* N/A */
1252 { 0x18, 0x90a60160 }, /* mic */
1253 { 0x19, 0x411111f0 }, /* N/A */
1254 { 0x1a, 0x411111f0 }, /* N/A */
1255 { 0x1b, 0x411111f0 }, /* N/A */
1256 { 0x1c, 0x411111f0 }, /* N/A */
1257 { 0x1d, 0x411111f0 }, /* N/A */
1258 { 0x1e, 0xb743111e }, /* SPDIF out */
1259 { }
1260 },
1261 .chained = true,
1262 .chain_id = ALC880_FIXUP_GPIO1,
1263 },
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001264 [ALC880_FIXUP_3ST_BASE] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001265 .type = HDA_FIXUP_PINS,
1266 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001267 { 0x14, 0x01014010 }, /* line-out */
1268 { 0x15, 0x411111f0 }, /* N/A */
1269 { 0x16, 0x411111f0 }, /* N/A */
1270 { 0x17, 0x411111f0 }, /* N/A */
1271 { 0x18, 0x01a19c30 }, /* mic-in */
1272 { 0x19, 0x0121411f }, /* HP */
1273 { 0x1a, 0x01813031 }, /* line-in */
1274 { 0x1b, 0x02a19c40 }, /* front-mic */
1275 { 0x1c, 0x411111f0 }, /* N/A */
1276 { 0x1d, 0x411111f0 }, /* N/A */
1277 /* 0x1e is filled in below */
1278 { 0x1f, 0x411111f0 }, /* N/A */
1279 { }
1280 }
1281 },
1282 [ALC880_FIXUP_3ST] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001283 .type = HDA_FIXUP_PINS,
1284 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001285 { 0x1e, 0x411111f0 }, /* N/A */
1286 { }
1287 },
1288 .chained = true,
1289 .chain_id = ALC880_FIXUP_3ST_BASE,
1290 },
1291 [ALC880_FIXUP_3ST_DIG] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001292 .type = HDA_FIXUP_PINS,
1293 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001294 { 0x1e, 0x0144111e }, /* SPDIF */
1295 { }
1296 },
1297 .chained = true,
1298 .chain_id = ALC880_FIXUP_3ST_BASE,
1299 },
1300 [ALC880_FIXUP_5ST_BASE] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001301 .type = HDA_FIXUP_PINS,
1302 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001303 { 0x14, 0x01014010 }, /* front */
1304 { 0x15, 0x411111f0 }, /* N/A */
1305 { 0x16, 0x01011411 }, /* CLFE */
1306 { 0x17, 0x01016412 }, /* surr */
1307 { 0x18, 0x01a19c30 }, /* mic-in */
1308 { 0x19, 0x0121411f }, /* HP */
1309 { 0x1a, 0x01813031 }, /* line-in */
1310 { 0x1b, 0x02a19c40 }, /* front-mic */
1311 { 0x1c, 0x411111f0 }, /* N/A */
1312 { 0x1d, 0x411111f0 }, /* N/A */
1313 /* 0x1e is filled in below */
1314 { 0x1f, 0x411111f0 }, /* N/A */
1315 { }
1316 }
1317 },
1318 [ALC880_FIXUP_5ST] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001319 .type = HDA_FIXUP_PINS,
1320 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001321 { 0x1e, 0x411111f0 }, /* N/A */
1322 { }
1323 },
1324 .chained = true,
1325 .chain_id = ALC880_FIXUP_5ST_BASE,
1326 },
1327 [ALC880_FIXUP_5ST_DIG] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001328 .type = HDA_FIXUP_PINS,
1329 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001330 { 0x1e, 0x0144111e }, /* SPDIF */
1331 { }
1332 },
1333 .chained = true,
1334 .chain_id = ALC880_FIXUP_5ST_BASE,
1335 },
1336 [ALC880_FIXUP_6ST_BASE] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001337 .type = HDA_FIXUP_PINS,
1338 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001339 { 0x14, 0x01014010 }, /* front */
1340 { 0x15, 0x01016412 }, /* surr */
1341 { 0x16, 0x01011411 }, /* CLFE */
1342 { 0x17, 0x01012414 }, /* side */
1343 { 0x18, 0x01a19c30 }, /* mic-in */
1344 { 0x19, 0x02a19c40 }, /* front-mic */
1345 { 0x1a, 0x01813031 }, /* line-in */
1346 { 0x1b, 0x0121411f }, /* HP */
1347 { 0x1c, 0x411111f0 }, /* N/A */
1348 { 0x1d, 0x411111f0 }, /* N/A */
1349 /* 0x1e is filled in below */
1350 { 0x1f, 0x411111f0 }, /* N/A */
1351 { }
1352 }
1353 },
1354 [ALC880_FIXUP_6ST] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001355 .type = HDA_FIXUP_PINS,
1356 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001357 { 0x1e, 0x411111f0 }, /* N/A */
1358 { }
1359 },
1360 .chained = true,
1361 .chain_id = ALC880_FIXUP_6ST_BASE,
1362 },
1363 [ALC880_FIXUP_6ST_DIG] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001364 .type = HDA_FIXUP_PINS,
1365 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001366 { 0x1e, 0x0144111e }, /* SPDIF */
1367 { }
1368 },
1369 .chained = true,
1370 .chain_id = ALC880_FIXUP_6ST_BASE,
1371 },
Takashi Iwai53971452013-01-23 18:21:37 +01001372 [ALC880_FIXUP_6ST_AUTOMUTE] = {
1373 .type = HDA_FIXUP_PINS,
1374 .v.pins = (const struct hda_pintbl[]) {
1375 { 0x1b, 0x0121401f }, /* HP with jack detect */
1376 { }
1377 },
1378 .chained_before = true,
1379 .chain_id = ALC880_FIXUP_6ST_BASE,
1380 },
Takashi Iwaiee3b2962011-11-15 14:26:54 +01001381};
1382
1383static const struct snd_pci_quirk alc880_fixup_tbl[] = {
Takashi Iwaif02aab52012-02-17 16:33:56 +01001384 SND_PCI_QUIRK(0x1019, 0x0f69, "Coeus G610P", ALC880_FIXUP_W810),
Takashi Iwai487a5882013-11-07 07:29:30 +01001385 SND_PCI_QUIRK(0x1043, 0x10c3, "ASUS W5A", ALC880_FIXUP_ASUS_W5A),
Takashi Iwai96e225f2012-02-20 17:41:51 +01001386 SND_PCI_QUIRK(0x1043, 0x1964, "ASUS Z71V", ALC880_FIXUP_Z71V),
Takashi Iwai29e3fdc2012-02-20 17:56:57 +01001387 SND_PCI_QUIRK_VENDOR(0x1043, "ASUS", ALC880_FIXUP_GPIO1),
David Henningsson6538de02014-06-10 10:52:50 +02001388 SND_PCI_QUIRK(0x147b, 0x1045, "ABit AA8XE", ALC880_FIXUP_6ST_AUTOMUTE),
Takashi Iwai29e3fdc2012-02-20 17:56:57 +01001389 SND_PCI_QUIRK(0x1558, 0x5401, "Clevo GPIO2", ALC880_FIXUP_GPIO2),
Takashi Iwai27e917f2012-02-17 17:49:54 +01001390 SND_PCI_QUIRK_VENDOR(0x1558, "Clevo", ALC880_FIXUP_EAPD_COEF),
Takashi Iwai967b88c2012-02-20 17:31:02 +01001391 SND_PCI_QUIRK(0x1584, 0x9050, "Uniwill", ALC880_FIXUP_UNIWILL_DIG),
Takashi Iwaiba533812012-02-20 16:36:52 +01001392 SND_PCI_QUIRK(0x1584, 0x9054, "Uniwill", ALC880_FIXUP_F1734),
Takashi Iwai817de922012-02-20 17:20:48 +01001393 SND_PCI_QUIRK(0x1584, 0x9070, "Uniwill", ALC880_FIXUP_UNIWILL),
Takashi Iwai7833c7e2012-02-20 17:11:38 +01001394 SND_PCI_QUIRK(0x1584, 0x9077, "Uniwill P53", ALC880_FIXUP_VOL_KNOB),
Takashi Iwaif02aab52012-02-17 16:33:56 +01001395 SND_PCI_QUIRK(0x161f, 0x203d, "W810", ALC880_FIXUP_W810),
Takashi Iwaiee3b2962011-11-15 14:26:54 +01001396 SND_PCI_QUIRK(0x161f, 0x205d, "Medion Rim 2150", ALC880_FIXUP_MEDION_RIM),
Takashi Iwai53971452013-01-23 18:21:37 +01001397 SND_PCI_QUIRK(0x1631, 0xe011, "PB 13201056", ALC880_FIXUP_6ST_AUTOMUTE),
Takashi Iwaia1615742015-08-13 18:05:06 +02001398 SND_PCI_QUIRK(0x1734, 0x107c, "FSC Amilo M1437", ALC880_FIXUP_FUJITSU),
Takashi Iwaicf5a2272012-02-20 16:31:07 +01001399 SND_PCI_QUIRK(0x1734, 0x1094, "FSC Amilo M1451G", ALC880_FIXUP_FUJITSU),
Takashi Iwaiba533812012-02-20 16:36:52 +01001400 SND_PCI_QUIRK(0x1734, 0x10ac, "FSC AMILO Xi 1526", ALC880_FIXUP_F1734),
Takashi Iwaicf5a2272012-02-20 16:31:07 +01001401 SND_PCI_QUIRK(0x1734, 0x10b0, "FSC Amilo Pi1556", ALC880_FIXUP_FUJITSU),
Takashi Iwaidc6af522012-02-17 16:18:59 +01001402 SND_PCI_QUIRK(0x1854, 0x003b, "LG", ALC880_FIXUP_LG),
1403 SND_PCI_QUIRK(0x1854, 0x005f, "LG P1 Express", ALC880_FIXUP_LG),
1404 SND_PCI_QUIRK(0x1854, 0x0068, "LG w1", ALC880_FIXUP_LG),
Takashi Iwaidb8a38e2013-08-09 12:34:42 +02001405 SND_PCI_QUIRK(0x1854, 0x0077, "LG LW25", ALC880_FIXUP_LG_LW25),
Takashi Iwaib9368f52012-02-17 17:54:44 +01001406 SND_PCI_QUIRK(0x19db, 0x4188, "TCL S700", ALC880_FIXUP_TCL_S700),
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001407
1408 /* Below is the copied entries from alc880_quirks.c.
1409 * It's not quite sure whether BIOS sets the correct pin-config table
1410 * on these machines, thus they are kept to be compatible with
1411 * the old static quirks. Once when it's confirmed to work without
1412 * these overrides, it'd be better to remove.
1413 */
1414 SND_PCI_QUIRK(0x1019, 0xa880, "ECS", ALC880_FIXUP_5ST_DIG),
1415 SND_PCI_QUIRK(0x1019, 0xa884, "Acer APFV", ALC880_FIXUP_6ST),
1416 SND_PCI_QUIRK(0x1025, 0x0070, "ULI", ALC880_FIXUP_3ST_DIG),
1417 SND_PCI_QUIRK(0x1025, 0x0077, "ULI", ALC880_FIXUP_6ST_DIG),
1418 SND_PCI_QUIRK(0x1025, 0x0078, "ULI", ALC880_FIXUP_6ST_DIG),
1419 SND_PCI_QUIRK(0x1025, 0x0087, "ULI", ALC880_FIXUP_6ST_DIG),
1420 SND_PCI_QUIRK(0x1025, 0xe309, "ULI", ALC880_FIXUP_3ST_DIG),
1421 SND_PCI_QUIRK(0x1025, 0xe310, "ULI", ALC880_FIXUP_3ST),
1422 SND_PCI_QUIRK(0x1039, 0x1234, NULL, ALC880_FIXUP_6ST_DIG),
1423 SND_PCI_QUIRK(0x104d, 0x81a0, "Sony", ALC880_FIXUP_3ST),
1424 SND_PCI_QUIRK(0x104d, 0x81d6, "Sony", ALC880_FIXUP_3ST),
1425 SND_PCI_QUIRK(0x107b, 0x3032, "Gateway", ALC880_FIXUP_5ST),
1426 SND_PCI_QUIRK(0x107b, 0x3033, "Gateway", ALC880_FIXUP_5ST),
1427 SND_PCI_QUIRK(0x107b, 0x4039, "Gateway", ALC880_FIXUP_5ST),
1428 SND_PCI_QUIRK(0x1297, 0xc790, "Shuttle ST20G5", ALC880_FIXUP_6ST_DIG),
1429 SND_PCI_QUIRK(0x1458, 0xa102, "Gigabyte K8", ALC880_FIXUP_6ST_DIG),
1430 SND_PCI_QUIRK(0x1462, 0x1150, "MSI", ALC880_FIXUP_6ST_DIG),
1431 SND_PCI_QUIRK(0x1509, 0x925d, "FIC P4M", ALC880_FIXUP_6ST_DIG),
1432 SND_PCI_QUIRK(0x1565, 0x8202, "Biostar", ALC880_FIXUP_5ST_DIG),
1433 SND_PCI_QUIRK(0x1695, 0x400d, "EPoX", ALC880_FIXUP_5ST_DIG),
1434 SND_PCI_QUIRK(0x1695, 0x4012, "EPox EP-5LDA", ALC880_FIXUP_5ST_DIG),
1435 SND_PCI_QUIRK(0x2668, 0x8086, NULL, ALC880_FIXUP_6ST_DIG), /* broken BIOS */
1436 SND_PCI_QUIRK(0x8086, 0x2668, NULL, ALC880_FIXUP_6ST_DIG),
1437 SND_PCI_QUIRK(0x8086, 0xa100, "Intel mobo", ALC880_FIXUP_5ST_DIG),
1438 SND_PCI_QUIRK(0x8086, 0xd400, "Intel mobo", ALC880_FIXUP_5ST_DIG),
1439 SND_PCI_QUIRK(0x8086, 0xd401, "Intel mobo", ALC880_FIXUP_5ST_DIG),
1440 SND_PCI_QUIRK(0x8086, 0xd402, "Intel mobo", ALC880_FIXUP_3ST_DIG),
1441 SND_PCI_QUIRK(0x8086, 0xe224, "Intel mobo", ALC880_FIXUP_5ST_DIG),
1442 SND_PCI_QUIRK(0x8086, 0xe305, "Intel mobo", ALC880_FIXUP_3ST_DIG),
1443 SND_PCI_QUIRK(0x8086, 0xe308, "Intel mobo", ALC880_FIXUP_3ST_DIG),
1444 SND_PCI_QUIRK(0x8086, 0xe400, "Intel mobo", ALC880_FIXUP_5ST_DIG),
1445 SND_PCI_QUIRK(0x8086, 0xe401, "Intel mobo", ALC880_FIXUP_5ST_DIG),
1446 SND_PCI_QUIRK(0x8086, 0xe402, "Intel mobo", ALC880_FIXUP_5ST_DIG),
1447 /* default Intel */
1448 SND_PCI_QUIRK_VENDOR(0x8086, "Intel mobo", ALC880_FIXUP_3ST),
1449 SND_PCI_QUIRK(0xa0a0, 0x0560, "AOpen i915GMm-HFS", ALC880_FIXUP_5ST_DIG),
1450 SND_PCI_QUIRK(0xe803, 0x1019, NULL, ALC880_FIXUP_6ST_DIG),
1451 {}
1452};
1453
Takashi Iwai1727a772013-01-10 09:52:52 +01001454static const struct hda_model_fixup alc880_fixup_models[] = {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001455 {.id = ALC880_FIXUP_3ST, .name = "3stack"},
1456 {.id = ALC880_FIXUP_3ST_DIG, .name = "3stack-digout"},
1457 {.id = ALC880_FIXUP_5ST, .name = "5stack"},
1458 {.id = ALC880_FIXUP_5ST_DIG, .name = "5stack-digout"},
1459 {.id = ALC880_FIXUP_6ST, .name = "6stack"},
1460 {.id = ALC880_FIXUP_6ST_DIG, .name = "6stack-digout"},
Takashi Iwai53971452013-01-23 18:21:37 +01001461 {.id = ALC880_FIXUP_6ST_AUTOMUTE, .name = "6stack-automute"},
Takashi Iwaiee3b2962011-11-15 14:26:54 +01001462 {}
1463};
1464
1465
1466/*
Takashi Iwai1d045db2011-07-07 18:23:21 +02001467 * OK, here we have finally the patch for ALC880
1468 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02001469static int patch_alc880(struct hda_codec *codec)
1470{
1471 struct alc_spec *spec;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001472 int err;
1473
Takashi Iwai3de95172012-05-07 18:03:15 +02001474 err = alc_alloc_spec(codec, 0x0b);
1475 if (err < 0)
1476 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001477
Takashi Iwai3de95172012-05-07 18:03:15 +02001478 spec = codec->spec;
Takashi Iwai08c189f2012-12-19 15:22:24 +01001479 spec->gen.need_dac_fix = 1;
Takashi Iwai7504b6c2013-03-18 11:25:51 +01001480 spec->gen.beep_nid = 0x01;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001481
Takashi Iwai225068a2015-05-29 10:42:14 +02001482 codec->patch_ops.unsol_event = alc880_unsol_event;
1483
Takashi Iwai1727a772013-01-10 09:52:52 +01001484 snd_hda_pick_fixup(codec, alc880_fixup_models, alc880_fixup_tbl,
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001485 alc880_fixups);
Takashi Iwai1727a772013-01-10 09:52:52 +01001486 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
Takashi Iwai1d045db2011-07-07 18:23:21 +02001487
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001488 /* automatic parse from the BIOS config */
1489 err = alc880_parse_auto_config(codec);
1490 if (err < 0)
1491 goto error;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001492
Takashi Iwai7504b6c2013-03-18 11:25:51 +01001493 if (!spec->gen.no_analog)
Takashi Iwai3e6179b2011-07-08 16:55:13 +02001494 set_beep_amp(spec, 0x0b, 0x05, HDA_INPUT);
Takashi Iwai1d045db2011-07-07 18:23:21 +02001495
Takashi Iwai1727a772013-01-10 09:52:52 +01001496 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
Takashi Iwai589876e2012-02-20 15:47:55 +01001497
Takashi Iwai1d045db2011-07-07 18:23:21 +02001498 return 0;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02001499
1500 error:
1501 alc_free(codec);
1502 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001503}
1504
1505
1506/*
1507 * ALC260 support
1508 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02001509static int alc260_parse_auto_config(struct hda_codec *codec)
1510{
Takashi Iwai1d045db2011-07-07 18:23:21 +02001511 static const hda_nid_t alc260_ignore[] = { 0x17, 0 };
Takashi Iwai3e6179b2011-07-08 16:55:13 +02001512 static const hda_nid_t alc260_ssids[] = { 0x10, 0x15, 0x0f, 0 };
1513 return alc_parse_auto_config(codec, alc260_ignore, alc260_ssids);
Takashi Iwai1d045db2011-07-07 18:23:21 +02001514}
1515
Takashi Iwai1d045db2011-07-07 18:23:21 +02001516/*
1517 * Pin config fixes
1518 */
1519enum {
Takashi Iwaica8f0422012-02-16 11:51:19 +01001520 ALC260_FIXUP_HP_DC5750,
1521 ALC260_FIXUP_HP_PIN_0F,
1522 ALC260_FIXUP_COEF,
Takashi Iwai15317ab2012-02-16 12:02:53 +01001523 ALC260_FIXUP_GPIO1,
Takashi Iwai20f7d922012-02-16 12:35:16 +01001524 ALC260_FIXUP_GPIO1_TOGGLE,
1525 ALC260_FIXUP_REPLACER,
Takashi Iwai0a1c4fa2012-02-16 12:42:30 +01001526 ALC260_FIXUP_HP_B1900,
Takashi Iwai118cb4a2012-04-19 07:33:27 +02001527 ALC260_FIXUP_KN1,
Takashi Iwai39aedee2013-01-10 17:10:40 +01001528 ALC260_FIXUP_FSC_S7020,
Takashi Iwai5ebd3bb2013-02-19 18:23:31 +01001529 ALC260_FIXUP_FSC_S7020_JWSE,
Takashi Iwaid08c5ef2013-11-22 08:06:36 +01001530 ALC260_FIXUP_VAIO_PINS,
Takashi Iwai1d045db2011-07-07 18:23:21 +02001531};
1532
Takashi Iwai20f7d922012-02-16 12:35:16 +01001533static void alc260_gpio1_automute(struct hda_codec *codec)
1534{
1535 struct alc_spec *spec = codec->spec;
1536 snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DATA,
Takashi Iwai08c189f2012-12-19 15:22:24 +01001537 spec->gen.hp_jack_present);
Takashi Iwai20f7d922012-02-16 12:35:16 +01001538}
1539
1540static void alc260_fixup_gpio1_toggle(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01001541 const struct hda_fixup *fix, int action)
Takashi Iwai20f7d922012-02-16 12:35:16 +01001542{
1543 struct alc_spec *spec = codec->spec;
Takashi Iwai1727a772013-01-10 09:52:52 +01001544 if (action == HDA_FIXUP_ACT_PROBE) {
Takashi Iwai20f7d922012-02-16 12:35:16 +01001545 /* although the machine has only one output pin, we need to
1546 * toggle GPIO1 according to the jack state
1547 */
Takashi Iwai08c189f2012-12-19 15:22:24 +01001548 spec->gen.automute_hook = alc260_gpio1_automute;
1549 spec->gen.detect_hp = 1;
1550 spec->gen.automute_speaker = 1;
1551 spec->gen.autocfg.hp_pins[0] = 0x0f; /* copy it for automute */
Takashi Iwai62f949b2014-09-11 14:06:53 +02001552 snd_hda_jack_detect_enable_callback(codec, 0x0f,
Takashi Iwai08c189f2012-12-19 15:22:24 +01001553 snd_hda_gen_hp_automute);
Takashi Iwaic9ce6b22012-12-18 18:12:44 +01001554 snd_hda_add_verbs(codec, alc_gpio1_init_verbs);
Takashi Iwai20f7d922012-02-16 12:35:16 +01001555 }
1556}
1557
Takashi Iwai118cb4a2012-04-19 07:33:27 +02001558static void alc260_fixup_kn1(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01001559 const struct hda_fixup *fix, int action)
Takashi Iwai118cb4a2012-04-19 07:33:27 +02001560{
1561 struct alc_spec *spec = codec->spec;
Takashi Iwai1727a772013-01-10 09:52:52 +01001562 static const struct hda_pintbl pincfgs[] = {
Takashi Iwai118cb4a2012-04-19 07:33:27 +02001563 { 0x0f, 0x02214000 }, /* HP/speaker */
1564 { 0x12, 0x90a60160 }, /* int mic */
1565 { 0x13, 0x02a19000 }, /* ext mic */
1566 { 0x18, 0x01446000 }, /* SPDIF out */
1567 /* disable bogus I/O pins */
1568 { 0x10, 0x411111f0 },
1569 { 0x11, 0x411111f0 },
1570 { 0x14, 0x411111f0 },
1571 { 0x15, 0x411111f0 },
1572 { 0x16, 0x411111f0 },
1573 { 0x17, 0x411111f0 },
1574 { 0x19, 0x411111f0 },
1575 { }
1576 };
1577
1578 switch (action) {
Takashi Iwai1727a772013-01-10 09:52:52 +01001579 case HDA_FIXUP_ACT_PRE_PROBE:
1580 snd_hda_apply_pincfgs(codec, pincfgs);
Takashi Iwai118cb4a2012-04-19 07:33:27 +02001581 break;
Takashi Iwai1727a772013-01-10 09:52:52 +01001582 case HDA_FIXUP_ACT_PROBE:
Takashi Iwai118cb4a2012-04-19 07:33:27 +02001583 spec->init_amp = ALC_INIT_NONE;
1584 break;
1585 }
1586}
1587
Takashi Iwai39aedee2013-01-10 17:10:40 +01001588static void alc260_fixup_fsc_s7020(struct hda_codec *codec,
1589 const struct hda_fixup *fix, int action)
1590{
1591 struct alc_spec *spec = codec->spec;
Takashi Iwai5ebd3bb2013-02-19 18:23:31 +01001592 if (action == HDA_FIXUP_ACT_PROBE)
Takashi Iwaie6e0ee52013-02-18 17:04:20 +01001593 spec->init_amp = ALC_INIT_NONE;
Takashi Iwai5ebd3bb2013-02-19 18:23:31 +01001594}
1595
1596static void alc260_fixup_fsc_s7020_jwse(struct hda_codec *codec,
1597 const struct hda_fixup *fix, int action)
1598{
1599 struct alc_spec *spec = codec->spec;
1600 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwaif811c3c2013-03-07 18:32:59 +01001601 spec->gen.add_jack_modes = 1;
Takashi Iwai5ebd3bb2013-02-19 18:23:31 +01001602 spec->gen.hp_mic = 1;
Takashi Iwaie6e0ee52013-02-18 17:04:20 +01001603 }
Takashi Iwai39aedee2013-01-10 17:10:40 +01001604}
1605
Takashi Iwai1727a772013-01-10 09:52:52 +01001606static const struct hda_fixup alc260_fixups[] = {
Takashi Iwaica8f0422012-02-16 11:51:19 +01001607 [ALC260_FIXUP_HP_DC5750] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001608 .type = HDA_FIXUP_PINS,
1609 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02001610 { 0x11, 0x90130110 }, /* speaker */
1611 { }
1612 }
1613 },
Takashi Iwaica8f0422012-02-16 11:51:19 +01001614 [ALC260_FIXUP_HP_PIN_0F] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001615 .type = HDA_FIXUP_PINS,
1616 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaica8f0422012-02-16 11:51:19 +01001617 { 0x0f, 0x01214000 }, /* HP */
1618 { }
1619 }
1620 },
1621 [ALC260_FIXUP_COEF] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001622 .type = HDA_FIXUP_VERBS,
Takashi Iwaica8f0422012-02-16 11:51:19 +01001623 .v.verbs = (const struct hda_verb[]) {
Ronan Marquete30cf2d2014-06-01 18:38:53 +02001624 { 0x1a, AC_VERB_SET_COEF_INDEX, 0x07 },
1625 { 0x1a, AC_VERB_SET_PROC_COEF, 0x3040 },
Takashi Iwaica8f0422012-02-16 11:51:19 +01001626 { }
1627 },
Takashi Iwaica8f0422012-02-16 11:51:19 +01001628 },
Takashi Iwai15317ab2012-02-16 12:02:53 +01001629 [ALC260_FIXUP_GPIO1] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001630 .type = HDA_FIXUP_VERBS,
Takashi Iwai15317ab2012-02-16 12:02:53 +01001631 .v.verbs = alc_gpio1_init_verbs,
1632 },
Takashi Iwai20f7d922012-02-16 12:35:16 +01001633 [ALC260_FIXUP_GPIO1_TOGGLE] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001634 .type = HDA_FIXUP_FUNC,
Takashi Iwai20f7d922012-02-16 12:35:16 +01001635 .v.func = alc260_fixup_gpio1_toggle,
1636 .chained = true,
1637 .chain_id = ALC260_FIXUP_HP_PIN_0F,
1638 },
1639 [ALC260_FIXUP_REPLACER] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001640 .type = HDA_FIXUP_VERBS,
Takashi Iwai20f7d922012-02-16 12:35:16 +01001641 .v.verbs = (const struct hda_verb[]) {
Takashi Iwai192a98e2014-06-02 15:16:07 +02001642 { 0x1a, AC_VERB_SET_COEF_INDEX, 0x07 },
1643 { 0x1a, AC_VERB_SET_PROC_COEF, 0x3050 },
Takashi Iwai20f7d922012-02-16 12:35:16 +01001644 { }
1645 },
1646 .chained = true,
1647 .chain_id = ALC260_FIXUP_GPIO1_TOGGLE,
1648 },
Takashi Iwai0a1c4fa2012-02-16 12:42:30 +01001649 [ALC260_FIXUP_HP_B1900] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001650 .type = HDA_FIXUP_FUNC,
Takashi Iwai0a1c4fa2012-02-16 12:42:30 +01001651 .v.func = alc260_fixup_gpio1_toggle,
1652 .chained = true,
1653 .chain_id = ALC260_FIXUP_COEF,
Takashi Iwai118cb4a2012-04-19 07:33:27 +02001654 },
1655 [ALC260_FIXUP_KN1] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001656 .type = HDA_FIXUP_FUNC,
Takashi Iwai118cb4a2012-04-19 07:33:27 +02001657 .v.func = alc260_fixup_kn1,
1658 },
Takashi Iwai39aedee2013-01-10 17:10:40 +01001659 [ALC260_FIXUP_FSC_S7020] = {
1660 .type = HDA_FIXUP_FUNC,
1661 .v.func = alc260_fixup_fsc_s7020,
1662 },
Takashi Iwai5ebd3bb2013-02-19 18:23:31 +01001663 [ALC260_FIXUP_FSC_S7020_JWSE] = {
1664 .type = HDA_FIXUP_FUNC,
1665 .v.func = alc260_fixup_fsc_s7020_jwse,
1666 .chained = true,
1667 .chain_id = ALC260_FIXUP_FSC_S7020,
1668 },
Takashi Iwaid08c5ef2013-11-22 08:06:36 +01001669 [ALC260_FIXUP_VAIO_PINS] = {
1670 .type = HDA_FIXUP_PINS,
1671 .v.pins = (const struct hda_pintbl[]) {
1672 /* Pin configs are missing completely on some VAIOs */
1673 { 0x0f, 0x01211020 },
1674 { 0x10, 0x0001003f },
1675 { 0x11, 0x411111f0 },
1676 { 0x12, 0x01a15930 },
1677 { 0x13, 0x411111f0 },
1678 { 0x14, 0x411111f0 },
1679 { 0x15, 0x411111f0 },
1680 { 0x16, 0x411111f0 },
1681 { 0x17, 0x411111f0 },
1682 { 0x18, 0x411111f0 },
1683 { 0x19, 0x411111f0 },
1684 { }
1685 }
1686 },
Takashi Iwai1d045db2011-07-07 18:23:21 +02001687};
1688
1689static const struct snd_pci_quirk alc260_fixup_tbl[] = {
Takashi Iwai15317ab2012-02-16 12:02:53 +01001690 SND_PCI_QUIRK(0x1025, 0x007b, "Acer C20x", ALC260_FIXUP_GPIO1),
Takashi Iwaica8f0422012-02-16 11:51:19 +01001691 SND_PCI_QUIRK(0x1025, 0x007f, "Acer Aspire 9500", ALC260_FIXUP_COEF),
Takashi Iwai15317ab2012-02-16 12:02:53 +01001692 SND_PCI_QUIRK(0x1025, 0x008f, "Acer", ALC260_FIXUP_GPIO1),
Takashi Iwaica8f0422012-02-16 11:51:19 +01001693 SND_PCI_QUIRK(0x103c, 0x280a, "HP dc5750", ALC260_FIXUP_HP_DC5750),
Takashi Iwai0a1c4fa2012-02-16 12:42:30 +01001694 SND_PCI_QUIRK(0x103c, 0x30ba, "HP Presario B1900", ALC260_FIXUP_HP_B1900),
Takashi Iwaid08c5ef2013-11-22 08:06:36 +01001695 SND_PCI_QUIRK(0x104d, 0x81bb, "Sony VAIO", ALC260_FIXUP_VAIO_PINS),
Takashi Iwai0f5a5b82013-11-21 09:12:52 +01001696 SND_PCI_QUIRK(0x104d, 0x81e2, "Sony VAIO TX", ALC260_FIXUP_HP_PIN_0F),
Takashi Iwai39aedee2013-01-10 17:10:40 +01001697 SND_PCI_QUIRK(0x10cf, 0x1326, "FSC LifeBook S7020", ALC260_FIXUP_FSC_S7020),
Takashi Iwaib1f58082012-02-16 12:45:03 +01001698 SND_PCI_QUIRK(0x1509, 0x4540, "Favorit 100XS", ALC260_FIXUP_GPIO1),
Takashi Iwai118cb4a2012-04-19 07:33:27 +02001699 SND_PCI_QUIRK(0x152d, 0x0729, "Quanta KN1", ALC260_FIXUP_KN1),
Takashi Iwai20f7d922012-02-16 12:35:16 +01001700 SND_PCI_QUIRK(0x161f, 0x2057, "Replacer 672V", ALC260_FIXUP_REPLACER),
Takashi Iwaica8f0422012-02-16 11:51:19 +01001701 SND_PCI_QUIRK(0x1631, 0xc017, "PB V7900", ALC260_FIXUP_COEF),
Takashi Iwai1d045db2011-07-07 18:23:21 +02001702 {}
1703};
1704
Takashi Iwai5ebd3bb2013-02-19 18:23:31 +01001705static const struct hda_model_fixup alc260_fixup_models[] = {
1706 {.id = ALC260_FIXUP_GPIO1, .name = "gpio1"},
1707 {.id = ALC260_FIXUP_COEF, .name = "coef"},
1708 {.id = ALC260_FIXUP_FSC_S7020, .name = "fujitsu"},
1709 {.id = ALC260_FIXUP_FSC_S7020_JWSE, .name = "fujitsu-jwse"},
1710 {}
1711};
1712
Takashi Iwai1d045db2011-07-07 18:23:21 +02001713/*
1714 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02001715static int patch_alc260(struct hda_codec *codec)
1716{
1717 struct alc_spec *spec;
Takashi Iwaic3c2c9e2012-02-16 12:59:55 +01001718 int err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001719
Takashi Iwai3de95172012-05-07 18:03:15 +02001720 err = alc_alloc_spec(codec, 0x07);
1721 if (err < 0)
1722 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001723
Takashi Iwai3de95172012-05-07 18:03:15 +02001724 spec = codec->spec;
Takashi Iwaiea46c3c2013-01-15 18:45:53 +01001725 /* as quite a few machines require HP amp for speaker outputs,
1726 * it's easier to enable it unconditionally; even if it's unneeded,
1727 * it's almost harmless.
1728 */
1729 spec->gen.prefer_hp_amp = 1;
Takashi Iwai7504b6c2013-03-18 11:25:51 +01001730 spec->gen.beep_nid = 0x01;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001731
Takashi Iwai225068a2015-05-29 10:42:14 +02001732 spec->shutup = alc_eapd_shutup;
1733
Takashi Iwai5ebd3bb2013-02-19 18:23:31 +01001734 snd_hda_pick_fixup(codec, alc260_fixup_models, alc260_fixup_tbl,
1735 alc260_fixups);
Takashi Iwai1727a772013-01-10 09:52:52 +01001736 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
Takashi Iwai1d045db2011-07-07 18:23:21 +02001737
Takashi Iwaic3c2c9e2012-02-16 12:59:55 +01001738 /* automatic parse from the BIOS config */
1739 err = alc260_parse_auto_config(codec);
1740 if (err < 0)
1741 goto error;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001742
Takashi Iwai7504b6c2013-03-18 11:25:51 +01001743 if (!spec->gen.no_analog)
Takashi Iwai3e6179b2011-07-08 16:55:13 +02001744 set_beep_amp(spec, 0x07, 0x05, HDA_INPUT);
Takashi Iwai1d045db2011-07-07 18:23:21 +02001745
Takashi Iwai1727a772013-01-10 09:52:52 +01001746 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
Takashi Iwai589876e2012-02-20 15:47:55 +01001747
Takashi Iwai1d045db2011-07-07 18:23:21 +02001748 return 0;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02001749
1750 error:
1751 alc_free(codec);
1752 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001753}
1754
1755
1756/*
1757 * ALC882/883/885/888/889 support
1758 *
1759 * ALC882 is almost identical with ALC880 but has cleaner and more flexible
1760 * configuration. Each pin widget can choose any input DACs and a mixer.
1761 * Each ADC is connected from a mixer of all inputs. This makes possible
1762 * 6-channel independent captures.
1763 *
1764 * In addition, an independent DAC for the multi-playback (not used in this
1765 * driver yet).
1766 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02001767
1768/*
1769 * Pin config fixes
1770 */
1771enum {
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01001772 ALC882_FIXUP_ABIT_AW9D_MAX,
1773 ALC882_FIXUP_LENOVO_Y530,
1774 ALC882_FIXUP_PB_M5210,
1775 ALC882_FIXUP_ACER_ASPIRE_7736,
1776 ALC882_FIXUP_ASUS_W90V,
Marton Balint8f239212012-03-05 21:33:23 +01001777 ALC889_FIXUP_CD,
David Henningssonb2c53e22014-01-01 14:01:34 +01001778 ALC889_FIXUP_FRONT_HP_NO_PRESENCE,
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01001779 ALC889_FIXUP_VAIO_TT,
Takashi Iwai0e7cc2e2011-11-09 12:42:48 +01001780 ALC888_FIXUP_EEE1601,
Takashi Iwai177943a2011-11-09 12:55:18 +01001781 ALC882_FIXUP_EAPD,
Takashi Iwai7a6069b2011-11-09 15:22:01 +01001782 ALC883_FIXUP_EAPD,
Takashi Iwai8812c4f2011-11-09 17:39:15 +01001783 ALC883_FIXUP_ACER_EAPD,
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001784 ALC882_FIXUP_GPIO1,
1785 ALC882_FIXUP_GPIO2,
Takashi Iwaieb844d52011-11-09 18:03:07 +01001786 ALC882_FIXUP_GPIO3,
Takashi Iwai68ef0562011-11-09 18:24:44 +01001787 ALC889_FIXUP_COEF,
1788 ALC882_FIXUP_ASUS_W2JC,
Takashi Iwaic3e837b2011-11-10 16:01:47 +01001789 ALC882_FIXUP_ACER_ASPIRE_4930G,
1790 ALC882_FIXUP_ACER_ASPIRE_8930G,
1791 ALC882_FIXUP_ASPIRE_8930G_VERBS,
Takashi Iwai56710872011-11-14 17:42:11 +01001792 ALC885_FIXUP_MACPRO_GPIO,
Takashi Iwai02a237b2012-02-13 15:25:07 +01001793 ALC889_FIXUP_DAC_ROUTE,
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001794 ALC889_FIXUP_MBP_VREF,
1795 ALC889_FIXUP_IMAC91_VREF,
Adrien Vergée7729a42014-01-24 14:56:14 -05001796 ALC889_FIXUP_MBA11_VREF,
Takashi Iwai0756f092013-12-04 13:59:45 +01001797 ALC889_FIXUP_MBA21_VREF,
Takashi Iwaic20f31e2014-02-03 11:02:10 +01001798 ALC889_FIXUP_MP11_VREF,
Mario Kleiner9f660a12015-12-22 00:45:43 +01001799 ALC889_FIXUP_MP41_VREF,
Takashi Iwai6e72aa52012-06-25 10:52:25 +02001800 ALC882_FIXUP_INV_DMIC,
Takashi Iwaie427c232012-07-29 10:04:08 +02001801 ALC882_FIXUP_NO_PRIMARY_HP,
Takashi Iwai1f0bbf02013-11-28 15:21:21 +01001802 ALC887_FIXUP_ASUS_BASS,
Takashi Iwaieb9ca3a2013-11-28 15:24:34 +01001803 ALC887_FIXUP_BASS_CHMAP,
Takashi Iwai1d045db2011-07-07 18:23:21 +02001804};
1805
Takashi Iwai68ef0562011-11-09 18:24:44 +01001806static void alc889_fixup_coef(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01001807 const struct hda_fixup *fix, int action)
Takashi Iwai68ef0562011-11-09 18:24:44 +01001808{
Takashi Iwai1727a772013-01-10 09:52:52 +01001809 if (action != HDA_FIXUP_ACT_INIT)
Takashi Iwai68ef0562011-11-09 18:24:44 +01001810 return;
Kailang Yang1df88742014-10-29 16:10:13 +08001811 alc_update_coef_idx(codec, 7, 0, 0x2030);
Takashi Iwai68ef0562011-11-09 18:24:44 +01001812}
1813
Takashi Iwai56710872011-11-14 17:42:11 +01001814/* toggle speaker-output according to the hp-jack state */
1815static void alc882_gpio_mute(struct hda_codec *codec, int pin, int muted)
1816{
1817 unsigned int gpiostate, gpiomask, gpiodir;
1818
Takashi Iwai7639a062015-03-03 10:07:24 +01001819 gpiostate = snd_hda_codec_read(codec, codec->core.afg, 0,
Takashi Iwai56710872011-11-14 17:42:11 +01001820 AC_VERB_GET_GPIO_DATA, 0);
1821
1822 if (!muted)
1823 gpiostate |= (1 << pin);
1824 else
1825 gpiostate &= ~(1 << pin);
1826
Takashi Iwai7639a062015-03-03 10:07:24 +01001827 gpiomask = snd_hda_codec_read(codec, codec->core.afg, 0,
Takashi Iwai56710872011-11-14 17:42:11 +01001828 AC_VERB_GET_GPIO_MASK, 0);
1829 gpiomask |= (1 << pin);
1830
Takashi Iwai7639a062015-03-03 10:07:24 +01001831 gpiodir = snd_hda_codec_read(codec, codec->core.afg, 0,
Takashi Iwai56710872011-11-14 17:42:11 +01001832 AC_VERB_GET_GPIO_DIRECTION, 0);
1833 gpiodir |= (1 << pin);
1834
1835
Takashi Iwai7639a062015-03-03 10:07:24 +01001836 snd_hda_codec_write(codec, codec->core.afg, 0,
Takashi Iwai56710872011-11-14 17:42:11 +01001837 AC_VERB_SET_GPIO_MASK, gpiomask);
Takashi Iwai7639a062015-03-03 10:07:24 +01001838 snd_hda_codec_write(codec, codec->core.afg, 0,
Takashi Iwai56710872011-11-14 17:42:11 +01001839 AC_VERB_SET_GPIO_DIRECTION, gpiodir);
1840
1841 msleep(1);
1842
Takashi Iwai7639a062015-03-03 10:07:24 +01001843 snd_hda_codec_write(codec, codec->core.afg, 0,
Takashi Iwai56710872011-11-14 17:42:11 +01001844 AC_VERB_SET_GPIO_DATA, gpiostate);
1845}
1846
1847/* set up GPIO at initialization */
1848static void alc885_fixup_macpro_gpio(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01001849 const struct hda_fixup *fix, int action)
Takashi Iwai56710872011-11-14 17:42:11 +01001850{
Takashi Iwai1727a772013-01-10 09:52:52 +01001851 if (action != HDA_FIXUP_ACT_INIT)
Takashi Iwai56710872011-11-14 17:42:11 +01001852 return;
1853 alc882_gpio_mute(codec, 0, 0);
1854 alc882_gpio_mute(codec, 1, 0);
1855}
1856
Takashi Iwai02a237b2012-02-13 15:25:07 +01001857/* Fix the connection of some pins for ALC889:
1858 * At least, Acer Aspire 5935 shows the connections to DAC3/4 don't
1859 * work correctly (bko#42740)
1860 */
1861static void alc889_fixup_dac_route(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01001862 const struct hda_fixup *fix, int action)
Takashi Iwai02a237b2012-02-13 15:25:07 +01001863{
Takashi Iwai1727a772013-01-10 09:52:52 +01001864 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwaief8d60f2012-02-17 10:12:38 +01001865 /* fake the connections during parsing the tree */
Takashi Iwai02a237b2012-02-13 15:25:07 +01001866 hda_nid_t conn1[2] = { 0x0c, 0x0d };
1867 hda_nid_t conn2[2] = { 0x0e, 0x0f };
1868 snd_hda_override_conn_list(codec, 0x14, 2, conn1);
1869 snd_hda_override_conn_list(codec, 0x15, 2, conn1);
1870 snd_hda_override_conn_list(codec, 0x18, 2, conn2);
1871 snd_hda_override_conn_list(codec, 0x1a, 2, conn2);
Takashi Iwai1727a772013-01-10 09:52:52 +01001872 } else if (action == HDA_FIXUP_ACT_PROBE) {
Takashi Iwaief8d60f2012-02-17 10:12:38 +01001873 /* restore the connections */
1874 hda_nid_t conn[5] = { 0x0c, 0x0d, 0x0e, 0x0f, 0x26 };
1875 snd_hda_override_conn_list(codec, 0x14, 5, conn);
1876 snd_hda_override_conn_list(codec, 0x15, 5, conn);
1877 snd_hda_override_conn_list(codec, 0x18, 5, conn);
1878 snd_hda_override_conn_list(codec, 0x1a, 5, conn);
Takashi Iwai02a237b2012-02-13 15:25:07 +01001879 }
1880}
1881
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001882/* Set VREF on HP pin */
1883static void alc889_fixup_mbp_vref(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01001884 const struct hda_fixup *fix, int action)
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001885{
1886 struct alc_spec *spec = codec->spec;
Mario Kleiner9f660a12015-12-22 00:45:43 +01001887 static hda_nid_t nids[3] = { 0x14, 0x15, 0x19 };
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001888 int i;
1889
Takashi Iwai1727a772013-01-10 09:52:52 +01001890 if (action != HDA_FIXUP_ACT_INIT)
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001891 return;
1892 for (i = 0; i < ARRAY_SIZE(nids); i++) {
1893 unsigned int val = snd_hda_codec_get_pincfg(codec, nids[i]);
1894 if (get_defcfg_device(val) != AC_JACK_HP_OUT)
1895 continue;
Takashi Iwaid3f02d62013-01-10 10:12:22 +01001896 val = snd_hda_codec_get_pin_target(codec, nids[i]);
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001897 val |= AC_PINCTL_VREF_80;
Takashi Iwaicdd03ce2012-04-20 12:34:50 +02001898 snd_hda_set_pin_ctl(codec, nids[i], val);
Takashi Iwai08c189f2012-12-19 15:22:24 +01001899 spec->gen.keep_vref_in_automute = 1;
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001900 break;
1901 }
1902}
1903
Takashi Iwai0756f092013-12-04 13:59:45 +01001904static void alc889_fixup_mac_pins(struct hda_codec *codec,
1905 const hda_nid_t *nids, int num_nids)
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001906{
1907 struct alc_spec *spec = codec->spec;
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001908 int i;
1909
Takashi Iwai0756f092013-12-04 13:59:45 +01001910 for (i = 0; i < num_nids; i++) {
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001911 unsigned int val;
Takashi Iwaid3f02d62013-01-10 10:12:22 +01001912 val = snd_hda_codec_get_pin_target(codec, nids[i]);
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001913 val |= AC_PINCTL_VREF_50;
Takashi Iwaicdd03ce2012-04-20 12:34:50 +02001914 snd_hda_set_pin_ctl(codec, nids[i], val);
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001915 }
Takashi Iwai08c189f2012-12-19 15:22:24 +01001916 spec->gen.keep_vref_in_automute = 1;
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001917}
1918
Takashi Iwai0756f092013-12-04 13:59:45 +01001919/* Set VREF on speaker pins on imac91 */
1920static void alc889_fixup_imac91_vref(struct hda_codec *codec,
1921 const struct hda_fixup *fix, int action)
1922{
1923 static hda_nid_t nids[2] = { 0x18, 0x1a };
1924
1925 if (action == HDA_FIXUP_ACT_INIT)
1926 alc889_fixup_mac_pins(codec, nids, ARRAY_SIZE(nids));
1927}
1928
Adrien Vergée7729a42014-01-24 14:56:14 -05001929/* Set VREF on speaker pins on mba11 */
1930static void alc889_fixup_mba11_vref(struct hda_codec *codec,
1931 const struct hda_fixup *fix, int action)
1932{
1933 static hda_nid_t nids[1] = { 0x18 };
1934
1935 if (action == HDA_FIXUP_ACT_INIT)
1936 alc889_fixup_mac_pins(codec, nids, ARRAY_SIZE(nids));
1937}
1938
Takashi Iwai0756f092013-12-04 13:59:45 +01001939/* Set VREF on speaker pins on mba21 */
1940static void alc889_fixup_mba21_vref(struct hda_codec *codec,
1941 const struct hda_fixup *fix, int action)
1942{
1943 static hda_nid_t nids[2] = { 0x18, 0x19 };
1944
1945 if (action == HDA_FIXUP_ACT_INIT)
1946 alc889_fixup_mac_pins(codec, nids, ARRAY_SIZE(nids));
1947}
1948
Takashi Iwaie427c232012-07-29 10:04:08 +02001949/* Don't take HP output as primary
Fernando Luis Vázquez Caod9111492013-02-12 16:49:46 +09001950 * Strangely, the speaker output doesn't work on Vaio Z and some Vaio
1951 * all-in-one desktop PCs (for example VGC-LN51JGB) through DAC 0x05
Takashi Iwaie427c232012-07-29 10:04:08 +02001952 */
1953static void alc882_fixup_no_primary_hp(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01001954 const struct hda_fixup *fix, int action)
Takashi Iwaie427c232012-07-29 10:04:08 +02001955{
1956 struct alc_spec *spec = codec->spec;
Takashi Iwaida96fb52013-07-29 16:54:36 +02001957 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwai08c189f2012-12-19 15:22:24 +01001958 spec->gen.no_primary_hp = 1;
Takashi Iwaida96fb52013-07-29 16:54:36 +02001959 spec->gen.no_multi_io = 1;
1960 }
Takashi Iwaie427c232012-07-29 10:04:08 +02001961}
1962
Takashi Iwaieb9ca3a2013-11-28 15:24:34 +01001963static void alc_fixup_bass_chmap(struct hda_codec *codec,
1964 const struct hda_fixup *fix, int action);
1965
Takashi Iwai1727a772013-01-10 09:52:52 +01001966static const struct hda_fixup alc882_fixups[] = {
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01001967 [ALC882_FIXUP_ABIT_AW9D_MAX] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001968 .type = HDA_FIXUP_PINS,
1969 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02001970 { 0x15, 0x01080104 }, /* side */
1971 { 0x16, 0x01011012 }, /* rear */
1972 { 0x17, 0x01016011 }, /* clfe */
1973 { }
1974 }
1975 },
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01001976 [ALC882_FIXUP_LENOVO_Y530] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001977 .type = HDA_FIXUP_PINS,
1978 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02001979 { 0x15, 0x99130112 }, /* rear int speakers */
1980 { 0x16, 0x99130111 }, /* subwoofer */
1981 { }
1982 }
1983 },
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01001984 [ALC882_FIXUP_PB_M5210] = {
Takashi Iwaifd108212013-01-10 10:18:14 +01001985 .type = HDA_FIXUP_PINCTLS,
1986 .v.pins = (const struct hda_pintbl[]) {
1987 { 0x19, PIN_VREF50 },
Takashi Iwai1d045db2011-07-07 18:23:21 +02001988 {}
1989 }
1990 },
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01001991 [ALC882_FIXUP_ACER_ASPIRE_7736] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001992 .type = HDA_FIXUP_FUNC,
Takashi Iwai23d30f22012-05-07 17:17:32 +02001993 .v.func = alc_fixup_sku_ignore,
Takashi Iwai1d045db2011-07-07 18:23:21 +02001994 },
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01001995 [ALC882_FIXUP_ASUS_W90V] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001996 .type = HDA_FIXUP_PINS,
1997 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai5cdf7452011-10-26 23:04:08 +02001998 { 0x16, 0x99130110 }, /* fix sequence for CLFE */
1999 { }
2000 }
2001 },
Marton Balint8f239212012-03-05 21:33:23 +01002002 [ALC889_FIXUP_CD] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002003 .type = HDA_FIXUP_PINS,
2004 .v.pins = (const struct hda_pintbl[]) {
Marton Balint8f239212012-03-05 21:33:23 +01002005 { 0x1c, 0x993301f0 }, /* CD */
2006 { }
2007 }
2008 },
David Henningssonb2c53e22014-01-01 14:01:34 +01002009 [ALC889_FIXUP_FRONT_HP_NO_PRESENCE] = {
2010 .type = HDA_FIXUP_PINS,
2011 .v.pins = (const struct hda_pintbl[]) {
2012 { 0x1b, 0x02214120 }, /* Front HP jack is flaky, disable jack detect */
2013 { }
2014 },
2015 .chained = true,
2016 .chain_id = ALC889_FIXUP_CD,
2017 },
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01002018 [ALC889_FIXUP_VAIO_TT] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002019 .type = HDA_FIXUP_PINS,
2020 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01002021 { 0x17, 0x90170111 }, /* hidden surround speaker */
2022 { }
2023 }
2024 },
Takashi Iwai0e7cc2e2011-11-09 12:42:48 +01002025 [ALC888_FIXUP_EEE1601] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002026 .type = HDA_FIXUP_VERBS,
Takashi Iwai0e7cc2e2011-11-09 12:42:48 +01002027 .v.verbs = (const struct hda_verb[]) {
2028 { 0x20, AC_VERB_SET_COEF_INDEX, 0x0b },
2029 { 0x20, AC_VERB_SET_PROC_COEF, 0x0838 },
2030 { }
2031 }
Takashi Iwai177943a2011-11-09 12:55:18 +01002032 },
2033 [ALC882_FIXUP_EAPD] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002034 .type = HDA_FIXUP_VERBS,
Takashi Iwai177943a2011-11-09 12:55:18 +01002035 .v.verbs = (const struct hda_verb[]) {
2036 /* change to EAPD mode */
2037 { 0x20, AC_VERB_SET_COEF_INDEX, 0x07 },
2038 { 0x20, AC_VERB_SET_PROC_COEF, 0x3060 },
2039 { }
2040 }
2041 },
Takashi Iwai7a6069b2011-11-09 15:22:01 +01002042 [ALC883_FIXUP_EAPD] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002043 .type = HDA_FIXUP_VERBS,
Takashi Iwai7a6069b2011-11-09 15:22:01 +01002044 .v.verbs = (const struct hda_verb[]) {
2045 /* change to EAPD mode */
2046 { 0x20, AC_VERB_SET_COEF_INDEX, 0x07 },
2047 { 0x20, AC_VERB_SET_PROC_COEF, 0x3070 },
2048 { }
2049 }
2050 },
Takashi Iwai8812c4f2011-11-09 17:39:15 +01002051 [ALC883_FIXUP_ACER_EAPD] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002052 .type = HDA_FIXUP_VERBS,
Takashi Iwai8812c4f2011-11-09 17:39:15 +01002053 .v.verbs = (const struct hda_verb[]) {
2054 /* eanable EAPD on Acer laptops */
2055 { 0x20, AC_VERB_SET_COEF_INDEX, 0x07 },
2056 { 0x20, AC_VERB_SET_PROC_COEF, 0x3050 },
2057 { }
2058 }
2059 },
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002060 [ALC882_FIXUP_GPIO1] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002061 .type = HDA_FIXUP_VERBS,
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002062 .v.verbs = alc_gpio1_init_verbs,
2063 },
2064 [ALC882_FIXUP_GPIO2] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002065 .type = HDA_FIXUP_VERBS,
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002066 .v.verbs = alc_gpio2_init_verbs,
2067 },
Takashi Iwaieb844d52011-11-09 18:03:07 +01002068 [ALC882_FIXUP_GPIO3] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002069 .type = HDA_FIXUP_VERBS,
Takashi Iwaieb844d52011-11-09 18:03:07 +01002070 .v.verbs = alc_gpio3_init_verbs,
2071 },
Takashi Iwai68ef0562011-11-09 18:24:44 +01002072 [ALC882_FIXUP_ASUS_W2JC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002073 .type = HDA_FIXUP_VERBS,
Takashi Iwai68ef0562011-11-09 18:24:44 +01002074 .v.verbs = alc_gpio1_init_verbs,
2075 .chained = true,
2076 .chain_id = ALC882_FIXUP_EAPD,
2077 },
2078 [ALC889_FIXUP_COEF] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002079 .type = HDA_FIXUP_FUNC,
Takashi Iwai68ef0562011-11-09 18:24:44 +01002080 .v.func = alc889_fixup_coef,
2081 },
Takashi Iwaic3e837b2011-11-10 16:01:47 +01002082 [ALC882_FIXUP_ACER_ASPIRE_4930G] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002083 .type = HDA_FIXUP_PINS,
2084 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaic3e837b2011-11-10 16:01:47 +01002085 { 0x16, 0x99130111 }, /* CLFE speaker */
2086 { 0x17, 0x99130112 }, /* surround speaker */
2087 { }
Takashi Iwai038d4fe2012-04-11 17:18:12 +02002088 },
2089 .chained = true,
2090 .chain_id = ALC882_FIXUP_GPIO1,
Takashi Iwaic3e837b2011-11-10 16:01:47 +01002091 },
2092 [ALC882_FIXUP_ACER_ASPIRE_8930G] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002093 .type = HDA_FIXUP_PINS,
2094 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaic3e837b2011-11-10 16:01:47 +01002095 { 0x16, 0x99130111 }, /* CLFE speaker */
2096 { 0x1b, 0x99130112 }, /* surround speaker */
2097 { }
2098 },
2099 .chained = true,
2100 .chain_id = ALC882_FIXUP_ASPIRE_8930G_VERBS,
2101 },
2102 [ALC882_FIXUP_ASPIRE_8930G_VERBS] = {
2103 /* additional init verbs for Acer Aspire 8930G */
Takashi Iwai1727a772013-01-10 09:52:52 +01002104 .type = HDA_FIXUP_VERBS,
Takashi Iwaic3e837b2011-11-10 16:01:47 +01002105 .v.verbs = (const struct hda_verb[]) {
2106 /* Enable all DACs */
2107 /* DAC DISABLE/MUTE 1? */
2108 /* setting bits 1-5 disables DAC nids 0x02-0x06
2109 * apparently. Init=0x38 */
2110 { 0x20, AC_VERB_SET_COEF_INDEX, 0x03 },
2111 { 0x20, AC_VERB_SET_PROC_COEF, 0x0000 },
2112 /* DAC DISABLE/MUTE 2? */
2113 /* some bit here disables the other DACs.
2114 * Init=0x4900 */
2115 { 0x20, AC_VERB_SET_COEF_INDEX, 0x08 },
2116 { 0x20, AC_VERB_SET_PROC_COEF, 0x0000 },
2117 /* DMIC fix
2118 * This laptop has a stereo digital microphone.
2119 * The mics are only 1cm apart which makes the stereo
2120 * useless. However, either the mic or the ALC889
2121 * makes the signal become a difference/sum signal
2122 * instead of standard stereo, which is annoying.
2123 * So instead we flip this bit which makes the
2124 * codec replicate the sum signal to both channels,
2125 * turning it into a normal mono mic.
2126 */
2127 /* DMIC_CONTROL? Init value = 0x0001 */
2128 { 0x20, AC_VERB_SET_COEF_INDEX, 0x0b },
2129 { 0x20, AC_VERB_SET_PROC_COEF, 0x0003 },
2130 { 0x20, AC_VERB_SET_COEF_INDEX, 0x07 },
2131 { 0x20, AC_VERB_SET_PROC_COEF, 0x3050 },
2132 { }
Takashi Iwai038d4fe2012-04-11 17:18:12 +02002133 },
2134 .chained = true,
2135 .chain_id = ALC882_FIXUP_GPIO1,
Takashi Iwaic3e837b2011-11-10 16:01:47 +01002136 },
Takashi Iwai56710872011-11-14 17:42:11 +01002137 [ALC885_FIXUP_MACPRO_GPIO] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002138 .type = HDA_FIXUP_FUNC,
Takashi Iwai56710872011-11-14 17:42:11 +01002139 .v.func = alc885_fixup_macpro_gpio,
2140 },
Takashi Iwai02a237b2012-02-13 15:25:07 +01002141 [ALC889_FIXUP_DAC_ROUTE] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002142 .type = HDA_FIXUP_FUNC,
Takashi Iwai02a237b2012-02-13 15:25:07 +01002143 .v.func = alc889_fixup_dac_route,
2144 },
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002145 [ALC889_FIXUP_MBP_VREF] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002146 .type = HDA_FIXUP_FUNC,
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002147 .v.func = alc889_fixup_mbp_vref,
2148 .chained = true,
2149 .chain_id = ALC882_FIXUP_GPIO1,
2150 },
2151 [ALC889_FIXUP_IMAC91_VREF] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002152 .type = HDA_FIXUP_FUNC,
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002153 .v.func = alc889_fixup_imac91_vref,
2154 .chained = true,
2155 .chain_id = ALC882_FIXUP_GPIO1,
2156 },
Adrien Vergée7729a42014-01-24 14:56:14 -05002157 [ALC889_FIXUP_MBA11_VREF] = {
2158 .type = HDA_FIXUP_FUNC,
2159 .v.func = alc889_fixup_mba11_vref,
2160 .chained = true,
2161 .chain_id = ALC889_FIXUP_MBP_VREF,
2162 },
Takashi Iwai0756f092013-12-04 13:59:45 +01002163 [ALC889_FIXUP_MBA21_VREF] = {
2164 .type = HDA_FIXUP_FUNC,
2165 .v.func = alc889_fixup_mba21_vref,
2166 .chained = true,
2167 .chain_id = ALC889_FIXUP_MBP_VREF,
2168 },
Takashi Iwaic20f31e2014-02-03 11:02:10 +01002169 [ALC889_FIXUP_MP11_VREF] = {
2170 .type = HDA_FIXUP_FUNC,
2171 .v.func = alc889_fixup_mba11_vref,
2172 .chained = true,
2173 .chain_id = ALC885_FIXUP_MACPRO_GPIO,
2174 },
Mario Kleiner9f660a12015-12-22 00:45:43 +01002175 [ALC889_FIXUP_MP41_VREF] = {
2176 .type = HDA_FIXUP_FUNC,
2177 .v.func = alc889_fixup_mbp_vref,
2178 .chained = true,
2179 .chain_id = ALC885_FIXUP_MACPRO_GPIO,
2180 },
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002181 [ALC882_FIXUP_INV_DMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002182 .type = HDA_FIXUP_FUNC,
David Henningsson9d36a7d2014-10-07 10:18:42 +02002183 .v.func = alc_fixup_inv_dmic,
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002184 },
Takashi Iwaie427c232012-07-29 10:04:08 +02002185 [ALC882_FIXUP_NO_PRIMARY_HP] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002186 .type = HDA_FIXUP_FUNC,
Takashi Iwaie427c232012-07-29 10:04:08 +02002187 .v.func = alc882_fixup_no_primary_hp,
2188 },
Takashi Iwai1f0bbf02013-11-28 15:21:21 +01002189 [ALC887_FIXUP_ASUS_BASS] = {
2190 .type = HDA_FIXUP_PINS,
2191 .v.pins = (const struct hda_pintbl[]) {
2192 {0x16, 0x99130130}, /* bass speaker */
2193 {}
2194 },
Takashi Iwaieb9ca3a2013-11-28 15:24:34 +01002195 .chained = true,
2196 .chain_id = ALC887_FIXUP_BASS_CHMAP,
2197 },
2198 [ALC887_FIXUP_BASS_CHMAP] = {
2199 .type = HDA_FIXUP_FUNC,
2200 .v.func = alc_fixup_bass_chmap,
Takashi Iwai1f0bbf02013-11-28 15:21:21 +01002201 },
Takashi Iwai1d045db2011-07-07 18:23:21 +02002202};
2203
2204static const struct snd_pci_quirk alc882_fixup_tbl[] = {
Takashi Iwai8812c4f2011-11-09 17:39:15 +01002205 SND_PCI_QUIRK(0x1025, 0x006c, "Acer Aspire 9810", ALC883_FIXUP_ACER_EAPD),
2206 SND_PCI_QUIRK(0x1025, 0x0090, "Acer Aspire", ALC883_FIXUP_ACER_EAPD),
Takashi Iwaib5d724b2015-06-02 19:57:08 +02002207 SND_PCI_QUIRK(0x1025, 0x0107, "Acer Aspire", ALC883_FIXUP_ACER_EAPD),
Takashi Iwai8812c4f2011-11-09 17:39:15 +01002208 SND_PCI_QUIRK(0x1025, 0x010a, "Acer Ferrari 5000", ALC883_FIXUP_ACER_EAPD),
2209 SND_PCI_QUIRK(0x1025, 0x0110, "Acer Aspire", ALC883_FIXUP_ACER_EAPD),
2210 SND_PCI_QUIRK(0x1025, 0x0112, "Acer Aspire 9303", ALC883_FIXUP_ACER_EAPD),
2211 SND_PCI_QUIRK(0x1025, 0x0121, "Acer Aspire 5920G", ALC883_FIXUP_ACER_EAPD),
Takashi Iwaic3e837b2011-11-10 16:01:47 +01002212 SND_PCI_QUIRK(0x1025, 0x013e, "Acer Aspire 4930G",
2213 ALC882_FIXUP_ACER_ASPIRE_4930G),
2214 SND_PCI_QUIRK(0x1025, 0x013f, "Acer Aspire 5930G",
2215 ALC882_FIXUP_ACER_ASPIRE_4930G),
2216 SND_PCI_QUIRK(0x1025, 0x0145, "Acer Aspire 8930G",
2217 ALC882_FIXUP_ACER_ASPIRE_8930G),
2218 SND_PCI_QUIRK(0x1025, 0x0146, "Acer Aspire 6935G",
2219 ALC882_FIXUP_ACER_ASPIRE_8930G),
2220 SND_PCI_QUIRK(0x1025, 0x015e, "Acer Aspire 6930G",
2221 ALC882_FIXUP_ACER_ASPIRE_4930G),
2222 SND_PCI_QUIRK(0x1025, 0x0166, "Acer Aspire 6530G",
2223 ALC882_FIXUP_ACER_ASPIRE_4930G),
2224 SND_PCI_QUIRK(0x1025, 0x0142, "Acer Aspire 7730G",
2225 ALC882_FIXUP_ACER_ASPIRE_4930G),
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01002226 SND_PCI_QUIRK(0x1025, 0x0155, "Packard-Bell M5120", ALC882_FIXUP_PB_M5210),
Takashi Iwaif5c53d82012-05-07 10:07:33 +02002227 SND_PCI_QUIRK(0x1025, 0x021e, "Acer Aspire 5739G",
2228 ALC882_FIXUP_ACER_ASPIRE_4930G),
Takashi Iwai02a237b2012-02-13 15:25:07 +01002229 SND_PCI_QUIRK(0x1025, 0x0259, "Acer Aspire 5935", ALC889_FIXUP_DAC_ROUTE),
Takashi Iwaife97da12012-04-12 08:00:19 +02002230 SND_PCI_QUIRK(0x1025, 0x026b, "Acer Aspire 8940G", ALC882_FIXUP_ACER_ASPIRE_8930G),
Takashi Iwaiac9b1cd2011-11-09 17:45:55 +01002231 SND_PCI_QUIRK(0x1025, 0x0296, "Acer Aspire 7736z", ALC882_FIXUP_ACER_ASPIRE_7736),
Takashi Iwai177943a2011-11-09 12:55:18 +01002232 SND_PCI_QUIRK(0x1043, 0x13c2, "Asus A7M", ALC882_FIXUP_EAPD),
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01002233 SND_PCI_QUIRK(0x1043, 0x1873, "ASUS W90V", ALC882_FIXUP_ASUS_W90V),
Takashi Iwai68ef0562011-11-09 18:24:44 +01002234 SND_PCI_QUIRK(0x1043, 0x1971, "Asus W2JC", ALC882_FIXUP_ASUS_W2JC),
Takashi Iwai0e7cc2e2011-11-09 12:42:48 +01002235 SND_PCI_QUIRK(0x1043, 0x835f, "Asus Eee 1601", ALC888_FIXUP_EEE1601),
Takashi Iwai1f0bbf02013-11-28 15:21:21 +01002236 SND_PCI_QUIRK(0x1043, 0x84bc, "ASUS ET2700", ALC887_FIXUP_ASUS_BASS),
Takashi Iwai71c88fc2016-12-06 16:20:36 +01002237 SND_PCI_QUIRK(0x1043, 0x8691, "ASUS ROG Ranger VIII", ALC882_FIXUP_GPIO3),
Takashi Iwaiac9b1cd2011-11-09 17:45:55 +01002238 SND_PCI_QUIRK(0x104d, 0x9047, "Sony Vaio TT", ALC889_FIXUP_VAIO_TT),
Takashi Iwaie427c232012-07-29 10:04:08 +02002239 SND_PCI_QUIRK(0x104d, 0x905a, "Sony Vaio Z", ALC882_FIXUP_NO_PRIMARY_HP),
Sergei A. Trusovc531a242017-08-02 20:23:48 +10002240 SND_PCI_QUIRK(0x104d, 0x9060, "Sony Vaio VPCL14M1R", ALC882_FIXUP_NO_PRIMARY_HP),
Fernando Luis Vázquez Cao12e31a72013-02-12 16:47:44 +09002241 SND_PCI_QUIRK(0x104d, 0x9043, "Sony Vaio VGC-LN51JGB", ALC882_FIXUP_NO_PRIMARY_HP),
Takashi Iwaic44d9b12016-02-07 09:38:26 +01002242 SND_PCI_QUIRK(0x104d, 0x9044, "Sony VAIO AiO", ALC882_FIXUP_NO_PRIMARY_HP),
Takashi Iwai56710872011-11-14 17:42:11 +01002243
2244 /* All Apple entries are in codec SSIDs */
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002245 SND_PCI_QUIRK(0x106b, 0x00a0, "MacBookPro 3,1", ALC889_FIXUP_MBP_VREF),
2246 SND_PCI_QUIRK(0x106b, 0x00a1, "Macbook", ALC889_FIXUP_MBP_VREF),
2247 SND_PCI_QUIRK(0x106b, 0x00a4, "MacbookPro 4,1", ALC889_FIXUP_MBP_VREF),
Takashi Iwaic20f31e2014-02-03 11:02:10 +01002248 SND_PCI_QUIRK(0x106b, 0x0c00, "Mac Pro", ALC889_FIXUP_MP11_VREF),
Takashi Iwai56710872011-11-14 17:42:11 +01002249 SND_PCI_QUIRK(0x106b, 0x1000, "iMac 24", ALC885_FIXUP_MACPRO_GPIO),
2250 SND_PCI_QUIRK(0x106b, 0x2800, "AppleTV", ALC885_FIXUP_MACPRO_GPIO),
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002251 SND_PCI_QUIRK(0x106b, 0x2c00, "MacbookPro rev3", ALC889_FIXUP_MBP_VREF),
2252 SND_PCI_QUIRK(0x106b, 0x3000, "iMac", ALC889_FIXUP_MBP_VREF),
Takashi Iwai56710872011-11-14 17:42:11 +01002253 SND_PCI_QUIRK(0x106b, 0x3200, "iMac 7,1 Aluminum", ALC882_FIXUP_EAPD),
Adrien Vergée7729a42014-01-24 14:56:14 -05002254 SND_PCI_QUIRK(0x106b, 0x3400, "MacBookAir 1,1", ALC889_FIXUP_MBA11_VREF),
Takashi Iwai0756f092013-12-04 13:59:45 +01002255 SND_PCI_QUIRK(0x106b, 0x3500, "MacBookAir 2,1", ALC889_FIXUP_MBA21_VREF),
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002256 SND_PCI_QUIRK(0x106b, 0x3600, "Macbook 3,1", ALC889_FIXUP_MBP_VREF),
2257 SND_PCI_QUIRK(0x106b, 0x3800, "MacbookPro 4,1", ALC889_FIXUP_MBP_VREF),
Takashi Iwai56710872011-11-14 17:42:11 +01002258 SND_PCI_QUIRK(0x106b, 0x3e00, "iMac 24 Aluminum", ALC885_FIXUP_MACPRO_GPIO),
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002259 SND_PCI_QUIRK(0x106b, 0x3f00, "Macbook 5,1", ALC889_FIXUP_IMAC91_VREF),
2260 SND_PCI_QUIRK(0x106b, 0x4000, "MacbookPro 5,1", ALC889_FIXUP_IMAC91_VREF),
2261 SND_PCI_QUIRK(0x106b, 0x4100, "Macmini 3,1", ALC889_FIXUP_IMAC91_VREF),
Mario Kleiner9f660a12015-12-22 00:45:43 +01002262 SND_PCI_QUIRK(0x106b, 0x4200, "Mac Pro 4,1/5,1", ALC889_FIXUP_MP41_VREF),
Takashi Iwai05193632012-11-12 10:07:36 +01002263 SND_PCI_QUIRK(0x106b, 0x4300, "iMac 9,1", ALC889_FIXUP_IMAC91_VREF),
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002264 SND_PCI_QUIRK(0x106b, 0x4600, "MacbookPro 5,2", ALC889_FIXUP_IMAC91_VREF),
2265 SND_PCI_QUIRK(0x106b, 0x4900, "iMac 9,1 Aluminum", ALC889_FIXUP_IMAC91_VREF),
Takashi Iwai649ccd02015-07-30 22:30:29 +02002266 SND_PCI_QUIRK(0x106b, 0x4a00, "Macbook 5,2", ALC889_FIXUP_MBA11_VREF),
Takashi Iwai56710872011-11-14 17:42:11 +01002267
Takashi Iwai7a6069b2011-11-09 15:22:01 +01002268 SND_PCI_QUIRK(0x1071, 0x8258, "Evesham Voyaeger", ALC882_FIXUP_EAPD),
Takashi Iwaibca40132012-05-07 11:13:14 +02002269 SND_PCI_QUIRK(0x1462, 0x7350, "MSI-7350", ALC889_FIXUP_CD),
Takashi Iwaieb844d52011-11-09 18:03:07 +01002270 SND_PCI_QUIRK_VENDOR(0x1462, "MSI", ALC882_FIXUP_GPIO3),
David Henningssonb2c53e22014-01-01 14:01:34 +01002271 SND_PCI_QUIRK(0x1458, 0xa002, "Gigabyte EP45-DS3/Z87X-UD3H", ALC889_FIXUP_FRONT_HP_NO_PRESENCE),
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01002272 SND_PCI_QUIRK(0x147b, 0x107a, "Abit AW9D-MAX", ALC882_FIXUP_ABIT_AW9D_MAX),
Takashi Iwai7a6069b2011-11-09 15:22:01 +01002273 SND_PCI_QUIRK_VENDOR(0x1558, "Clevo laptop", ALC882_FIXUP_EAPD),
2274 SND_PCI_QUIRK(0x161f, 0x2054, "Medion laptop", ALC883_FIXUP_EAPD),
Takashi Iwaiac9b1cd2011-11-09 17:45:55 +01002275 SND_PCI_QUIRK(0x17aa, 0x3a0d, "Lenovo Y530", ALC882_FIXUP_LENOVO_Y530),
Takashi Iwai68ef0562011-11-09 18:24:44 +01002276 SND_PCI_QUIRK(0x8086, 0x0022, "DX58SO", ALC889_FIXUP_COEF),
Takashi Iwai1d045db2011-07-07 18:23:21 +02002277 {}
2278};
2279
Takashi Iwai1727a772013-01-10 09:52:52 +01002280static const struct hda_model_fixup alc882_fixup_models[] = {
Takashi Iwai912093b2012-04-11 14:03:41 +02002281 {.id = ALC882_FIXUP_ACER_ASPIRE_4930G, .name = "acer-aspire-4930g"},
2282 {.id = ALC882_FIXUP_ACER_ASPIRE_8930G, .name = "acer-aspire-8930g"},
2283 {.id = ALC883_FIXUP_ACER_EAPD, .name = "acer-aspire"},
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002284 {.id = ALC882_FIXUP_INV_DMIC, .name = "inv-dmic"},
Takashi Iwaie427c232012-07-29 10:04:08 +02002285 {.id = ALC882_FIXUP_NO_PRIMARY_HP, .name = "no-primary-hp"},
Takashi Iwai912093b2012-04-11 14:03:41 +02002286 {}
2287};
2288
Takashi Iwai1d045db2011-07-07 18:23:21 +02002289/*
2290 * BIOS auto configuration
2291 */
2292/* almost identical with ALC880 parser... */
2293static int alc882_parse_auto_config(struct hda_codec *codec)
2294{
Takashi Iwai1d045db2011-07-07 18:23:21 +02002295 static const hda_nid_t alc882_ignore[] = { 0x1d, 0 };
Takashi Iwai3e6179b2011-07-08 16:55:13 +02002296 static const hda_nid_t alc882_ssids[] = { 0x15, 0x1b, 0x14, 0 };
2297 return alc_parse_auto_config(codec, alc882_ignore, alc882_ssids);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002298}
2299
Takashi Iwai1d045db2011-07-07 18:23:21 +02002300/*
2301 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02002302static int patch_alc882(struct hda_codec *codec)
2303{
2304 struct alc_spec *spec;
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002305 int err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002306
Takashi Iwai3de95172012-05-07 18:03:15 +02002307 err = alc_alloc_spec(codec, 0x0b);
2308 if (err < 0)
2309 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002310
Takashi Iwai3de95172012-05-07 18:03:15 +02002311 spec = codec->spec;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002312
Takashi Iwai7639a062015-03-03 10:07:24 +01002313 switch (codec->core.vendor_id) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02002314 case 0x10ec0882:
2315 case 0x10ec0885:
Takashi Iwaiacf08082014-09-02 07:21:56 +02002316 case 0x10ec0900:
Takashi Iwai1d045db2011-07-07 18:23:21 +02002317 break;
2318 default:
2319 /* ALC883 and variants */
2320 alc_fix_pll_init(codec, 0x20, 0x0a, 10);
2321 break;
2322 }
2323
Takashi Iwai1727a772013-01-10 09:52:52 +01002324 snd_hda_pick_fixup(codec, alc882_fixup_models, alc882_fixup_tbl,
Takashi Iwai912093b2012-04-11 14:03:41 +02002325 alc882_fixups);
Takashi Iwai1727a772013-01-10 09:52:52 +01002326 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002327
2328 alc_auto_parse_customize_define(codec);
2329
Takashi Iwai7504b6c2013-03-18 11:25:51 +01002330 if (has_cdefine_beep(codec))
2331 spec->gen.beep_nid = 0x01;
2332
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002333 /* automatic parse from the BIOS config */
2334 err = alc882_parse_auto_config(codec);
2335 if (err < 0)
2336 goto error;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002337
Takashi Iwai7504b6c2013-03-18 11:25:51 +01002338 if (!spec->gen.no_analog && spec->gen.beep_nid)
Takashi Iwai1d045db2011-07-07 18:23:21 +02002339 set_beep_amp(spec, 0x0b, 0x05, HDA_INPUT);
2340
Takashi Iwai1727a772013-01-10 09:52:52 +01002341 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
Takashi Iwai589876e2012-02-20 15:47:55 +01002342
Takashi Iwai1d045db2011-07-07 18:23:21 +02002343 return 0;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02002344
2345 error:
2346 alc_free(codec);
2347 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002348}
2349
2350
2351/*
2352 * ALC262 support
2353 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02002354static int alc262_parse_auto_config(struct hda_codec *codec)
2355{
Takashi Iwai1d045db2011-07-07 18:23:21 +02002356 static const hda_nid_t alc262_ignore[] = { 0x1d, 0 };
Takashi Iwai3e6179b2011-07-08 16:55:13 +02002357 static const hda_nid_t alc262_ssids[] = { 0x15, 0x1b, 0x14, 0 };
2358 return alc_parse_auto_config(codec, alc262_ignore, alc262_ssids);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002359}
2360
2361/*
2362 * Pin config fixes
2363 */
2364enum {
Takashi Iwaiea4e7af2011-11-07 12:23:55 +01002365 ALC262_FIXUP_FSC_H270,
Takashi Iwai7513e6d2013-01-18 15:41:34 +01002366 ALC262_FIXUP_FSC_S7110,
Takashi Iwaiea4e7af2011-11-07 12:23:55 +01002367 ALC262_FIXUP_HP_Z200,
2368 ALC262_FIXUP_TYAN,
Takashi Iwaic4701502011-11-07 14:20:07 +01002369 ALC262_FIXUP_LENOVO_3000,
Takashi Iwaib42590b2011-11-07 14:41:01 +01002370 ALC262_FIXUP_BENQ,
2371 ALC262_FIXUP_BENQ_T31,
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002372 ALC262_FIXUP_INV_DMIC,
Mengdong Linb5c66112013-11-29 00:35:35 -05002373 ALC262_FIXUP_INTEL_BAYLEYBAY,
Takashi Iwai1d045db2011-07-07 18:23:21 +02002374};
2375
Takashi Iwai1727a772013-01-10 09:52:52 +01002376static const struct hda_fixup alc262_fixups[] = {
Takashi Iwaiea4e7af2011-11-07 12:23:55 +01002377 [ALC262_FIXUP_FSC_H270] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002378 .type = HDA_FIXUP_PINS,
2379 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02002380 { 0x14, 0x99130110 }, /* speaker */
2381 { 0x15, 0x0221142f }, /* front HP */
2382 { 0x1b, 0x0121141f }, /* rear HP */
2383 { }
2384 }
2385 },
Takashi Iwai7513e6d2013-01-18 15:41:34 +01002386 [ALC262_FIXUP_FSC_S7110] = {
2387 .type = HDA_FIXUP_PINS,
2388 .v.pins = (const struct hda_pintbl[]) {
2389 { 0x15, 0x90170110 }, /* speaker */
2390 { }
2391 },
2392 .chained = true,
2393 .chain_id = ALC262_FIXUP_BENQ,
2394 },
Takashi Iwaiea4e7af2011-11-07 12:23:55 +01002395 [ALC262_FIXUP_HP_Z200] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002396 .type = HDA_FIXUP_PINS,
2397 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02002398 { 0x16, 0x99130120 }, /* internal speaker */
2399 { }
2400 }
2401 },
Takashi Iwaiea4e7af2011-11-07 12:23:55 +01002402 [ALC262_FIXUP_TYAN] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002403 .type = HDA_FIXUP_PINS,
2404 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaiea4e7af2011-11-07 12:23:55 +01002405 { 0x14, 0x1993e1f0 }, /* int AUX */
2406 { }
2407 }
2408 },
Takashi Iwaic4701502011-11-07 14:20:07 +01002409 [ALC262_FIXUP_LENOVO_3000] = {
Takashi Iwaifd108212013-01-10 10:18:14 +01002410 .type = HDA_FIXUP_PINCTLS,
2411 .v.pins = (const struct hda_pintbl[]) {
2412 { 0x19, PIN_VREF50 },
Takashi Iwaib42590b2011-11-07 14:41:01 +01002413 {}
2414 },
2415 .chained = true,
2416 .chain_id = ALC262_FIXUP_BENQ,
2417 },
2418 [ALC262_FIXUP_BENQ] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002419 .type = HDA_FIXUP_VERBS,
Takashi Iwaib42590b2011-11-07 14:41:01 +01002420 .v.verbs = (const struct hda_verb[]) {
Takashi Iwaic4701502011-11-07 14:20:07 +01002421 { 0x20, AC_VERB_SET_COEF_INDEX, 0x07 },
2422 { 0x20, AC_VERB_SET_PROC_COEF, 0x3070 },
2423 {}
2424 }
2425 },
Takashi Iwaib42590b2011-11-07 14:41:01 +01002426 [ALC262_FIXUP_BENQ_T31] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002427 .type = HDA_FIXUP_VERBS,
Takashi Iwaib42590b2011-11-07 14:41:01 +01002428 .v.verbs = (const struct hda_verb[]) {
2429 { 0x20, AC_VERB_SET_COEF_INDEX, 0x07 },
2430 { 0x20, AC_VERB_SET_PROC_COEF, 0x3050 },
2431 {}
2432 }
2433 },
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002434 [ALC262_FIXUP_INV_DMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002435 .type = HDA_FIXUP_FUNC,
David Henningsson9d36a7d2014-10-07 10:18:42 +02002436 .v.func = alc_fixup_inv_dmic,
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002437 },
Mengdong Linb5c66112013-11-29 00:35:35 -05002438 [ALC262_FIXUP_INTEL_BAYLEYBAY] = {
2439 .type = HDA_FIXUP_FUNC,
2440 .v.func = alc_fixup_no_depop_delay,
2441 },
Takashi Iwai1d045db2011-07-07 18:23:21 +02002442};
2443
2444static const struct snd_pci_quirk alc262_fixup_tbl[] = {
Takashi Iwaiea4e7af2011-11-07 12:23:55 +01002445 SND_PCI_QUIRK(0x103c, 0x170b, "HP Z200", ALC262_FIXUP_HP_Z200),
Takashi Iwai7513e6d2013-01-18 15:41:34 +01002446 SND_PCI_QUIRK(0x10cf, 0x1397, "Fujitsu Lifebook S7110", ALC262_FIXUP_FSC_S7110),
Takashi Iwai3dcd3be2011-11-07 14:59:40 +01002447 SND_PCI_QUIRK(0x10cf, 0x142d, "Fujitsu Lifebook E8410", ALC262_FIXUP_BENQ),
Takashi Iwaiea4e7af2011-11-07 12:23:55 +01002448 SND_PCI_QUIRK(0x10f1, 0x2915, "Tyan Thunder n6650W", ALC262_FIXUP_TYAN),
2449 SND_PCI_QUIRK(0x1734, 0x1147, "FSC Celsius H270", ALC262_FIXUP_FSC_H270),
Takashi Iwaic4701502011-11-07 14:20:07 +01002450 SND_PCI_QUIRK(0x17aa, 0x384e, "Lenovo 3000", ALC262_FIXUP_LENOVO_3000),
Takashi Iwaib42590b2011-11-07 14:41:01 +01002451 SND_PCI_QUIRK(0x17ff, 0x0560, "Benq ED8", ALC262_FIXUP_BENQ),
2452 SND_PCI_QUIRK(0x17ff, 0x058d, "Benq T31-16", ALC262_FIXUP_BENQ_T31),
Mengdong Linb5c66112013-11-29 00:35:35 -05002453 SND_PCI_QUIRK(0x8086, 0x7270, "BayleyBay", ALC262_FIXUP_INTEL_BAYLEYBAY),
Takashi Iwai1d045db2011-07-07 18:23:21 +02002454 {}
2455};
2456
Takashi Iwai1727a772013-01-10 09:52:52 +01002457static const struct hda_model_fixup alc262_fixup_models[] = {
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002458 {.id = ALC262_FIXUP_INV_DMIC, .name = "inv-dmic"},
2459 {}
2460};
Takashi Iwai1d045db2011-07-07 18:23:21 +02002461
Takashi Iwai1d045db2011-07-07 18:23:21 +02002462/*
2463 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02002464static int patch_alc262(struct hda_codec *codec)
2465{
2466 struct alc_spec *spec;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002467 int err;
2468
Takashi Iwai3de95172012-05-07 18:03:15 +02002469 err = alc_alloc_spec(codec, 0x0b);
2470 if (err < 0)
2471 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002472
Takashi Iwai3de95172012-05-07 18:03:15 +02002473 spec = codec->spec;
Takashi Iwai08c189f2012-12-19 15:22:24 +01002474 spec->gen.shared_mic_vref_pin = 0x18;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002475
Takashi Iwai225068a2015-05-29 10:42:14 +02002476 spec->shutup = alc_eapd_shutup;
2477
Takashi Iwai1d045db2011-07-07 18:23:21 +02002478#if 0
2479 /* pshou 07/11/05 set a zero PCM sample to DAC when FIFO is
2480 * under-run
2481 */
Takashi Iwai98b24882014-08-18 13:47:50 +02002482 alc_update_coefex_idx(codec, 0x1a, 7, 0, 0x80);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002483#endif
Takashi Iwai1d045db2011-07-07 18:23:21 +02002484 alc_fix_pll_init(codec, 0x20, 0x0a, 10);
2485
Takashi Iwai1727a772013-01-10 09:52:52 +01002486 snd_hda_pick_fixup(codec, alc262_fixup_models, alc262_fixup_tbl,
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002487 alc262_fixups);
Takashi Iwai1727a772013-01-10 09:52:52 +01002488 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002489
Takashi Iwaiaf741c12012-05-07 18:09:48 +02002490 alc_auto_parse_customize_define(codec);
2491
Takashi Iwai7504b6c2013-03-18 11:25:51 +01002492 if (has_cdefine_beep(codec))
2493 spec->gen.beep_nid = 0x01;
2494
Takashi Iwai42399f72011-11-07 17:18:44 +01002495 /* automatic parse from the BIOS config */
2496 err = alc262_parse_auto_config(codec);
2497 if (err < 0)
2498 goto error;
Takashi Iwai3e6179b2011-07-08 16:55:13 +02002499
Takashi Iwai7504b6c2013-03-18 11:25:51 +01002500 if (!spec->gen.no_analog && spec->gen.beep_nid)
Takashi Iwai1d045db2011-07-07 18:23:21 +02002501 set_beep_amp(spec, 0x0b, 0x05, HDA_INPUT);
2502
Takashi Iwai1727a772013-01-10 09:52:52 +01002503 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
Takashi Iwai589876e2012-02-20 15:47:55 +01002504
Takashi Iwai1d045db2011-07-07 18:23:21 +02002505 return 0;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02002506
2507 error:
2508 alc_free(codec);
2509 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002510}
2511
2512/*
2513 * ALC268
2514 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02002515/* bind Beep switches of both NID 0x0f and 0x10 */
2516static const struct hda_bind_ctls alc268_bind_beep_sw = {
2517 .ops = &snd_hda_bind_sw,
2518 .values = {
2519 HDA_COMPOSE_AMP_VAL(0x0f, 3, 1, HDA_INPUT),
2520 HDA_COMPOSE_AMP_VAL(0x10, 3, 1, HDA_INPUT),
2521 0
2522 },
2523};
2524
2525static const struct snd_kcontrol_new alc268_beep_mixer[] = {
2526 HDA_CODEC_VOLUME("Beep Playback Volume", 0x1d, 0x0, HDA_INPUT),
2527 HDA_BIND_SW("Beep Playback Switch", &alc268_bind_beep_sw),
2528 { }
2529};
2530
2531/* set PCBEEP vol = 0, mute connections */
2532static const struct hda_verb alc268_beep_init_verbs[] = {
2533 {0x1d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
2534 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
2535 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
2536 { }
2537};
2538
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002539enum {
2540 ALC268_FIXUP_INV_DMIC,
Takashi Iwaicb766402012-10-20 10:55:21 +02002541 ALC268_FIXUP_HP_EAPD,
Takashi Iwai24eff322013-11-04 18:21:08 +01002542 ALC268_FIXUP_SPDIF,
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002543};
2544
Takashi Iwai1727a772013-01-10 09:52:52 +01002545static const struct hda_fixup alc268_fixups[] = {
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002546 [ALC268_FIXUP_INV_DMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002547 .type = HDA_FIXUP_FUNC,
David Henningsson9d36a7d2014-10-07 10:18:42 +02002548 .v.func = alc_fixup_inv_dmic,
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002549 },
Takashi Iwaicb766402012-10-20 10:55:21 +02002550 [ALC268_FIXUP_HP_EAPD] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002551 .type = HDA_FIXUP_VERBS,
Takashi Iwaicb766402012-10-20 10:55:21 +02002552 .v.verbs = (const struct hda_verb[]) {
2553 {0x15, AC_VERB_SET_EAPD_BTLENABLE, 0},
2554 {}
2555 }
2556 },
Takashi Iwai24eff322013-11-04 18:21:08 +01002557 [ALC268_FIXUP_SPDIF] = {
2558 .type = HDA_FIXUP_PINS,
2559 .v.pins = (const struct hda_pintbl[]) {
2560 { 0x1e, 0x014b1180 }, /* enable SPDIF out */
2561 {}
2562 }
2563 },
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002564};
2565
Takashi Iwai1727a772013-01-10 09:52:52 +01002566static const struct hda_model_fixup alc268_fixup_models[] = {
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002567 {.id = ALC268_FIXUP_INV_DMIC, .name = "inv-dmic"},
Takashi Iwaicb766402012-10-20 10:55:21 +02002568 {.id = ALC268_FIXUP_HP_EAPD, .name = "hp-eapd"},
2569 {}
2570};
2571
2572static const struct snd_pci_quirk alc268_fixup_tbl[] = {
Takashi Iwai24eff322013-11-04 18:21:08 +01002573 SND_PCI_QUIRK(0x1025, 0x0139, "Acer TravelMate 6293", ALC268_FIXUP_SPDIF),
David Henningssonfcd8f3b2013-01-28 05:45:47 +01002574 SND_PCI_QUIRK(0x1025, 0x015b, "Acer AOA 150 (ZG5)", ALC268_FIXUP_INV_DMIC),
Takashi Iwaicb766402012-10-20 10:55:21 +02002575 /* below is codec SSID since multiple Toshiba laptops have the
2576 * same PCI SSID 1179:ff00
2577 */
2578 SND_PCI_QUIRK(0x1179, 0xff06, "Toshiba P200", ALC268_FIXUP_HP_EAPD),
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002579 {}
2580};
2581
Takashi Iwai1d045db2011-07-07 18:23:21 +02002582/*
2583 * BIOS auto configuration
2584 */
2585static int alc268_parse_auto_config(struct hda_codec *codec)
2586{
Takashi Iwai3e6179b2011-07-08 16:55:13 +02002587 static const hda_nid_t alc268_ssids[] = { 0x15, 0x1b, 0x14, 0 };
Takashi Iwai7504b6c2013-03-18 11:25:51 +01002588 return alc_parse_auto_config(codec, NULL, alc268_ssids);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002589}
2590
Takashi Iwai1d045db2011-07-07 18:23:21 +02002591/*
2592 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02002593static int patch_alc268(struct hda_codec *codec)
2594{
2595 struct alc_spec *spec;
Takashi Iwai7504b6c2013-03-18 11:25:51 +01002596 int err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002597
Takashi Iwai1d045db2011-07-07 18:23:21 +02002598 /* ALC268 has no aa-loopback mixer */
Takashi Iwai3de95172012-05-07 18:03:15 +02002599 err = alc_alloc_spec(codec, 0);
2600 if (err < 0)
2601 return err;
2602
2603 spec = codec->spec;
Takashi Iwai7504b6c2013-03-18 11:25:51 +01002604 spec->gen.beep_nid = 0x01;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002605
Takashi Iwai225068a2015-05-29 10:42:14 +02002606 spec->shutup = alc_eapd_shutup;
2607
Takashi Iwai1727a772013-01-10 09:52:52 +01002608 snd_hda_pick_fixup(codec, alc268_fixup_models, alc268_fixup_tbl, alc268_fixups);
2609 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002610
Takashi Iwai6ebb8052011-08-16 15:15:40 +02002611 /* automatic parse from the BIOS config */
2612 err = alc268_parse_auto_config(codec);
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02002613 if (err < 0)
2614 goto error;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002615
Takashi Iwai7504b6c2013-03-18 11:25:51 +01002616 if (err > 0 && !spec->gen.no_analog &&
2617 spec->gen.autocfg.speaker_pins[0] != 0x1d) {
2618 add_mixer(spec, alc268_beep_mixer);
2619 snd_hda_add_verbs(codec, alc268_beep_init_verbs);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002620 if (!query_amp_caps(codec, 0x1d, HDA_INPUT))
2621 /* override the amp caps for beep generator */
2622 snd_hda_override_amp_caps(codec, 0x1d, HDA_INPUT,
2623 (0x0c << AC_AMPCAP_OFFSET_SHIFT) |
2624 (0x0c << AC_AMPCAP_NUM_STEPS_SHIFT) |
2625 (0x07 << AC_AMPCAP_STEP_SIZE_SHIFT) |
2626 (0 << AC_AMPCAP_MUTE_SHIFT));
2627 }
2628
Takashi Iwai1727a772013-01-10 09:52:52 +01002629 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002630
Takashi Iwai1d045db2011-07-07 18:23:21 +02002631 return 0;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02002632
2633 error:
2634 alc_free(codec);
2635 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002636}
2637
2638/*
2639 * ALC269
2640 */
Takashi Iwai08c189f2012-12-19 15:22:24 +01002641
Takashi Iwai1d045db2011-07-07 18:23:21 +02002642static const struct hda_pcm_stream alc269_44k_pcm_analog_playback = {
Takashi Iwai1d045db2011-07-07 18:23:21 +02002643 .rates = SNDRV_PCM_RATE_44100, /* fixed rate */
Takashi Iwai1d045db2011-07-07 18:23:21 +02002644};
2645
2646static const struct hda_pcm_stream alc269_44k_pcm_analog_capture = {
Takashi Iwai1d045db2011-07-07 18:23:21 +02002647 .rates = SNDRV_PCM_RATE_44100, /* fixed rate */
Takashi Iwai1d045db2011-07-07 18:23:21 +02002648};
2649
Takashi Iwai1d045db2011-07-07 18:23:21 +02002650/* different alc269-variants */
2651enum {
2652 ALC269_TYPE_ALC269VA,
2653 ALC269_TYPE_ALC269VB,
2654 ALC269_TYPE_ALC269VC,
Kailang Yangadcc70b2012-05-25 08:08:38 +02002655 ALC269_TYPE_ALC269VD,
Kailang Yang065380f2013-01-10 10:25:48 +01002656 ALC269_TYPE_ALC280,
2657 ALC269_TYPE_ALC282,
Kailang Yang2af02be2013-08-22 10:03:50 +02002658 ALC269_TYPE_ALC283,
Kailang Yang065380f2013-01-10 10:25:48 +01002659 ALC269_TYPE_ALC284,
Kailang Yang161ebf22013-10-24 11:36:33 +02002660 ALC269_TYPE_ALC285,
Kailang Yang7fc7d042013-04-25 11:04:43 +02002661 ALC269_TYPE_ALC286,
Kailang Yang506b62c2014-12-18 17:07:44 +08002662 ALC269_TYPE_ALC298,
Kailang Yang1d04c9d2013-10-24 11:35:18 +02002663 ALC269_TYPE_ALC255,
Kailang Yang4344aec2014-12-17 17:39:05 +08002664 ALC269_TYPE_ALC256,
Kailang Yang42314302016-02-03 15:03:50 +08002665 ALC269_TYPE_ALC225,
Kailang Yangdcd4f0d2016-05-04 15:50:18 +08002666 ALC269_TYPE_ALC294,
Kailang Yang6fbae352016-05-30 16:44:20 +08002667 ALC269_TYPE_ALC700,
Takashi Iwai1d045db2011-07-07 18:23:21 +02002668};
2669
2670/*
2671 * BIOS auto configuration
2672 */
2673static int alc269_parse_auto_config(struct hda_codec *codec)
2674{
Takashi Iwai1d045db2011-07-07 18:23:21 +02002675 static const hda_nid_t alc269_ignore[] = { 0x1d, 0 };
Takashi Iwai3e6179b2011-07-08 16:55:13 +02002676 static const hda_nid_t alc269_ssids[] = { 0, 0x1b, 0x14, 0x21 };
2677 static const hda_nid_t alc269va_ssids[] = { 0x15, 0x1b, 0x14, 0 };
2678 struct alc_spec *spec = codec->spec;
Kailang Yangadcc70b2012-05-25 08:08:38 +02002679 const hda_nid_t *ssids;
2680
2681 switch (spec->codec_variant) {
2682 case ALC269_TYPE_ALC269VA:
2683 case ALC269_TYPE_ALC269VC:
Kailang Yang065380f2013-01-10 10:25:48 +01002684 case ALC269_TYPE_ALC280:
2685 case ALC269_TYPE_ALC284:
Kailang Yang161ebf22013-10-24 11:36:33 +02002686 case ALC269_TYPE_ALC285:
Kailang Yangadcc70b2012-05-25 08:08:38 +02002687 ssids = alc269va_ssids;
2688 break;
2689 case ALC269_TYPE_ALC269VB:
2690 case ALC269_TYPE_ALC269VD:
Kailang Yang065380f2013-01-10 10:25:48 +01002691 case ALC269_TYPE_ALC282:
Kailang Yang2af02be2013-08-22 10:03:50 +02002692 case ALC269_TYPE_ALC283:
Kailang Yang7fc7d042013-04-25 11:04:43 +02002693 case ALC269_TYPE_ALC286:
Kailang Yang506b62c2014-12-18 17:07:44 +08002694 case ALC269_TYPE_ALC298:
Kailang Yang1d04c9d2013-10-24 11:35:18 +02002695 case ALC269_TYPE_ALC255:
Kailang Yang4344aec2014-12-17 17:39:05 +08002696 case ALC269_TYPE_ALC256:
Kailang Yang42314302016-02-03 15:03:50 +08002697 case ALC269_TYPE_ALC225:
Kailang Yangdcd4f0d2016-05-04 15:50:18 +08002698 case ALC269_TYPE_ALC294:
Kailang Yang6fbae352016-05-30 16:44:20 +08002699 case ALC269_TYPE_ALC700:
Kailang Yangadcc70b2012-05-25 08:08:38 +02002700 ssids = alc269_ssids;
2701 break;
2702 default:
2703 ssids = alc269_ssids;
2704 break;
2705 }
Takashi Iwai1d045db2011-07-07 18:23:21 +02002706
Takashi Iwai3e6179b2011-07-08 16:55:13 +02002707 return alc_parse_auto_config(codec, alc269_ignore, ssids);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002708}
2709
Kailang Yangf7ae9ba2014-06-30 16:10:37 +08002710static int find_ext_mic_pin(struct hda_codec *codec);
2711
2712static void alc286_shutup(struct hda_codec *codec)
2713{
2714 int i;
2715 int mic_pin = find_ext_mic_pin(codec);
2716 /* don't shut up pins when unloading the driver; otherwise it breaks
2717 * the default pin setup at the next load of the driver
2718 */
2719 if (codec->bus->shutdown)
2720 return;
2721 for (i = 0; i < codec->init_pins.used; i++) {
2722 struct hda_pincfg *pin = snd_array_elem(&codec->init_pins, i);
2723 /* use read here for syncing after issuing each verb */
2724 if (pin->nid != mic_pin)
2725 snd_hda_codec_read(codec, pin->nid, 0,
2726 AC_VERB_SET_PIN_WIDGET_CONTROL, 0);
2727 }
2728 codec->pins_shutup = 1;
2729}
2730
Kailang Yang1387e2d2012-11-08 10:23:18 +01002731static void alc269vb_toggle_power_output(struct hda_codec *codec, int power_up)
Takashi Iwai1d045db2011-07-07 18:23:21 +02002732{
Takashi Iwai98b24882014-08-18 13:47:50 +02002733 alc_update_coef_idx(codec, 0x04, 1 << 11, power_up ? (1 << 11) : 0);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002734}
2735
2736static void alc269_shutup(struct hda_codec *codec)
2737{
Kailang Yangadcc70b2012-05-25 08:08:38 +02002738 struct alc_spec *spec = codec->spec;
2739
Kailang Yang1387e2d2012-11-08 10:23:18 +01002740 if (spec->codec_variant == ALC269_TYPE_ALC269VB)
2741 alc269vb_toggle_power_output(codec, 0);
2742 if (spec->codec_variant == ALC269_TYPE_ALC269VB &&
2743 (alc_get_coef0(codec) & 0x00ff) == 0x018) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02002744 msleep(150);
2745 }
Takashi Iwai9bfb2842013-07-24 14:31:50 +02002746 snd_hda_shutup_pins(codec);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002747}
2748
Takashi Iwai54db6c32014-08-18 15:11:19 +02002749static struct coef_fw alc282_coefs[] = {
2750 WRITE_COEF(0x03, 0x0002), /* Power Down Control */
Kailang Yang32fa7e42014-10-24 16:26:01 +08002751 UPDATE_COEF(0x05, 0xff3f, 0x0700), /* FIFO and filter clock */
Takashi Iwai54db6c32014-08-18 15:11:19 +02002752 WRITE_COEF(0x07, 0x0200), /* DMIC control */
2753 UPDATE_COEF(0x06, 0x00f0, 0), /* Analog clock */
2754 UPDATE_COEF(0x08, 0xfffc, 0x0c2c), /* JD */
2755 WRITE_COEF(0x0a, 0xcccc), /* JD offset1 */
2756 WRITE_COEF(0x0b, 0xcccc), /* JD offset2 */
2757 WRITE_COEF(0x0e, 0x6e00), /* LDO1/2/3, DAC/ADC */
2758 UPDATE_COEF(0x0f, 0xf800, 0x1000), /* JD */
2759 UPDATE_COEF(0x10, 0xfc00, 0x0c00), /* Capless */
2760 WRITE_COEF(0x6f, 0x0), /* Class D test 4 */
2761 UPDATE_COEF(0x0c, 0xfe00, 0), /* IO power down directly */
2762 WRITE_COEF(0x34, 0xa0c0), /* ANC */
2763 UPDATE_COEF(0x16, 0x0008, 0), /* AGC MUX */
2764 UPDATE_COEF(0x1d, 0x00e0, 0), /* DAC simple content protection */
2765 UPDATE_COEF(0x1f, 0x00e0, 0), /* ADC simple content protection */
2766 WRITE_COEF(0x21, 0x8804), /* DAC ADC Zero Detection */
2767 WRITE_COEF(0x63, 0x2902), /* PLL */
2768 WRITE_COEF(0x68, 0xa080), /* capless control 2 */
2769 WRITE_COEF(0x69, 0x3400), /* capless control 3 */
2770 WRITE_COEF(0x6a, 0x2f3e), /* capless control 4 */
2771 WRITE_COEF(0x6b, 0x0), /* capless control 5 */
2772 UPDATE_COEF(0x6d, 0x0fff, 0x0900), /* class D test 2 */
2773 WRITE_COEF(0x6e, 0x110a), /* class D test 3 */
2774 UPDATE_COEF(0x70, 0x00f8, 0x00d8), /* class D test 5 */
2775 WRITE_COEF(0x71, 0x0014), /* class D test 6 */
2776 WRITE_COEF(0x72, 0xc2ba), /* classD OCP */
2777 UPDATE_COEF(0x77, 0x0f80, 0), /* classD pure DC test */
2778 WRITE_COEF(0x6c, 0xfc06), /* Class D amp control */
2779 {}
2780};
2781
Kailang Yangcb149cb2014-03-18 16:45:32 +08002782static void alc282_restore_default_value(struct hda_codec *codec)
2783{
Takashi Iwai54db6c32014-08-18 15:11:19 +02002784 alc_process_coef_fw(codec, alc282_coefs);
Kailang Yangcb149cb2014-03-18 16:45:32 +08002785}
2786
Kailang Yang7b5c7a02014-03-18 16:15:54 +08002787static void alc282_init(struct hda_codec *codec)
2788{
2789 struct alc_spec *spec = codec->spec;
2790 hda_nid_t hp_pin = spec->gen.autocfg.hp_pins[0];
2791 bool hp_pin_sense;
2792 int coef78;
2793
Kailang Yangcb149cb2014-03-18 16:45:32 +08002794 alc282_restore_default_value(codec);
2795
Kailang Yang7b5c7a02014-03-18 16:15:54 +08002796 if (!hp_pin)
2797 return;
2798 hp_pin_sense = snd_hda_jack_detect(codec, hp_pin);
2799 coef78 = alc_read_coef_idx(codec, 0x78);
2800
2801 /* Index 0x78 Direct Drive HP AMP LPM Control 1 */
2802 /* Headphone capless set to high power mode */
2803 alc_write_coef_idx(codec, 0x78, 0x9004);
2804
2805 if (hp_pin_sense)
2806 msleep(2);
2807
2808 snd_hda_codec_write(codec, hp_pin, 0,
2809 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
2810
2811 if (hp_pin_sense)
2812 msleep(85);
2813
2814 snd_hda_codec_write(codec, hp_pin, 0,
2815 AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
2816
2817 if (hp_pin_sense)
2818 msleep(100);
2819
2820 /* Headphone capless set to normal mode */
2821 alc_write_coef_idx(codec, 0x78, coef78);
2822}
2823
2824static void alc282_shutup(struct hda_codec *codec)
2825{
2826 struct alc_spec *spec = codec->spec;
2827 hda_nid_t hp_pin = spec->gen.autocfg.hp_pins[0];
2828 bool hp_pin_sense;
2829 int coef78;
2830
2831 if (!hp_pin) {
2832 alc269_shutup(codec);
2833 return;
2834 }
2835
2836 hp_pin_sense = snd_hda_jack_detect(codec, hp_pin);
2837 coef78 = alc_read_coef_idx(codec, 0x78);
2838 alc_write_coef_idx(codec, 0x78, 0x9004);
2839
2840 if (hp_pin_sense)
2841 msleep(2);
2842
2843 snd_hda_codec_write(codec, hp_pin, 0,
2844 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
2845
2846 if (hp_pin_sense)
2847 msleep(85);
2848
2849 snd_hda_codec_write(codec, hp_pin, 0,
2850 AC_VERB_SET_PIN_WIDGET_CONTROL, 0x0);
2851
2852 if (hp_pin_sense)
2853 msleep(100);
2854
2855 alc_auto_setup_eapd(codec, false);
2856 snd_hda_shutup_pins(codec);
2857 alc_write_coef_idx(codec, 0x78, coef78);
2858}
2859
Takashi Iwai54db6c32014-08-18 15:11:19 +02002860static struct coef_fw alc283_coefs[] = {
2861 WRITE_COEF(0x03, 0x0002), /* Power Down Control */
Kailang Yang56779862014-10-24 16:17:51 +08002862 UPDATE_COEF(0x05, 0xff3f, 0x0700), /* FIFO and filter clock */
Takashi Iwai54db6c32014-08-18 15:11:19 +02002863 WRITE_COEF(0x07, 0x0200), /* DMIC control */
2864 UPDATE_COEF(0x06, 0x00f0, 0), /* Analog clock */
2865 UPDATE_COEF(0x08, 0xfffc, 0x0c2c), /* JD */
2866 WRITE_COEF(0x0a, 0xcccc), /* JD offset1 */
2867 WRITE_COEF(0x0b, 0xcccc), /* JD offset2 */
2868 WRITE_COEF(0x0e, 0x6fc0), /* LDO1/2/3, DAC/ADC */
2869 UPDATE_COEF(0x0f, 0xf800, 0x1000), /* JD */
2870 UPDATE_COEF(0x10, 0xfc00, 0x0c00), /* Capless */
2871 WRITE_COEF(0x3a, 0x0), /* Class D test 4 */
2872 UPDATE_COEF(0x0c, 0xfe00, 0x0), /* IO power down directly */
2873 WRITE_COEF(0x22, 0xa0c0), /* ANC */
2874 UPDATE_COEFEX(0x53, 0x01, 0x000f, 0x0008), /* AGC MUX */
2875 UPDATE_COEF(0x1d, 0x00e0, 0), /* DAC simple content protection */
2876 UPDATE_COEF(0x1f, 0x00e0, 0), /* ADC simple content protection */
2877 WRITE_COEF(0x21, 0x8804), /* DAC ADC Zero Detection */
2878 WRITE_COEF(0x2e, 0x2902), /* PLL */
2879 WRITE_COEF(0x33, 0xa080), /* capless control 2 */
2880 WRITE_COEF(0x34, 0x3400), /* capless control 3 */
2881 WRITE_COEF(0x35, 0x2f3e), /* capless control 4 */
2882 WRITE_COEF(0x36, 0x0), /* capless control 5 */
2883 UPDATE_COEF(0x38, 0x0fff, 0x0900), /* class D test 2 */
2884 WRITE_COEF(0x39, 0x110a), /* class D test 3 */
2885 UPDATE_COEF(0x3b, 0x00f8, 0x00d8), /* class D test 5 */
2886 WRITE_COEF(0x3c, 0x0014), /* class D test 6 */
2887 WRITE_COEF(0x3d, 0xc2ba), /* classD OCP */
2888 UPDATE_COEF(0x42, 0x0f80, 0x0), /* classD pure DC test */
2889 WRITE_COEF(0x49, 0x0), /* test mode */
2890 UPDATE_COEF(0x40, 0xf800, 0x9800), /* Class D DC enable */
2891 UPDATE_COEF(0x42, 0xf000, 0x2000), /* DC offset */
2892 WRITE_COEF(0x37, 0xfc06), /* Class D amp control */
Kailang Yang56779862014-10-24 16:17:51 +08002893 UPDATE_COEF(0x1b, 0x8000, 0), /* HP JD control */
Takashi Iwai54db6c32014-08-18 15:11:19 +02002894 {}
2895};
2896
Kailang Yang6bd55b02014-03-17 13:51:27 +08002897static void alc283_restore_default_value(struct hda_codec *codec)
2898{
Takashi Iwai54db6c32014-08-18 15:11:19 +02002899 alc_process_coef_fw(codec, alc283_coefs);
Kailang Yang6bd55b02014-03-17 13:51:27 +08002900}
2901
Kailang Yang2af02be2013-08-22 10:03:50 +02002902static void alc283_init(struct hda_codec *codec)
2903{
2904 struct alc_spec *spec = codec->spec;
2905 hda_nid_t hp_pin = spec->gen.autocfg.hp_pins[0];
2906 bool hp_pin_sense;
Kailang Yang2af02be2013-08-22 10:03:50 +02002907
Kailang Yang8314f222014-04-03 17:28:39 +08002908 if (!spec->gen.autocfg.hp_outs) {
2909 if (spec->gen.autocfg.line_out_type == AC_JACK_HP_OUT)
2910 hp_pin = spec->gen.autocfg.line_out_pins[0];
2911 }
2912
Kailang Yang6bd55b02014-03-17 13:51:27 +08002913 alc283_restore_default_value(codec);
2914
Kailang Yang2af02be2013-08-22 10:03:50 +02002915 if (!hp_pin)
2916 return;
Kailang Yanga59d7192015-04-08 16:34:00 +08002917
2918 msleep(30);
Kailang Yang2af02be2013-08-22 10:03:50 +02002919 hp_pin_sense = snd_hda_jack_detect(codec, hp_pin);
2920
2921 /* Index 0x43 Direct Drive HP AMP LPM Control 1 */
2922 /* Headphone capless set to high power mode */
2923 alc_write_coef_idx(codec, 0x43, 0x9004);
2924
2925 snd_hda_codec_write(codec, hp_pin, 0,
2926 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
2927
2928 if (hp_pin_sense)
2929 msleep(85);
2930
2931 snd_hda_codec_write(codec, hp_pin, 0,
2932 AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
2933
2934 if (hp_pin_sense)
2935 msleep(85);
2936 /* Index 0x46 Combo jack auto switch control 2 */
2937 /* 3k pull low control for Headset jack. */
Takashi Iwai98b24882014-08-18 13:47:50 +02002938 alc_update_coef_idx(codec, 0x46, 3 << 12, 0);
Kailang Yang2af02be2013-08-22 10:03:50 +02002939 /* Headphone capless set to normal mode */
2940 alc_write_coef_idx(codec, 0x43, 0x9614);
2941}
2942
2943static void alc283_shutup(struct hda_codec *codec)
2944{
2945 struct alc_spec *spec = codec->spec;
2946 hda_nid_t hp_pin = spec->gen.autocfg.hp_pins[0];
2947 bool hp_pin_sense;
Kailang Yang2af02be2013-08-22 10:03:50 +02002948
Kailang Yang8314f222014-04-03 17:28:39 +08002949 if (!spec->gen.autocfg.hp_outs) {
2950 if (spec->gen.autocfg.line_out_type == AC_JACK_HP_OUT)
2951 hp_pin = spec->gen.autocfg.line_out_pins[0];
2952 }
2953
Kailang Yang2af02be2013-08-22 10:03:50 +02002954 if (!hp_pin) {
2955 alc269_shutup(codec);
2956 return;
2957 }
2958
2959 hp_pin_sense = snd_hda_jack_detect(codec, hp_pin);
2960
2961 alc_write_coef_idx(codec, 0x43, 0x9004);
2962
Harsha Priyab450b172014-10-09 11:04:56 +00002963 /*depop hp during suspend*/
2964 alc_write_coef_idx(codec, 0x06, 0x2100);
2965
Kailang Yang2af02be2013-08-22 10:03:50 +02002966 snd_hda_codec_write(codec, hp_pin, 0,
2967 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
2968
2969 if (hp_pin_sense)
Kailang Yang88011c092013-10-24 15:45:24 +02002970 msleep(100);
Kailang Yang2af02be2013-08-22 10:03:50 +02002971
2972 snd_hda_codec_write(codec, hp_pin, 0,
2973 AC_VERB_SET_PIN_WIDGET_CONTROL, 0x0);
2974
Takashi Iwai98b24882014-08-18 13:47:50 +02002975 alc_update_coef_idx(codec, 0x46, 0, 3 << 12);
Kailang Yang2af02be2013-08-22 10:03:50 +02002976
2977 if (hp_pin_sense)
Kailang Yang88011c092013-10-24 15:45:24 +02002978 msleep(100);
Kailang Yang0435b3f2014-04-08 17:14:14 +08002979 alc_auto_setup_eapd(codec, false);
Kailang Yang2af02be2013-08-22 10:03:50 +02002980 snd_hda_shutup_pins(codec);
2981 alc_write_coef_idx(codec, 0x43, 0x9614);
2982}
2983
Kailang Yangad60d502013-06-28 12:03:01 +02002984static void alc5505_coef_set(struct hda_codec *codec, unsigned int index_reg,
2985 unsigned int val)
2986{
2987 snd_hda_codec_write(codec, 0x51, 0, AC_VERB_SET_COEF_INDEX, index_reg >> 1);
2988 snd_hda_codec_write(codec, 0x51, 0, AC_VERB_SET_PROC_COEF, val & 0xffff); /* LSB */
2989 snd_hda_codec_write(codec, 0x51, 0, AC_VERB_SET_PROC_COEF, val >> 16); /* MSB */
2990}
2991
2992static int alc5505_coef_get(struct hda_codec *codec, unsigned int index_reg)
2993{
2994 unsigned int val;
2995
2996 snd_hda_codec_write(codec, 0x51, 0, AC_VERB_SET_COEF_INDEX, index_reg >> 1);
2997 val = snd_hda_codec_read(codec, 0x51, 0, AC_VERB_GET_PROC_COEF, 0)
2998 & 0xffff;
2999 val |= snd_hda_codec_read(codec, 0x51, 0, AC_VERB_GET_PROC_COEF, 0)
3000 << 16;
3001 return val;
3002}
3003
3004static void alc5505_dsp_halt(struct hda_codec *codec)
3005{
3006 unsigned int val;
3007
3008 alc5505_coef_set(codec, 0x3000, 0x000c); /* DSP CPU stop */
3009 alc5505_coef_set(codec, 0x880c, 0x0008); /* DDR enter self refresh */
3010 alc5505_coef_set(codec, 0x61c0, 0x11110080); /* Clock control for PLL and CPU */
3011 alc5505_coef_set(codec, 0x6230, 0xfc0d4011); /* Disable Input OP */
3012 alc5505_coef_set(codec, 0x61b4, 0x040a2b03); /* Stop PLL2 */
3013 alc5505_coef_set(codec, 0x61b0, 0x00005b17); /* Stop PLL1 */
3014 alc5505_coef_set(codec, 0x61b8, 0x04133303); /* Stop PLL3 */
3015 val = alc5505_coef_get(codec, 0x6220);
3016 alc5505_coef_set(codec, 0x6220, (val | 0x3000)); /* switch Ringbuffer clock to DBUS clock */
3017}
3018
3019static void alc5505_dsp_back_from_halt(struct hda_codec *codec)
3020{
3021 alc5505_coef_set(codec, 0x61b8, 0x04133302);
3022 alc5505_coef_set(codec, 0x61b0, 0x00005b16);
3023 alc5505_coef_set(codec, 0x61b4, 0x040a2b02);
3024 alc5505_coef_set(codec, 0x6230, 0xf80d4011);
3025 alc5505_coef_set(codec, 0x6220, 0x2002010f);
3026 alc5505_coef_set(codec, 0x880c, 0x00000004);
3027}
3028
3029static void alc5505_dsp_init(struct hda_codec *codec)
3030{
3031 unsigned int val;
3032
3033 alc5505_dsp_halt(codec);
3034 alc5505_dsp_back_from_halt(codec);
3035 alc5505_coef_set(codec, 0x61b0, 0x5b14); /* PLL1 control */
3036 alc5505_coef_set(codec, 0x61b0, 0x5b16);
3037 alc5505_coef_set(codec, 0x61b4, 0x04132b00); /* PLL2 control */
3038 alc5505_coef_set(codec, 0x61b4, 0x04132b02);
3039 alc5505_coef_set(codec, 0x61b8, 0x041f3300); /* PLL3 control*/
3040 alc5505_coef_set(codec, 0x61b8, 0x041f3302);
3041 snd_hda_codec_write(codec, 0x51, 0, AC_VERB_SET_CODEC_RESET, 0); /* Function reset */
3042 alc5505_coef_set(codec, 0x61b8, 0x041b3302);
3043 alc5505_coef_set(codec, 0x61b8, 0x04173302);
3044 alc5505_coef_set(codec, 0x61b8, 0x04163302);
3045 alc5505_coef_set(codec, 0x8800, 0x348b328b); /* DRAM control */
3046 alc5505_coef_set(codec, 0x8808, 0x00020022); /* DRAM control */
3047 alc5505_coef_set(codec, 0x8818, 0x00000400); /* DRAM control */
3048
3049 val = alc5505_coef_get(codec, 0x6200) >> 16; /* Read revision ID */
3050 if (val <= 3)
3051 alc5505_coef_set(codec, 0x6220, 0x2002010f); /* I/O PAD Configuration */
3052 else
3053 alc5505_coef_set(codec, 0x6220, 0x6002018f);
3054
3055 alc5505_coef_set(codec, 0x61ac, 0x055525f0); /**/
3056 alc5505_coef_set(codec, 0x61c0, 0x12230080); /* Clock control */
3057 alc5505_coef_set(codec, 0x61b4, 0x040e2b02); /* PLL2 control */
3058 alc5505_coef_set(codec, 0x61bc, 0x010234f8); /* OSC Control */
3059 alc5505_coef_set(codec, 0x880c, 0x00000004); /* DRAM Function control */
3060 alc5505_coef_set(codec, 0x880c, 0x00000003);
3061 alc5505_coef_set(codec, 0x880c, 0x00000010);
Takashi Iwaicd63a5f2013-07-05 12:13:59 +02003062
3063#ifdef HALT_REALTEK_ALC5505
3064 alc5505_dsp_halt(codec);
3065#endif
Kailang Yangad60d502013-06-28 12:03:01 +02003066}
3067
Takashi Iwaicd63a5f2013-07-05 12:13:59 +02003068#ifdef HALT_REALTEK_ALC5505
3069#define alc5505_dsp_suspend(codec) /* NOP */
3070#define alc5505_dsp_resume(codec) /* NOP */
3071#else
3072#define alc5505_dsp_suspend(codec) alc5505_dsp_halt(codec)
3073#define alc5505_dsp_resume(codec) alc5505_dsp_back_from_halt(codec)
3074#endif
3075
Takashi Iwai2a439522011-07-26 09:52:50 +02003076#ifdef CONFIG_PM
Kailang Yangad60d502013-06-28 12:03:01 +02003077static int alc269_suspend(struct hda_codec *codec)
3078{
3079 struct alc_spec *spec = codec->spec;
3080
3081 if (spec->has_alc5505_dsp)
Takashi Iwaicd63a5f2013-07-05 12:13:59 +02003082 alc5505_dsp_suspend(codec);
Kailang Yangad60d502013-06-28 12:03:01 +02003083 return alc_suspend(codec);
3084}
3085
Takashi Iwai1d045db2011-07-07 18:23:21 +02003086static int alc269_resume(struct hda_codec *codec)
3087{
Kailang Yangadcc70b2012-05-25 08:08:38 +02003088 struct alc_spec *spec = codec->spec;
3089
Kailang Yang1387e2d2012-11-08 10:23:18 +01003090 if (spec->codec_variant == ALC269_TYPE_ALC269VB)
3091 alc269vb_toggle_power_output(codec, 0);
3092 if (spec->codec_variant == ALC269_TYPE_ALC269VB &&
Kailang Yangadcc70b2012-05-25 08:08:38 +02003093 (alc_get_coef0(codec) & 0x00ff) == 0x018) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02003094 msleep(150);
3095 }
3096
3097 codec->patch_ops.init(codec);
3098
Kailang Yang1387e2d2012-11-08 10:23:18 +01003099 if (spec->codec_variant == ALC269_TYPE_ALC269VB)
3100 alc269vb_toggle_power_output(codec, 1);
3101 if (spec->codec_variant == ALC269_TYPE_ALC269VB &&
Kailang Yangadcc70b2012-05-25 08:08:38 +02003102 (alc_get_coef0(codec) & 0x00ff) == 0x017) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02003103 msleep(200);
3104 }
3105
Takashi Iwaieeecd9d2015-02-25 15:18:50 +01003106 regcache_sync(codec->core.regmap);
Takashi Iwai1d045db2011-07-07 18:23:21 +02003107 hda_call_check_power_status(codec, 0x01);
Hui Wangf4753712014-08-19 12:07:03 +08003108
3109 /* on some machine, the BIOS will clear the codec gpio data when enter
3110 * suspend, and won't restore the data after resume, so we restore it
3111 * in the driver.
3112 */
3113 if (spec->gpio_led)
Takashi Iwai7639a062015-03-03 10:07:24 +01003114 snd_hda_codec_write(codec, codec->core.afg, 0, AC_VERB_SET_GPIO_DATA,
Hui Wangf4753712014-08-19 12:07:03 +08003115 spec->gpio_led);
3116
Kailang Yangad60d502013-06-28 12:03:01 +02003117 if (spec->has_alc5505_dsp)
Takashi Iwaicd63a5f2013-07-05 12:13:59 +02003118 alc5505_dsp_resume(codec);
Kailang Yangc5177c82013-07-24 14:39:49 +02003119
Takashi Iwai1d045db2011-07-07 18:23:21 +02003120 return 0;
3121}
Takashi Iwai2a439522011-07-26 09:52:50 +02003122#endif /* CONFIG_PM */
Takashi Iwai1d045db2011-07-07 18:23:21 +02003123
David Henningsson108cc102012-07-20 10:37:25 +02003124static void alc269_fixup_pincfg_no_hp_to_lineout(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01003125 const struct hda_fixup *fix, int action)
David Henningsson108cc102012-07-20 10:37:25 +02003126{
3127 struct alc_spec *spec = codec->spec;
3128
Takashi Iwai1727a772013-01-10 09:52:52 +01003129 if (action == HDA_FIXUP_ACT_PRE_PROBE)
David Henningsson108cc102012-07-20 10:37:25 +02003130 spec->parse_flags = HDA_PINCFG_NO_HP_FIXUP;
3131}
3132
Takashi Iwai1d045db2011-07-07 18:23:21 +02003133static void alc269_fixup_hweq(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01003134 const struct hda_fixup *fix, int action)
Takashi Iwai1d045db2011-07-07 18:23:21 +02003135{
Takashi Iwai98b24882014-08-18 13:47:50 +02003136 if (action == HDA_FIXUP_ACT_INIT)
3137 alc_update_coef_idx(codec, 0x1e, 0, 0x80);
Takashi Iwai1d045db2011-07-07 18:23:21 +02003138}
3139
David Henningsson7c478f02013-10-11 10:18:46 +02003140static void alc269_fixup_headset_mic(struct hda_codec *codec,
3141 const struct hda_fixup *fix, int action)
3142{
3143 struct alc_spec *spec = codec->spec;
3144
3145 if (action == HDA_FIXUP_ACT_PRE_PROBE)
3146 spec->parse_flags |= HDA_PINCFG_HEADSET_MIC;
3147}
3148
Takashi Iwai1d045db2011-07-07 18:23:21 +02003149static void alc271_fixup_dmic(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01003150 const struct hda_fixup *fix, int action)
Takashi Iwai1d045db2011-07-07 18:23:21 +02003151{
3152 static const struct hda_verb verbs[] = {
3153 {0x20, AC_VERB_SET_COEF_INDEX, 0x0d},
3154 {0x20, AC_VERB_SET_PROC_COEF, 0x4000},
3155 {}
3156 };
3157 unsigned int cfg;
3158
Takashi Iwai7639a062015-03-03 10:07:24 +01003159 if (strcmp(codec->core.chip_name, "ALC271X") &&
3160 strcmp(codec->core.chip_name, "ALC269VB"))
Takashi Iwai1d045db2011-07-07 18:23:21 +02003161 return;
3162 cfg = snd_hda_codec_get_pincfg(codec, 0x12);
3163 if (get_defcfg_connect(cfg) == AC_JACK_PORT_FIXED)
3164 snd_hda_sequence_write(codec, verbs);
3165}
3166
Takashi Iwai017f2a12011-07-09 14:42:25 +02003167static void alc269_fixup_pcm_44k(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01003168 const struct hda_fixup *fix, int action)
Takashi Iwai017f2a12011-07-09 14:42:25 +02003169{
3170 struct alc_spec *spec = codec->spec;
3171
Takashi Iwai1727a772013-01-10 09:52:52 +01003172 if (action != HDA_FIXUP_ACT_PROBE)
Takashi Iwai017f2a12011-07-09 14:42:25 +02003173 return;
3174
3175 /* Due to a hardware problem on Lenovo Ideadpad, we need to
3176 * fix the sample rate of analog I/O to 44.1kHz
3177 */
Takashi Iwai08c189f2012-12-19 15:22:24 +01003178 spec->gen.stream_analog_playback = &alc269_44k_pcm_analog_playback;
3179 spec->gen.stream_analog_capture = &alc269_44k_pcm_analog_capture;
Takashi Iwai017f2a12011-07-09 14:42:25 +02003180}
3181
Takashi Iwaiadabb3e2011-08-03 07:48:37 +02003182static void alc269_fixup_stereo_dmic(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01003183 const struct hda_fixup *fix, int action)
Takashi Iwaiadabb3e2011-08-03 07:48:37 +02003184{
Takashi Iwaiadabb3e2011-08-03 07:48:37 +02003185 /* The digital-mic unit sends PDM (differential signal) instead of
3186 * the standard PCM, thus you can't record a valid mono stream as is.
3187 * Below is a workaround specific to ALC269 to control the dmic
3188 * signal source as mono.
3189 */
Takashi Iwai98b24882014-08-18 13:47:50 +02003190 if (action == HDA_FIXUP_ACT_INIT)
3191 alc_update_coef_idx(codec, 0x07, 0, 0x80);
Takashi Iwaiadabb3e2011-08-03 07:48:37 +02003192}
3193
Takashi Iwai24519912011-08-16 15:08:49 +02003194static void alc269_quanta_automute(struct hda_codec *codec)
3195{
Takashi Iwai08c189f2012-12-19 15:22:24 +01003196 snd_hda_gen_update_outputs(codec);
Takashi Iwai24519912011-08-16 15:08:49 +02003197
Takashi Iwai1687ccc2014-08-18 13:49:35 +02003198 alc_write_coef_idx(codec, 0x0c, 0x680);
3199 alc_write_coef_idx(codec, 0x0c, 0x480);
Takashi Iwai24519912011-08-16 15:08:49 +02003200}
3201
3202static void alc269_fixup_quanta_mute(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01003203 const struct hda_fixup *fix, int action)
Takashi Iwai24519912011-08-16 15:08:49 +02003204{
3205 struct alc_spec *spec = codec->spec;
Takashi Iwai1727a772013-01-10 09:52:52 +01003206 if (action != HDA_FIXUP_ACT_PROBE)
Takashi Iwai24519912011-08-16 15:08:49 +02003207 return;
Takashi Iwai08c189f2012-12-19 15:22:24 +01003208 spec->gen.automute_hook = alc269_quanta_automute;
Takashi Iwai24519912011-08-16 15:08:49 +02003209}
3210
David Henningssond240d1d2013-04-15 12:50:02 +02003211static void alc269_x101_hp_automute_hook(struct hda_codec *codec,
Takashi Iwai1a4f69d2014-09-11 15:22:46 +02003212 struct hda_jack_callback *jack)
David Henningssond240d1d2013-04-15 12:50:02 +02003213{
3214 struct alc_spec *spec = codec->spec;
3215 int vref;
3216 msleep(200);
3217 snd_hda_gen_hp_automute(codec, jack);
3218
3219 vref = spec->gen.hp_jack_present ? PIN_VREF80 : 0;
3220 msleep(100);
3221 snd_hda_codec_write(codec, 0x18, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
3222 vref);
3223 msleep(500);
3224 snd_hda_codec_write(codec, 0x18, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
3225 vref);
3226}
3227
3228static void alc269_fixup_x101_headset_mic(struct hda_codec *codec,
3229 const struct hda_fixup *fix, int action)
3230{
3231 struct alc_spec *spec = codec->spec;
3232 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
3233 spec->parse_flags |= HDA_PINCFG_HEADSET_MIC;
3234 spec->gen.hp_automute_hook = alc269_x101_hp_automute_hook;
3235 }
3236}
3237
3238
Takashi Iwai08fb0d02013-01-10 17:33:58 +01003239/* update mute-LED according to the speaker mute state via mic VREF pin */
3240static void alc269_fixup_mic_mute_hook(void *private_data, int enabled)
David Henningsson6d3cd5d2013-01-07 12:03:47 +01003241{
3242 struct hda_codec *codec = private_data;
Takashi Iwai08fb0d02013-01-10 17:33:58 +01003243 struct alc_spec *spec = codec->spec;
3244 unsigned int pinval;
3245
3246 if (spec->mute_led_polarity)
3247 enabled = !enabled;
Takashi Iwai415d5552014-04-03 11:51:21 +02003248 pinval = snd_hda_codec_get_pin_target(codec, spec->mute_led_nid);
3249 pinval &= ~AC_PINCTL_VREFEN;
3250 pinval |= enabled ? AC_PINCTL_VREF_HIZ : AC_PINCTL_VREF_80;
Takashi Iwai08fb0d02013-01-10 17:33:58 +01003251 if (spec->mute_led_nid)
3252 snd_hda_set_pin_ctl_cache(codec, spec->mute_led_nid, pinval);
David Henningsson6d3cd5d2013-01-07 12:03:47 +01003253}
3254
David Henningssond5b6b652013-11-06 10:50:44 +01003255/* Make sure the led works even in runtime suspend */
3256static unsigned int led_power_filter(struct hda_codec *codec,
3257 hda_nid_t nid,
3258 unsigned int power_state)
3259{
3260 struct alc_spec *spec = codec->spec;
3261
Hui Wang50dd9052014-07-08 17:56:15 +08003262 if (power_state != AC_PWRST_D3 || nid == 0 ||
3263 (nid != spec->mute_led_nid && nid != spec->cap_mute_led_nid))
David Henningssond5b6b652013-11-06 10:50:44 +01003264 return power_state;
3265
3266 /* Set pin ctl again, it might have just been set to 0 */
3267 snd_hda_set_pin_ctl(codec, nid,
3268 snd_hda_codec_get_pin_target(codec, nid));
3269
Takashi Iwaicffd3962015-04-09 10:30:25 +02003270 return snd_hda_gen_path_power_filter(codec, nid, power_state);
David Henningssond5b6b652013-11-06 10:50:44 +01003271}
3272
Takashi Iwai08fb0d02013-01-10 17:33:58 +01003273static void alc269_fixup_hp_mute_led(struct hda_codec *codec,
3274 const struct hda_fixup *fix, int action)
David Henningsson6d3cd5d2013-01-07 12:03:47 +01003275{
3276 struct alc_spec *spec = codec->spec;
Takashi Iwai08fb0d02013-01-10 17:33:58 +01003277 const struct dmi_device *dev = NULL;
3278
3279 if (action != HDA_FIXUP_ACT_PRE_PROBE)
3280 return;
3281
3282 while ((dev = dmi_find_device(DMI_DEV_TYPE_OEM_STRING, NULL, dev))) {
3283 int pol, pin;
3284 if (sscanf(dev->name, "HP_Mute_LED_%d_%x", &pol, &pin) != 2)
3285 continue;
3286 if (pin < 0x0a || pin >= 0x10)
3287 break;
3288 spec->mute_led_polarity = pol;
3289 spec->mute_led_nid = pin - 0x0a + 0x18;
3290 spec->gen.vmaster_mute.hook = alc269_fixup_mic_mute_hook;
Takashi Iwaifd25a972012-12-20 14:57:18 +01003291 spec->gen.vmaster_mute_enum = 1;
David Henningssond5b6b652013-11-06 10:50:44 +01003292 codec->power_filter = led_power_filter;
Takashi Iwai4e76a882014-02-25 12:21:03 +01003293 codec_dbg(codec,
3294 "Detected mute LED for %x:%d\n", spec->mute_led_nid,
Takashi Iwai08fb0d02013-01-10 17:33:58 +01003295 spec->mute_led_polarity);
David Henningsson6d3cd5d2013-01-07 12:03:47 +01003296 break;
3297 }
3298}
3299
David Henningssond06ac142013-02-18 11:41:55 +01003300static void alc269_fixup_hp_mute_led_mic1(struct hda_codec *codec,
3301 const struct hda_fixup *fix, int action)
3302{
3303 struct alc_spec *spec = codec->spec;
3304 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
3305 spec->mute_led_polarity = 0;
3306 spec->mute_led_nid = 0x18;
3307 spec->gen.vmaster_mute.hook = alc269_fixup_mic_mute_hook;
3308 spec->gen.vmaster_mute_enum = 1;
David Henningssond5b6b652013-11-06 10:50:44 +01003309 codec->power_filter = led_power_filter;
David Henningssond06ac142013-02-18 11:41:55 +01003310 }
3311}
3312
Takashi Iwai08fb0d02013-01-10 17:33:58 +01003313static void alc269_fixup_hp_mute_led_mic2(struct hda_codec *codec,
3314 const struct hda_fixup *fix, int action)
Takashi Iwai420b0fe2012-03-12 12:35:27 +01003315{
3316 struct alc_spec *spec = codec->spec;
Takashi Iwai9bb1f062013-01-10 17:14:29 +01003317 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwai08fb0d02013-01-10 17:33:58 +01003318 spec->mute_led_polarity = 0;
3319 spec->mute_led_nid = 0x19;
3320 spec->gen.vmaster_mute.hook = alc269_fixup_mic_mute_hook;
Takashi Iwaifd25a972012-12-20 14:57:18 +01003321 spec->gen.vmaster_mute_enum = 1;
David Henningssond5b6b652013-11-06 10:50:44 +01003322 codec->power_filter = led_power_filter;
Takashi Iwai420b0fe2012-03-12 12:35:27 +01003323 }
3324}
3325
Takashi Iwai0f32fd192014-11-19 12:16:14 +01003326/* update LED status via GPIO */
3327static void alc_update_gpio_led(struct hda_codec *codec, unsigned int mask,
3328 bool enabled)
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003329{
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003330 struct alc_spec *spec = codec->spec;
3331 unsigned int oldval = spec->gpio_led;
3332
Takashi Iwai0f32fd192014-11-19 12:16:14 +01003333 if (spec->mute_led_polarity)
3334 enabled = !enabled;
3335
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003336 if (enabled)
Takashi Iwai0f32fd192014-11-19 12:16:14 +01003337 spec->gpio_led &= ~mask;
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003338 else
Takashi Iwai0f32fd192014-11-19 12:16:14 +01003339 spec->gpio_led |= mask;
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003340 if (spec->gpio_led != oldval)
3341 snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DATA,
3342 spec->gpio_led);
3343}
3344
Takashi Iwai0f32fd192014-11-19 12:16:14 +01003345/* turn on/off mute LED via GPIO per vmaster hook */
3346static void alc_fixup_gpio_mute_hook(void *private_data, int enabled)
3347{
3348 struct hda_codec *codec = private_data;
3349 struct alc_spec *spec = codec->spec;
3350
3351 alc_update_gpio_led(codec, spec->gpio_mute_led_mask, enabled);
3352}
3353
3354/* turn on/off mic-mute LED via GPIO per capture hook */
3355static void alc_fixup_gpio_mic_mute_hook(struct hda_codec *codec,
3356 struct snd_kcontrol *kcontrol,
3357 struct snd_ctl_elem_value *ucontrol)
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003358{
3359 struct alc_spec *spec = codec->spec;
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003360
Takashi Iwai0f32fd192014-11-19 12:16:14 +01003361 if (ucontrol)
3362 alc_update_gpio_led(codec, spec->gpio_mic_led_mask,
3363 ucontrol->value.integer.value[0] ||
3364 ucontrol->value.integer.value[1]);
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003365}
3366
3367static void alc269_fixup_hp_gpio_led(struct hda_codec *codec,
3368 const struct hda_fixup *fix, int action)
3369{
3370 struct alc_spec *spec = codec->spec;
3371 static const struct hda_verb gpio_init[] = {
3372 { 0x01, AC_VERB_SET_GPIO_MASK, 0x18 },
3373 { 0x01, AC_VERB_SET_GPIO_DIRECTION, 0x18 },
3374 {}
3375 };
3376
3377 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwai0f32fd192014-11-19 12:16:14 +01003378 spec->gen.vmaster_mute.hook = alc_fixup_gpio_mute_hook;
3379 spec->gen.cap_sync_hook = alc_fixup_gpio_mic_mute_hook;
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003380 spec->gpio_led = 0;
Takashi Iwai0f32fd192014-11-19 12:16:14 +01003381 spec->mute_led_polarity = 0;
3382 spec->gpio_mute_led_mask = 0x08;
3383 spec->gpio_mic_led_mask = 0x10;
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003384 snd_hda_add_verbs(codec, gpio_init);
3385 }
3386}
3387
Kailang Yangeaa8e5e2014-11-21 15:49:11 +08003388static void alc286_fixup_hp_gpio_led(struct hda_codec *codec,
3389 const struct hda_fixup *fix, int action)
3390{
3391 struct alc_spec *spec = codec->spec;
3392 static const struct hda_verb gpio_init[] = {
3393 { 0x01, AC_VERB_SET_GPIO_MASK, 0x22 },
3394 { 0x01, AC_VERB_SET_GPIO_DIRECTION, 0x22 },
3395 {}
3396 };
3397
3398 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
3399 spec->gen.vmaster_mute.hook = alc_fixup_gpio_mute_hook;
3400 spec->gen.cap_sync_hook = alc_fixup_gpio_mic_mute_hook;
3401 spec->gpio_led = 0;
3402 spec->mute_led_polarity = 0;
3403 spec->gpio_mute_led_mask = 0x02;
3404 spec->gpio_mic_led_mask = 0x20;
Takashi Iwai1d045db2011-07-07 18:23:21 +02003405 snd_hda_add_verbs(codec, gpio_init);
3406 }
3407}
3408
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08003409/* turn on/off mic-mute LED per capture hook */
3410static void alc269_fixup_hp_cap_mic_mute_hook(struct hda_codec *codec,
3411 struct snd_kcontrol *kcontrol,
3412 struct snd_ctl_elem_value *ucontrol)
3413{
3414 struct alc_spec *spec = codec->spec;
3415 unsigned int pinval, enable, disable;
3416
Hui Wangfc1fad92014-07-08 17:56:14 +08003417 pinval = snd_hda_codec_get_pin_target(codec, spec->cap_mute_led_nid);
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08003418 pinval &= ~AC_PINCTL_VREFEN;
3419 enable = pinval | AC_PINCTL_VREF_80;
3420 disable = pinval | AC_PINCTL_VREF_HIZ;
3421
3422 if (!ucontrol)
3423 return;
3424
3425 if (ucontrol->value.integer.value[0] ||
3426 ucontrol->value.integer.value[1])
3427 pinval = disable;
3428 else
3429 pinval = enable;
3430
3431 if (spec->cap_mute_led_nid)
3432 snd_hda_set_pin_ctl_cache(codec, spec->cap_mute_led_nid, pinval);
3433}
3434
3435static void alc269_fixup_hp_gpio_mic1_led(struct hda_codec *codec,
3436 const struct hda_fixup *fix, int action)
3437{
3438 struct alc_spec *spec = codec->spec;
3439 static const struct hda_verb gpio_init[] = {
3440 { 0x01, AC_VERB_SET_GPIO_MASK, 0x08 },
3441 { 0x01, AC_VERB_SET_GPIO_DIRECTION, 0x08 },
3442 {}
3443 };
3444
3445 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwai0f32fd192014-11-19 12:16:14 +01003446 spec->gen.vmaster_mute.hook = alc_fixup_gpio_mute_hook;
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08003447 spec->gen.cap_sync_hook = alc269_fixup_hp_cap_mic_mute_hook;
3448 spec->gpio_led = 0;
Takashi Iwai0f32fd192014-11-19 12:16:14 +01003449 spec->mute_led_polarity = 0;
3450 spec->gpio_mute_led_mask = 0x08;
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08003451 spec->cap_mute_led_nid = 0x18;
3452 snd_hda_add_verbs(codec, gpio_init);
Hui Wang50dd9052014-07-08 17:56:15 +08003453 codec->power_filter = led_power_filter;
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08003454 }
3455}
3456
David Henningsson7a5255f2014-10-30 08:26:01 +01003457static void alc280_fixup_hp_gpio4(struct hda_codec *codec,
3458 const struct hda_fixup *fix, int action)
3459{
3460 /* Like hp_gpio_mic1_led, but also needs GPIO4 low to enable headphone amp */
3461 struct alc_spec *spec = codec->spec;
3462 static const struct hda_verb gpio_init[] = {
3463 { 0x01, AC_VERB_SET_GPIO_MASK, 0x18 },
3464 { 0x01, AC_VERB_SET_GPIO_DIRECTION, 0x18 },
3465 {}
3466 };
3467
3468 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwai0f32fd192014-11-19 12:16:14 +01003469 spec->gen.vmaster_mute.hook = alc_fixup_gpio_mute_hook;
David Henningsson7a5255f2014-10-30 08:26:01 +01003470 spec->gen.cap_sync_hook = alc269_fixup_hp_cap_mic_mute_hook;
3471 spec->gpio_led = 0;
Takashi Iwai0f32fd192014-11-19 12:16:14 +01003472 spec->mute_led_polarity = 0;
3473 spec->gpio_mute_led_mask = 0x08;
David Henningsson7a5255f2014-10-30 08:26:01 +01003474 spec->cap_mute_led_nid = 0x18;
3475 snd_hda_add_verbs(codec, gpio_init);
3476 codec->power_filter = led_power_filter;
3477 }
3478}
3479
David Henningsson33f4acd2015-01-07 15:50:13 +01003480static void gpio2_mic_hotkey_event(struct hda_codec *codec,
3481 struct hda_jack_callback *event)
3482{
3483 struct alc_spec *spec = codec->spec;
3484
3485 /* GPIO2 just toggles on a keypress/keyrelease cycle. Therefore
3486 send both key on and key off event for every interrupt. */
Hui Wangc7b60a82015-12-28 11:35:25 +08003487 input_report_key(spec->kb_dev, spec->alc_mute_keycode_map[ALC_KEY_MICMUTE_INDEX], 1);
David Henningsson33f4acd2015-01-07 15:50:13 +01003488 input_sync(spec->kb_dev);
Hui Wangc7b60a82015-12-28 11:35:25 +08003489 input_report_key(spec->kb_dev, spec->alc_mute_keycode_map[ALC_KEY_MICMUTE_INDEX], 0);
David Henningsson33f4acd2015-01-07 15:50:13 +01003490 input_sync(spec->kb_dev);
3491}
David Henningsson33f4acd2015-01-07 15:50:13 +01003492
Kailang3694cb22015-12-28 11:35:24 +08003493static int alc_register_micmute_input_device(struct hda_codec *codec)
3494{
3495 struct alc_spec *spec = codec->spec;
Hui Wangc7b60a82015-12-28 11:35:25 +08003496 int i;
Kailang3694cb22015-12-28 11:35:24 +08003497
3498 spec->kb_dev = input_allocate_device();
3499 if (!spec->kb_dev) {
3500 codec_err(codec, "Out of memory (input_allocate_device)\n");
3501 return -ENOMEM;
3502 }
Hui Wangc7b60a82015-12-28 11:35:25 +08003503
3504 spec->alc_mute_keycode_map[ALC_KEY_MICMUTE_INDEX] = KEY_MICMUTE;
3505
Kailang3694cb22015-12-28 11:35:24 +08003506 spec->kb_dev->name = "Microphone Mute Button";
3507 spec->kb_dev->evbit[0] = BIT_MASK(EV_KEY);
Hui Wangc7b60a82015-12-28 11:35:25 +08003508 spec->kb_dev->keycodesize = sizeof(spec->alc_mute_keycode_map[0]);
3509 spec->kb_dev->keycodemax = ARRAY_SIZE(spec->alc_mute_keycode_map);
3510 spec->kb_dev->keycode = spec->alc_mute_keycode_map;
3511 for (i = 0; i < ARRAY_SIZE(spec->alc_mute_keycode_map); i++)
3512 set_bit(spec->alc_mute_keycode_map[i], spec->kb_dev->keybit);
Kailang3694cb22015-12-28 11:35:24 +08003513
3514 if (input_register_device(spec->kb_dev)) {
3515 codec_err(codec, "input_register_device failed\n");
3516 input_free_device(spec->kb_dev);
3517 spec->kb_dev = NULL;
3518 return -ENOMEM;
3519 }
3520
3521 return 0;
3522}
3523
David Henningsson33f4acd2015-01-07 15:50:13 +01003524static void alc280_fixup_hp_gpio2_mic_hotkey(struct hda_codec *codec,
3525 const struct hda_fixup *fix, int action)
3526{
David Henningsson33f4acd2015-01-07 15:50:13 +01003527 /* GPIO1 = set according to SKU external amp
3528 GPIO2 = mic mute hotkey
3529 GPIO3 = mute LED
3530 GPIO4 = mic mute LED */
3531 static const struct hda_verb gpio_init[] = {
3532 { 0x01, AC_VERB_SET_GPIO_MASK, 0x1e },
3533 { 0x01, AC_VERB_SET_GPIO_DIRECTION, 0x1a },
3534 { 0x01, AC_VERB_SET_GPIO_DATA, 0x02 },
3535 {}
3536 };
3537
3538 struct alc_spec *spec = codec->spec;
3539
3540 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Kailang3694cb22015-12-28 11:35:24 +08003541 if (alc_register_micmute_input_device(codec) != 0)
David Henningsson33f4acd2015-01-07 15:50:13 +01003542 return;
David Henningsson33f4acd2015-01-07 15:50:13 +01003543
3544 snd_hda_add_verbs(codec, gpio_init);
Takashi Iwai7639a062015-03-03 10:07:24 +01003545 snd_hda_codec_write_cache(codec, codec->core.afg, 0,
David Henningsson33f4acd2015-01-07 15:50:13 +01003546 AC_VERB_SET_GPIO_UNSOLICITED_RSP_MASK, 0x04);
Takashi Iwai7639a062015-03-03 10:07:24 +01003547 snd_hda_jack_detect_enable_callback(codec, codec->core.afg,
David Henningsson33f4acd2015-01-07 15:50:13 +01003548 gpio2_mic_hotkey_event);
3549
3550 spec->gen.vmaster_mute.hook = alc_fixup_gpio_mute_hook;
3551 spec->gen.cap_sync_hook = alc_fixup_gpio_mic_mute_hook;
3552 spec->gpio_led = 0;
3553 spec->mute_led_polarity = 0;
3554 spec->gpio_mute_led_mask = 0x08;
3555 spec->gpio_mic_led_mask = 0x10;
3556 return;
3557 }
3558
3559 if (!spec->kb_dev)
3560 return;
3561
3562 switch (action) {
3563 case HDA_FIXUP_ACT_PROBE:
3564 spec->init_amp = ALC_INIT_DEFAULT;
3565 break;
3566 case HDA_FIXUP_ACT_FREE:
3567 input_unregister_device(spec->kb_dev);
David Henningsson33f4acd2015-01-07 15:50:13 +01003568 spec->kb_dev = NULL;
3569 }
David Henningsson33f4acd2015-01-07 15:50:13 +01003570}
3571
Kailang3694cb22015-12-28 11:35:24 +08003572static void alc233_fixup_lenovo_line2_mic_hotkey(struct hda_codec *codec,
3573 const struct hda_fixup *fix, int action)
3574{
3575 /* Line2 = mic mute hotkey
3576 GPIO2 = mic mute LED */
3577 static const struct hda_verb gpio_init[] = {
3578 { 0x01, AC_VERB_SET_GPIO_MASK, 0x04 },
3579 { 0x01, AC_VERB_SET_GPIO_DIRECTION, 0x04 },
3580 {}
3581 };
3582
3583 struct alc_spec *spec = codec->spec;
3584
3585 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
3586 if (alc_register_micmute_input_device(codec) != 0)
3587 return;
3588
3589 snd_hda_add_verbs(codec, gpio_init);
3590 snd_hda_jack_detect_enable_callback(codec, 0x1b,
3591 gpio2_mic_hotkey_event);
3592
3593 spec->gen.cap_sync_hook = alc_fixup_gpio_mic_mute_hook;
3594 spec->gpio_led = 0;
3595 spec->mute_led_polarity = 0;
3596 spec->gpio_mic_led_mask = 0x04;
3597 return;
3598 }
3599
3600 if (!spec->kb_dev)
3601 return;
3602
3603 switch (action) {
3604 case HDA_FIXUP_ACT_PROBE:
3605 spec->init_amp = ALC_INIT_DEFAULT;
3606 break;
3607 case HDA_FIXUP_ACT_FREE:
3608 input_unregister_device(spec->kb_dev);
3609 spec->kb_dev = NULL;
3610 }
3611}
3612
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08003613static void alc269_fixup_hp_line1_mic1_led(struct hda_codec *codec,
3614 const struct hda_fixup *fix, int action)
3615{
3616 struct alc_spec *spec = codec->spec;
3617
3618 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
3619 spec->gen.vmaster_mute.hook = alc269_fixup_mic_mute_hook;
3620 spec->gen.cap_sync_hook = alc269_fixup_hp_cap_mic_mute_hook;
3621 spec->mute_led_polarity = 0;
3622 spec->mute_led_nid = 0x1a;
3623 spec->cap_mute_led_nid = 0x18;
3624 spec->gen.vmaster_mute_enum = 1;
3625 codec->power_filter = led_power_filter;
3626 }
3627}
3628
David Henningsson73bdd592013-04-15 15:44:14 +02003629static void alc_headset_mode_unplugged(struct hda_codec *codec)
3630{
Takashi Iwai54db6c32014-08-18 15:11:19 +02003631 static struct coef_fw coef0255[] = {
Takashi Iwai54db6c32014-08-18 15:11:19 +02003632 WRITE_COEF(0x45, 0xd089), /* UAJ function set to menual mode */
3633 UPDATE_COEFEX(0x57, 0x05, 1<<14, 0), /* Direct Drive HP Amp control(Set to verb control)*/
3634 WRITE_COEF(0x06, 0x6104), /* Set MIC2 Vref gate with HP */
3635 WRITE_COEFEX(0x57, 0x03, 0x8aa6), /* Direct Drive HP Amp control */
3636 {}
3637 };
Kailang Yange69e7e02016-05-30 15:58:28 +08003638 static struct coef_fw coef0255_1[] = {
3639 WRITE_COEF(0x1b, 0x0c0b), /* LDO and MISC control */
3640 {}
3641 };
3642 static struct coef_fw coef0256[] = {
3643 WRITE_COEF(0x1b, 0x0c4b), /* LDO and MISC control */
3644 {}
3645 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02003646 static struct coef_fw coef0233[] = {
3647 WRITE_COEF(0x1b, 0x0c0b),
3648 WRITE_COEF(0x45, 0xc429),
3649 UPDATE_COEF(0x35, 0x4000, 0),
3650 WRITE_COEF(0x06, 0x2104),
3651 WRITE_COEF(0x1a, 0x0001),
3652 WRITE_COEF(0x26, 0x0004),
3653 WRITE_COEF(0x32, 0x42a3),
3654 {}
3655 };
Kailang Yangf3b70332015-04-08 15:01:17 +08003656 static struct coef_fw coef0288[] = {
3657 UPDATE_COEF(0x4f, 0xfcc0, 0xc400),
3658 UPDATE_COEF(0x50, 0x2000, 0x2000),
3659 UPDATE_COEF(0x56, 0x0006, 0x0006),
3660 UPDATE_COEF(0x66, 0x0008, 0),
3661 UPDATE_COEF(0x67, 0x2000, 0),
3662 {}
3663 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02003664 static struct coef_fw coef0292[] = {
3665 WRITE_COEF(0x76, 0x000e),
3666 WRITE_COEF(0x6c, 0x2400),
3667 WRITE_COEF(0x18, 0x7308),
3668 WRITE_COEF(0x6b, 0xc429),
3669 {}
3670 };
3671 static struct coef_fw coef0293[] = {
3672 UPDATE_COEF(0x10, 7<<8, 6<<8), /* SET Line1 JD to 0 */
3673 UPDATE_COEFEX(0x57, 0x05, 1<<15|1<<13, 0x0), /* SET charge pump by verb */
3674 UPDATE_COEFEX(0x57, 0x03, 1<<10, 1<<10), /* SET EN_OSW to 1 */
3675 UPDATE_COEF(0x1a, 1<<3, 1<<3), /* Combo JD gating with LINE1-VREFO */
3676 WRITE_COEF(0x45, 0xc429), /* Set to TRS type */
3677 UPDATE_COEF(0x4a, 0x000f, 0x000e), /* Combo Jack auto detect */
3678 {}
3679 };
3680 static struct coef_fw coef0668[] = {
3681 WRITE_COEF(0x15, 0x0d40),
3682 WRITE_COEF(0xb7, 0x802b),
3683 {}
3684 };
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08003685 static struct coef_fw coef0225[] = {
3686 UPDATE_COEF(0x4a, 1<<8, 0),
3687 UPDATE_COEFEX(0x57, 0x05, 1<<14, 0),
3688 UPDATE_COEF(0x63, 3<<14, 3<<14),
3689 UPDATE_COEF(0x4a, 3<<4, 2<<4),
3690 UPDATE_COEF(0x4a, 3<<10, 3<<10),
3691 UPDATE_COEF(0x45, 0x3f<<10, 0x34<<10),
3692 UPDATE_COEF(0x4a, 3<<10, 0),
3693 {}
3694 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02003695
Takashi Iwai7639a062015-03-03 10:07:24 +01003696 switch (codec->core.vendor_id) {
Kailang Yang9a22a8f2013-11-08 15:54:49 +08003697 case 0x10ec0255:
Kailang Yange69e7e02016-05-30 15:58:28 +08003698 alc_process_coef_fw(codec, coef0255_1);
3699 alc_process_coef_fw(codec, coef0255);
3700 break;
Kailang Yang61ae3fb2017-10-20 15:06:34 +08003701 case 0x10ec0236:
Kailang Yang7081adf2015-03-30 17:05:37 +08003702 case 0x10ec0256:
Kailang Yange69e7e02016-05-30 15:58:28 +08003703 alc_process_coef_fw(codec, coef0256);
Takashi Iwai54db6c32014-08-18 15:11:19 +02003704 alc_process_coef_fw(codec, coef0255);
Kailang Yang9a22a8f2013-11-08 15:54:49 +08003705 break;
Kailang Yang13fd08a2014-05-21 16:49:48 +08003706 case 0x10ec0233:
David Henningsson73bdd592013-04-15 15:44:14 +02003707 case 0x10ec0283:
Takashi Iwai54db6c32014-08-18 15:11:19 +02003708 alc_process_coef_fw(codec, coef0233);
David Henningsson73bdd592013-04-15 15:44:14 +02003709 break;
Kailang Yangf3b70332015-04-08 15:01:17 +08003710 case 0x10ec0286:
3711 case 0x10ec0288:
Kailang Yang1a5bc8d2015-05-12 17:11:10 +08003712 case 0x10ec0298:
Kailang Yangf3b70332015-04-08 15:01:17 +08003713 alc_process_coef_fw(codec, coef0288);
3714 break;
David Henningsson73bdd592013-04-15 15:44:14 +02003715 case 0x10ec0292:
Takashi Iwai54db6c32014-08-18 15:11:19 +02003716 alc_process_coef_fw(codec, coef0292);
David Henningsson73bdd592013-04-15 15:44:14 +02003717 break;
Kailang Yanga22aa262014-04-23 17:34:28 +08003718 case 0x10ec0293:
Takashi Iwai54db6c32014-08-18 15:11:19 +02003719 alc_process_coef_fw(codec, coef0293);
Kailang Yanga22aa262014-04-23 17:34:28 +08003720 break;
David Henningsson73bdd592013-04-15 15:44:14 +02003721 case 0x10ec0668:
Takashi Iwai54db6c32014-08-18 15:11:19 +02003722 alc_process_coef_fw(codec, coef0668);
David Henningsson73bdd592013-04-15 15:44:14 +02003723 break;
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08003724 case 0x10ec0225:
Kailang Yang7d727862016-05-24 16:46:07 +08003725 case 0x10ec0295:
Kailang Yangf6e94c22017-01-04 14:49:07 +08003726 case 0x10ec0299:
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08003727 alc_process_coef_fw(codec, coef0225);
3728 break;
Kailang Yang78f4f7c2016-06-07 11:31:34 +08003729 case 0x10ec0867:
3730 alc_update_coefex_idx(codec, 0x57, 0x5, 1<<14, 0);
3731 break;
David Henningsson73bdd592013-04-15 15:44:14 +02003732 }
Takashi Iwai4e76a882014-02-25 12:21:03 +01003733 codec_dbg(codec, "Headset jack set to unplugged mode.\n");
David Henningsson73bdd592013-04-15 15:44:14 +02003734}
3735
3736
3737static void alc_headset_mode_mic_in(struct hda_codec *codec, hda_nid_t hp_pin,
3738 hda_nid_t mic_pin)
3739{
Takashi Iwai54db6c32014-08-18 15:11:19 +02003740 static struct coef_fw coef0255[] = {
3741 WRITE_COEFEX(0x57, 0x03, 0x8aa6),
3742 WRITE_COEF(0x06, 0x6100), /* Set MIC2 Vref gate to normal */
3743 {}
3744 };
3745 static struct coef_fw coef0233[] = {
3746 UPDATE_COEF(0x35, 0, 1<<14),
3747 WRITE_COEF(0x06, 0x2100),
3748 WRITE_COEF(0x1a, 0x0021),
3749 WRITE_COEF(0x26, 0x008c),
3750 {}
3751 };
Kailang Yangf3b70332015-04-08 15:01:17 +08003752 static struct coef_fw coef0288[] = {
3753 UPDATE_COEF(0x50, 0x2000, 0),
3754 UPDATE_COEF(0x56, 0x0006, 0),
3755 UPDATE_COEF(0x4f, 0xfcc0, 0xc400),
3756 UPDATE_COEF(0x66, 0x0008, 0x0008),
3757 UPDATE_COEF(0x67, 0x2000, 0x2000),
3758 {}
3759 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02003760 static struct coef_fw coef0292[] = {
3761 WRITE_COEF(0x19, 0xa208),
3762 WRITE_COEF(0x2e, 0xacf0),
3763 {}
3764 };
3765 static struct coef_fw coef0293[] = {
3766 UPDATE_COEFEX(0x57, 0x05, 0, 1<<15|1<<13), /* SET charge pump by verb */
3767 UPDATE_COEFEX(0x57, 0x03, 1<<10, 0), /* SET EN_OSW to 0 */
3768 UPDATE_COEF(0x1a, 1<<3, 0), /* Combo JD gating without LINE1-VREFO */
3769 {}
3770 };
3771 static struct coef_fw coef0688[] = {
3772 WRITE_COEF(0xb7, 0x802b),
3773 WRITE_COEF(0xb5, 0x1040),
3774 UPDATE_COEF(0xc3, 0, 1<<12),
3775 {}
3776 };
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08003777 static struct coef_fw coef0225[] = {
3778 UPDATE_COEFEX(0x57, 0x05, 1<<14, 1<<14),
3779 UPDATE_COEF(0x4a, 3<<4, 2<<4),
3780 UPDATE_COEF(0x63, 3<<14, 0),
3781 {}
3782 };
3783
Takashi Iwai54db6c32014-08-18 15:11:19 +02003784
Takashi Iwai7639a062015-03-03 10:07:24 +01003785 switch (codec->core.vendor_id) {
Kailang Yang61ae3fb2017-10-20 15:06:34 +08003786 case 0x10ec0236:
Kailang Yang9a22a8f2013-11-08 15:54:49 +08003787 case 0x10ec0255:
Kailang Yang7081adf2015-03-30 17:05:37 +08003788 case 0x10ec0256:
Kailang Yang9a22a8f2013-11-08 15:54:49 +08003789 alc_write_coef_idx(codec, 0x45, 0xc489);
3790 snd_hda_set_pin_ctl_cache(codec, hp_pin, 0);
Takashi Iwai54db6c32014-08-18 15:11:19 +02003791 alc_process_coef_fw(codec, coef0255);
Kailang Yang9a22a8f2013-11-08 15:54:49 +08003792 snd_hda_set_pin_ctl_cache(codec, mic_pin, PIN_VREF50);
3793 break;
Kailang Yang13fd08a2014-05-21 16:49:48 +08003794 case 0x10ec0233:
David Henningsson73bdd592013-04-15 15:44:14 +02003795 case 0x10ec0283:
3796 alc_write_coef_idx(codec, 0x45, 0xc429);
3797 snd_hda_set_pin_ctl_cache(codec, hp_pin, 0);
Takashi Iwai54db6c32014-08-18 15:11:19 +02003798 alc_process_coef_fw(codec, coef0233);
David Henningsson73bdd592013-04-15 15:44:14 +02003799 snd_hda_set_pin_ctl_cache(codec, mic_pin, PIN_VREF50);
3800 break;
Kailang Yangf3b70332015-04-08 15:01:17 +08003801 case 0x10ec0286:
3802 case 0x10ec0288:
Kailang Yang1a5bc8d2015-05-12 17:11:10 +08003803 case 0x10ec0298:
Kailang Yangf3b70332015-04-08 15:01:17 +08003804 alc_update_coef_idx(codec, 0x4f, 0x000c, 0);
3805 snd_hda_set_pin_ctl_cache(codec, hp_pin, 0);
3806 alc_process_coef_fw(codec, coef0288);
3807 snd_hda_set_pin_ctl_cache(codec, mic_pin, PIN_VREF50);
3808 break;
David Henningsson73bdd592013-04-15 15:44:14 +02003809 case 0x10ec0292:
3810 snd_hda_set_pin_ctl_cache(codec, hp_pin, 0);
Takashi Iwai54db6c32014-08-18 15:11:19 +02003811 alc_process_coef_fw(codec, coef0292);
David Henningsson73bdd592013-04-15 15:44:14 +02003812 break;
Kailang Yanga22aa262014-04-23 17:34:28 +08003813 case 0x10ec0293:
3814 /* Set to TRS mode */
3815 alc_write_coef_idx(codec, 0x45, 0xc429);
3816 snd_hda_set_pin_ctl_cache(codec, hp_pin, 0);
Takashi Iwai54db6c32014-08-18 15:11:19 +02003817 alc_process_coef_fw(codec, coef0293);
Kailang Yanga22aa262014-04-23 17:34:28 +08003818 snd_hda_set_pin_ctl_cache(codec, mic_pin, PIN_VREF50);
3819 break;
Kailang Yang78f4f7c2016-06-07 11:31:34 +08003820 case 0x10ec0867:
3821 alc_update_coefex_idx(codec, 0x57, 0x5, 0, 1<<14);
3822 /* fallthru */
David Henningsson1f8b46c2015-05-12 14:38:15 +02003823 case 0x10ec0662:
3824 snd_hda_set_pin_ctl_cache(codec, hp_pin, 0);
3825 snd_hda_set_pin_ctl_cache(codec, mic_pin, PIN_VREF50);
3826 break;
David Henningsson73bdd592013-04-15 15:44:14 +02003827 case 0x10ec0668:
3828 alc_write_coef_idx(codec, 0x11, 0x0001);
3829 snd_hda_set_pin_ctl_cache(codec, hp_pin, 0);
Takashi Iwai54db6c32014-08-18 15:11:19 +02003830 alc_process_coef_fw(codec, coef0688);
David Henningsson73bdd592013-04-15 15:44:14 +02003831 snd_hda_set_pin_ctl_cache(codec, mic_pin, PIN_VREF50);
3832 break;
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08003833 case 0x10ec0225:
Kailang Yang7d727862016-05-24 16:46:07 +08003834 case 0x10ec0295:
Kailang Yangf6e94c22017-01-04 14:49:07 +08003835 case 0x10ec0299:
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08003836 alc_update_coef_idx(codec, 0x45, 0x3f<<10, 0x31<<10);
3837 snd_hda_set_pin_ctl_cache(codec, hp_pin, 0);
3838 alc_process_coef_fw(codec, coef0225);
3839 snd_hda_set_pin_ctl_cache(codec, mic_pin, PIN_VREF50);
3840 break;
David Henningsson73bdd592013-04-15 15:44:14 +02003841 }
Takashi Iwai4e76a882014-02-25 12:21:03 +01003842 codec_dbg(codec, "Headset jack set to mic-in mode.\n");
David Henningsson73bdd592013-04-15 15:44:14 +02003843}
3844
3845static void alc_headset_mode_default(struct hda_codec *codec)
3846{
David Henningsson2ae95572016-02-25 09:37:05 +01003847 static struct coef_fw coef0225[] = {
3848 UPDATE_COEF(0x45, 0x3f<<10, 0x34<<10),
3849 {}
3850 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02003851 static struct coef_fw coef0255[] = {
3852 WRITE_COEF(0x45, 0xc089),
3853 WRITE_COEF(0x45, 0xc489),
3854 WRITE_COEFEX(0x57, 0x03, 0x8ea6),
3855 WRITE_COEF(0x49, 0x0049),
3856 {}
3857 };
3858 static struct coef_fw coef0233[] = {
3859 WRITE_COEF(0x06, 0x2100),
3860 WRITE_COEF(0x32, 0x4ea3),
3861 {}
3862 };
Kailang Yangf3b70332015-04-08 15:01:17 +08003863 static struct coef_fw coef0288[] = {
3864 UPDATE_COEF(0x4f, 0xfcc0, 0xc400), /* Set to TRS type */
3865 UPDATE_COEF(0x50, 0x2000, 0x2000),
3866 UPDATE_COEF(0x56, 0x0006, 0x0006),
3867 UPDATE_COEF(0x66, 0x0008, 0),
3868 UPDATE_COEF(0x67, 0x2000, 0),
3869 {}
3870 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02003871 static struct coef_fw coef0292[] = {
3872 WRITE_COEF(0x76, 0x000e),
3873 WRITE_COEF(0x6c, 0x2400),
3874 WRITE_COEF(0x6b, 0xc429),
3875 WRITE_COEF(0x18, 0x7308),
3876 {}
3877 };
3878 static struct coef_fw coef0293[] = {
3879 UPDATE_COEF(0x4a, 0x000f, 0x000e), /* Combo Jack auto detect */
3880 WRITE_COEF(0x45, 0xC429), /* Set to TRS type */
3881 UPDATE_COEF(0x1a, 1<<3, 0), /* Combo JD gating without LINE1-VREFO */
3882 {}
3883 };
3884 static struct coef_fw coef0688[] = {
3885 WRITE_COEF(0x11, 0x0041),
3886 WRITE_COEF(0x15, 0x0d40),
3887 WRITE_COEF(0xb7, 0x802b),
3888 {}
3889 };
3890
Takashi Iwai7639a062015-03-03 10:07:24 +01003891 switch (codec->core.vendor_id) {
David Henningsson2ae95572016-02-25 09:37:05 +01003892 case 0x10ec0225:
Kailang Yang7d727862016-05-24 16:46:07 +08003893 case 0x10ec0295:
Kailang Yangf6e94c22017-01-04 14:49:07 +08003894 case 0x10ec0299:
David Henningsson2ae95572016-02-25 09:37:05 +01003895 alc_process_coef_fw(codec, coef0225);
3896 break;
Kailang Yang61ae3fb2017-10-20 15:06:34 +08003897 case 0x10ec0236:
Kailang Yang9a22a8f2013-11-08 15:54:49 +08003898 case 0x10ec0255:
Kailang Yang7081adf2015-03-30 17:05:37 +08003899 case 0x10ec0256:
Takashi Iwai54db6c32014-08-18 15:11:19 +02003900 alc_process_coef_fw(codec, coef0255);
Kailang Yang9a22a8f2013-11-08 15:54:49 +08003901 break;
Kailang Yang13fd08a2014-05-21 16:49:48 +08003902 case 0x10ec0233:
David Henningsson73bdd592013-04-15 15:44:14 +02003903 case 0x10ec0283:
Takashi Iwai54db6c32014-08-18 15:11:19 +02003904 alc_process_coef_fw(codec, coef0233);
David Henningsson73bdd592013-04-15 15:44:14 +02003905 break;
Kailang Yangf3b70332015-04-08 15:01:17 +08003906 case 0x10ec0286:
3907 case 0x10ec0288:
Kailang Yang1a5bc8d2015-05-12 17:11:10 +08003908 case 0x10ec0298:
Kailang Yangf3b70332015-04-08 15:01:17 +08003909 alc_process_coef_fw(codec, coef0288);
3910 break;
David Henningsson73bdd592013-04-15 15:44:14 +02003911 case 0x10ec0292:
Takashi Iwai54db6c32014-08-18 15:11:19 +02003912 alc_process_coef_fw(codec, coef0292);
David Henningsson73bdd592013-04-15 15:44:14 +02003913 break;
Kailang Yanga22aa262014-04-23 17:34:28 +08003914 case 0x10ec0293:
Takashi Iwai54db6c32014-08-18 15:11:19 +02003915 alc_process_coef_fw(codec, coef0293);
Kailang Yanga22aa262014-04-23 17:34:28 +08003916 break;
David Henningsson73bdd592013-04-15 15:44:14 +02003917 case 0x10ec0668:
Takashi Iwai54db6c32014-08-18 15:11:19 +02003918 alc_process_coef_fw(codec, coef0688);
David Henningsson73bdd592013-04-15 15:44:14 +02003919 break;
Kailang Yang78f4f7c2016-06-07 11:31:34 +08003920 case 0x10ec0867:
3921 alc_update_coefex_idx(codec, 0x57, 0x5, 1<<14, 0);
3922 break;
David Henningsson73bdd592013-04-15 15:44:14 +02003923 }
Takashi Iwai4e76a882014-02-25 12:21:03 +01003924 codec_dbg(codec, "Headset jack set to headphone (default) mode.\n");
David Henningsson73bdd592013-04-15 15:44:14 +02003925}
3926
3927/* Iphone type */
3928static void alc_headset_mode_ctia(struct hda_codec *codec)
3929{
Takashi Iwai54db6c32014-08-18 15:11:19 +02003930 static struct coef_fw coef0255[] = {
3931 WRITE_COEF(0x45, 0xd489), /* Set to CTIA type */
3932 WRITE_COEF(0x1b, 0x0c2b),
3933 WRITE_COEFEX(0x57, 0x03, 0x8ea6),
3934 {}
3935 };
Kailang Yange69e7e02016-05-30 15:58:28 +08003936 static struct coef_fw coef0256[] = {
3937 WRITE_COEF(0x45, 0xd489), /* Set to CTIA type */
3938 WRITE_COEF(0x1b, 0x0c6b),
3939 WRITE_COEFEX(0x57, 0x03, 0x8ea6),
3940 {}
3941 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02003942 static struct coef_fw coef0233[] = {
3943 WRITE_COEF(0x45, 0xd429),
3944 WRITE_COEF(0x1b, 0x0c2b),
3945 WRITE_COEF(0x32, 0x4ea3),
3946 {}
3947 };
Kailang Yangf3b70332015-04-08 15:01:17 +08003948 static struct coef_fw coef0288[] = {
3949 UPDATE_COEF(0x50, 0x2000, 0x2000),
3950 UPDATE_COEF(0x56, 0x0006, 0x0006),
3951 UPDATE_COEF(0x66, 0x0008, 0),
3952 UPDATE_COEF(0x67, 0x2000, 0),
3953 {}
3954 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02003955 static struct coef_fw coef0292[] = {
3956 WRITE_COEF(0x6b, 0xd429),
3957 WRITE_COEF(0x76, 0x0008),
3958 WRITE_COEF(0x18, 0x7388),
3959 {}
3960 };
3961 static struct coef_fw coef0293[] = {
3962 WRITE_COEF(0x45, 0xd429), /* Set to ctia type */
3963 UPDATE_COEF(0x10, 7<<8, 7<<8), /* SET Line1 JD to 1 */
3964 {}
3965 };
3966 static struct coef_fw coef0688[] = {
3967 WRITE_COEF(0x11, 0x0001),
3968 WRITE_COEF(0x15, 0x0d60),
3969 WRITE_COEF(0xc3, 0x0000),
3970 {}
3971 };
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08003972 static struct coef_fw coef0225[] = {
3973 UPDATE_COEF(0x45, 0x3f<<10, 0x35<<10),
3974 UPDATE_COEF(0x49, 1<<8, 1<<8),
3975 UPDATE_COEF(0x4a, 7<<6, 7<<6),
3976 UPDATE_COEF(0x4a, 3<<4, 3<<4),
3977 {}
3978 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02003979
Takashi Iwai7639a062015-03-03 10:07:24 +01003980 switch (codec->core.vendor_id) {
Kailang Yang9a22a8f2013-11-08 15:54:49 +08003981 case 0x10ec0255:
Takashi Iwai54db6c32014-08-18 15:11:19 +02003982 alc_process_coef_fw(codec, coef0255);
Kailang Yang9a22a8f2013-11-08 15:54:49 +08003983 break;
Kailang Yang61ae3fb2017-10-20 15:06:34 +08003984 case 0x10ec0236:
Kailang Yange69e7e02016-05-30 15:58:28 +08003985 case 0x10ec0256:
3986 alc_process_coef_fw(codec, coef0256);
3987 break;
Kailang Yang13fd08a2014-05-21 16:49:48 +08003988 case 0x10ec0233:
David Henningsson73bdd592013-04-15 15:44:14 +02003989 case 0x10ec0283:
Takashi Iwai54db6c32014-08-18 15:11:19 +02003990 alc_process_coef_fw(codec, coef0233);
David Henningsson73bdd592013-04-15 15:44:14 +02003991 break;
Kailang Yang1a5bc8d2015-05-12 17:11:10 +08003992 case 0x10ec0298:
3993 alc_update_coef_idx(codec, 0x8e, 0x0070, 0x0020);/* Headset output enable */
3994 /* ALC298 jack type setting is the same with ALC286/ALC288 */
Kailang Yangf3b70332015-04-08 15:01:17 +08003995 case 0x10ec0286:
3996 case 0x10ec0288:
3997 alc_update_coef_idx(codec, 0x4f, 0xfcc0, 0xd400);
3998 msleep(300);
3999 alc_process_coef_fw(codec, coef0288);
4000 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004001 case 0x10ec0292:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004002 alc_process_coef_fw(codec, coef0292);
David Henningsson73bdd592013-04-15 15:44:14 +02004003 break;
Kailang Yanga22aa262014-04-23 17:34:28 +08004004 case 0x10ec0293:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004005 alc_process_coef_fw(codec, coef0293);
Kailang Yanga22aa262014-04-23 17:34:28 +08004006 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004007 case 0x10ec0668:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004008 alc_process_coef_fw(codec, coef0688);
David Henningsson73bdd592013-04-15 15:44:14 +02004009 break;
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004010 case 0x10ec0225:
Kailang Yang7d727862016-05-24 16:46:07 +08004011 case 0x10ec0295:
Kailang Yangf6e94c22017-01-04 14:49:07 +08004012 case 0x10ec0299:
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004013 alc_process_coef_fw(codec, coef0225);
4014 break;
Kailang Yang78f4f7c2016-06-07 11:31:34 +08004015 case 0x10ec0867:
4016 alc_update_coefex_idx(codec, 0x57, 0x5, 1<<14, 0);
4017 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004018 }
Takashi Iwai4e76a882014-02-25 12:21:03 +01004019 codec_dbg(codec, "Headset jack set to iPhone-style headset mode.\n");
David Henningsson73bdd592013-04-15 15:44:14 +02004020}
4021
4022/* Nokia type */
4023static void alc_headset_mode_omtp(struct hda_codec *codec)
4024{
Takashi Iwai54db6c32014-08-18 15:11:19 +02004025 static struct coef_fw coef0255[] = {
4026 WRITE_COEF(0x45, 0xe489), /* Set to OMTP Type */
4027 WRITE_COEF(0x1b, 0x0c2b),
4028 WRITE_COEFEX(0x57, 0x03, 0x8ea6),
4029 {}
4030 };
Kailang Yange69e7e02016-05-30 15:58:28 +08004031 static struct coef_fw coef0256[] = {
4032 WRITE_COEF(0x45, 0xe489), /* Set to OMTP Type */
4033 WRITE_COEF(0x1b, 0x0c6b),
4034 WRITE_COEFEX(0x57, 0x03, 0x8ea6),
4035 {}
4036 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004037 static struct coef_fw coef0233[] = {
4038 WRITE_COEF(0x45, 0xe429),
4039 WRITE_COEF(0x1b, 0x0c2b),
4040 WRITE_COEF(0x32, 0x4ea3),
4041 {}
4042 };
Kailang Yangf3b70332015-04-08 15:01:17 +08004043 static struct coef_fw coef0288[] = {
4044 UPDATE_COEF(0x50, 0x2000, 0x2000),
4045 UPDATE_COEF(0x56, 0x0006, 0x0006),
4046 UPDATE_COEF(0x66, 0x0008, 0),
4047 UPDATE_COEF(0x67, 0x2000, 0),
4048 {}
4049 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004050 static struct coef_fw coef0292[] = {
4051 WRITE_COEF(0x6b, 0xe429),
4052 WRITE_COEF(0x76, 0x0008),
4053 WRITE_COEF(0x18, 0x7388),
4054 {}
4055 };
4056 static struct coef_fw coef0293[] = {
4057 WRITE_COEF(0x45, 0xe429), /* Set to omtp type */
4058 UPDATE_COEF(0x10, 7<<8, 7<<8), /* SET Line1 JD to 1 */
4059 {}
4060 };
4061 static struct coef_fw coef0688[] = {
4062 WRITE_COEF(0x11, 0x0001),
4063 WRITE_COEF(0x15, 0x0d50),
4064 WRITE_COEF(0xc3, 0x0000),
4065 {}
4066 };
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004067 static struct coef_fw coef0225[] = {
4068 UPDATE_COEF(0x45, 0x3f<<10, 0x39<<10),
4069 UPDATE_COEF(0x49, 1<<8, 1<<8),
4070 UPDATE_COEF(0x4a, 7<<6, 7<<6),
4071 UPDATE_COEF(0x4a, 3<<4, 3<<4),
4072 {}
4073 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004074
Takashi Iwai7639a062015-03-03 10:07:24 +01004075 switch (codec->core.vendor_id) {
Kailang Yang9a22a8f2013-11-08 15:54:49 +08004076 case 0x10ec0255:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004077 alc_process_coef_fw(codec, coef0255);
Kailang Yang9a22a8f2013-11-08 15:54:49 +08004078 break;
Kailang Yang61ae3fb2017-10-20 15:06:34 +08004079 case 0x10ec0236:
Kailang Yange69e7e02016-05-30 15:58:28 +08004080 case 0x10ec0256:
4081 alc_process_coef_fw(codec, coef0256);
4082 break;
Kailang Yang13fd08a2014-05-21 16:49:48 +08004083 case 0x10ec0233:
David Henningsson73bdd592013-04-15 15:44:14 +02004084 case 0x10ec0283:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004085 alc_process_coef_fw(codec, coef0233);
David Henningsson73bdd592013-04-15 15:44:14 +02004086 break;
Kailang Yang1a5bc8d2015-05-12 17:11:10 +08004087 case 0x10ec0298:
4088 alc_update_coef_idx(codec, 0x8e, 0x0070, 0x0010);/* Headset output enable */
4089 /* ALC298 jack type setting is the same with ALC286/ALC288 */
Kailang Yangf3b70332015-04-08 15:01:17 +08004090 case 0x10ec0286:
4091 case 0x10ec0288:
4092 alc_update_coef_idx(codec, 0x4f, 0xfcc0, 0xe400);
4093 msleep(300);
4094 alc_process_coef_fw(codec, coef0288);
4095 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004096 case 0x10ec0292:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004097 alc_process_coef_fw(codec, coef0292);
David Henningsson73bdd592013-04-15 15:44:14 +02004098 break;
Kailang Yanga22aa262014-04-23 17:34:28 +08004099 case 0x10ec0293:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004100 alc_process_coef_fw(codec, coef0293);
Kailang Yanga22aa262014-04-23 17:34:28 +08004101 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004102 case 0x10ec0668:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004103 alc_process_coef_fw(codec, coef0688);
David Henningsson73bdd592013-04-15 15:44:14 +02004104 break;
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004105 case 0x10ec0225:
Kailang Yang7d727862016-05-24 16:46:07 +08004106 case 0x10ec0295:
Kailang Yangf6e94c22017-01-04 14:49:07 +08004107 case 0x10ec0299:
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004108 alc_process_coef_fw(codec, coef0225);
4109 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004110 }
Takashi Iwai4e76a882014-02-25 12:21:03 +01004111 codec_dbg(codec, "Headset jack set to Nokia-style headset mode.\n");
David Henningsson73bdd592013-04-15 15:44:14 +02004112}
4113
4114static void alc_determine_headset_type(struct hda_codec *codec)
4115{
4116 int val;
4117 bool is_ctia = false;
4118 struct alc_spec *spec = codec->spec;
Takashi Iwai54db6c32014-08-18 15:11:19 +02004119 static struct coef_fw coef0255[] = {
4120 WRITE_COEF(0x45, 0xd089), /* combo jack auto switch control(Check type)*/
4121 WRITE_COEF(0x49, 0x0149), /* combo jack auto switch control(Vref
4122 conteol) */
4123 {}
4124 };
Kailang Yangf3b70332015-04-08 15:01:17 +08004125 static struct coef_fw coef0288[] = {
4126 UPDATE_COEF(0x4f, 0xfcc0, 0xd400), /* Check Type */
4127 {}
4128 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004129 static struct coef_fw coef0293[] = {
4130 UPDATE_COEF(0x4a, 0x000f, 0x0008), /* Combo Jack auto detect */
4131 WRITE_COEF(0x45, 0xD429), /* Set to ctia type */
4132 {}
4133 };
4134 static struct coef_fw coef0688[] = {
4135 WRITE_COEF(0x11, 0x0001),
4136 WRITE_COEF(0xb7, 0x802b),
4137 WRITE_COEF(0x15, 0x0d60),
4138 WRITE_COEF(0xc3, 0x0c00),
4139 {}
4140 };
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004141 static struct coef_fw coef0225[] = {
4142 UPDATE_COEF(0x45, 0x3f<<10, 0x34<<10),
4143 UPDATE_COEF(0x49, 1<<8, 1<<8),
4144 {}
4145 };
David Henningsson73bdd592013-04-15 15:44:14 +02004146
Takashi Iwai7639a062015-03-03 10:07:24 +01004147 switch (codec->core.vendor_id) {
Kailang Yang61ae3fb2017-10-20 15:06:34 +08004148 case 0x10ec0236:
Kailang Yang9a22a8f2013-11-08 15:54:49 +08004149 case 0x10ec0255:
Kailang Yang7081adf2015-03-30 17:05:37 +08004150 case 0x10ec0256:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004151 alc_process_coef_fw(codec, coef0255);
Kailang Yang9a22a8f2013-11-08 15:54:49 +08004152 msleep(300);
4153 val = alc_read_coef_idx(codec, 0x46);
4154 is_ctia = (val & 0x0070) == 0x0070;
4155 break;
Kailang Yang13fd08a2014-05-21 16:49:48 +08004156 case 0x10ec0233:
David Henningsson73bdd592013-04-15 15:44:14 +02004157 case 0x10ec0283:
4158 alc_write_coef_idx(codec, 0x45, 0xd029);
4159 msleep(300);
4160 val = alc_read_coef_idx(codec, 0x46);
4161 is_ctia = (val & 0x0070) == 0x0070;
4162 break;
Kailang Yang1a5bc8d2015-05-12 17:11:10 +08004163 case 0x10ec0298:
4164 alc_update_coef_idx(codec, 0x8e, 0x0070, 0x0020); /* Headset output enable */
4165 /* ALC298 check jack type is the same with ALC286/ALC288 */
Kailang Yangf3b70332015-04-08 15:01:17 +08004166 case 0x10ec0286:
4167 case 0x10ec0288:
4168 alc_process_coef_fw(codec, coef0288);
4169 msleep(350);
4170 val = alc_read_coef_idx(codec, 0x50);
4171 is_ctia = (val & 0x0070) == 0x0070;
4172 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004173 case 0x10ec0292:
4174 alc_write_coef_idx(codec, 0x6b, 0xd429);
4175 msleep(300);
4176 val = alc_read_coef_idx(codec, 0x6c);
4177 is_ctia = (val & 0x001c) == 0x001c;
4178 break;
Kailang Yanga22aa262014-04-23 17:34:28 +08004179 case 0x10ec0293:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004180 alc_process_coef_fw(codec, coef0293);
Kailang Yanga22aa262014-04-23 17:34:28 +08004181 msleep(300);
4182 val = alc_read_coef_idx(codec, 0x46);
4183 is_ctia = (val & 0x0070) == 0x0070;
4184 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004185 case 0x10ec0668:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004186 alc_process_coef_fw(codec, coef0688);
David Henningsson73bdd592013-04-15 15:44:14 +02004187 msleep(300);
4188 val = alc_read_coef_idx(codec, 0xbe);
4189 is_ctia = (val & 0x1c02) == 0x1c02;
4190 break;
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004191 case 0x10ec0225:
Kailang Yang7d727862016-05-24 16:46:07 +08004192 case 0x10ec0295:
Kailang Yangf6e94c22017-01-04 14:49:07 +08004193 case 0x10ec0299:
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004194 alc_process_coef_fw(codec, coef0225);
4195 msleep(800);
4196 val = alc_read_coef_idx(codec, 0x46);
4197 is_ctia = (val & 0x00f0) == 0x00f0;
4198 break;
Kailang Yang78f4f7c2016-06-07 11:31:34 +08004199 case 0x10ec0867:
4200 is_ctia = true;
4201 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004202 }
4203
Takashi Iwai4e76a882014-02-25 12:21:03 +01004204 codec_dbg(codec, "Headset jack detected iPhone-style headset: %s\n",
David Henningsson73bdd592013-04-15 15:44:14 +02004205 is_ctia ? "yes" : "no");
4206 spec->current_headset_type = is_ctia ? ALC_HEADSET_TYPE_CTIA : ALC_HEADSET_TYPE_OMTP;
4207}
4208
4209static void alc_update_headset_mode(struct hda_codec *codec)
4210{
4211 struct alc_spec *spec = codec->spec;
4212
4213 hda_nid_t mux_pin = spec->gen.imux_pins[spec->gen.cur_mux[0]];
4214 hda_nid_t hp_pin = spec->gen.autocfg.hp_pins[0];
4215
4216 int new_headset_mode;
4217
4218 if (!snd_hda_jack_detect(codec, hp_pin))
4219 new_headset_mode = ALC_HEADSET_MODE_UNPLUGGED;
4220 else if (mux_pin == spec->headset_mic_pin)
4221 new_headset_mode = ALC_HEADSET_MODE_HEADSET;
4222 else if (mux_pin == spec->headphone_mic_pin)
4223 new_headset_mode = ALC_HEADSET_MODE_MIC;
4224 else
4225 new_headset_mode = ALC_HEADSET_MODE_HEADPHONE;
4226
David Henningsson5959a6b2013-11-12 11:10:57 +01004227 if (new_headset_mode == spec->current_headset_mode) {
4228 snd_hda_gen_update_outputs(codec);
David Henningsson73bdd592013-04-15 15:44:14 +02004229 return;
David Henningsson5959a6b2013-11-12 11:10:57 +01004230 }
David Henningsson73bdd592013-04-15 15:44:14 +02004231
4232 switch (new_headset_mode) {
4233 case ALC_HEADSET_MODE_UNPLUGGED:
4234 alc_headset_mode_unplugged(codec);
4235 spec->gen.hp_jack_present = false;
4236 break;
4237 case ALC_HEADSET_MODE_HEADSET:
4238 if (spec->current_headset_type == ALC_HEADSET_TYPE_UNKNOWN)
4239 alc_determine_headset_type(codec);
4240 if (spec->current_headset_type == ALC_HEADSET_TYPE_CTIA)
4241 alc_headset_mode_ctia(codec);
4242 else if (spec->current_headset_type == ALC_HEADSET_TYPE_OMTP)
4243 alc_headset_mode_omtp(codec);
4244 spec->gen.hp_jack_present = true;
4245 break;
4246 case ALC_HEADSET_MODE_MIC:
4247 alc_headset_mode_mic_in(codec, hp_pin, spec->headphone_mic_pin);
4248 spec->gen.hp_jack_present = false;
4249 break;
4250 case ALC_HEADSET_MODE_HEADPHONE:
4251 alc_headset_mode_default(codec);
4252 spec->gen.hp_jack_present = true;
4253 break;
4254 }
4255 if (new_headset_mode != ALC_HEADSET_MODE_MIC) {
4256 snd_hda_set_pin_ctl_cache(codec, hp_pin,
4257 AC_PINCTL_OUT_EN | AC_PINCTL_HP_EN);
David Henningsson1f8b46c2015-05-12 14:38:15 +02004258 if (spec->headphone_mic_pin && spec->headphone_mic_pin != hp_pin)
David Henningsson73bdd592013-04-15 15:44:14 +02004259 snd_hda_set_pin_ctl_cache(codec, spec->headphone_mic_pin,
4260 PIN_VREFHIZ);
4261 }
4262 spec->current_headset_mode = new_headset_mode;
4263
4264 snd_hda_gen_update_outputs(codec);
4265}
4266
4267static void alc_update_headset_mode_hook(struct hda_codec *codec,
Takashi Iwai7fe30712014-01-30 17:59:02 +01004268 struct snd_kcontrol *kcontrol,
4269 struct snd_ctl_elem_value *ucontrol)
David Henningsson73bdd592013-04-15 15:44:14 +02004270{
4271 alc_update_headset_mode(codec);
4272}
4273
Takashi Iwai1a4f69d2014-09-11 15:22:46 +02004274static void alc_update_headset_jack_cb(struct hda_codec *codec,
4275 struct hda_jack_callback *jack)
David Henningsson73bdd592013-04-15 15:44:14 +02004276{
4277 struct alc_spec *spec = codec->spec;
David Henningsson5db4d342013-11-22 12:17:06 +01004278 spec->current_headset_type = ALC_HEADSET_TYPE_UNKNOWN;
David Henningsson73bdd592013-04-15 15:44:14 +02004279 snd_hda_gen_hp_automute(codec, jack);
4280}
4281
4282static void alc_probe_headset_mode(struct hda_codec *codec)
4283{
4284 int i;
4285 struct alc_spec *spec = codec->spec;
4286 struct auto_pin_cfg *cfg = &spec->gen.autocfg;
4287
4288 /* Find mic pins */
4289 for (i = 0; i < cfg->num_inputs; i++) {
4290 if (cfg->inputs[i].is_headset_mic && !spec->headset_mic_pin)
4291 spec->headset_mic_pin = cfg->inputs[i].pin;
4292 if (cfg->inputs[i].is_headphone_mic && !spec->headphone_mic_pin)
4293 spec->headphone_mic_pin = cfg->inputs[i].pin;
4294 }
4295
4296 spec->gen.cap_sync_hook = alc_update_headset_mode_hook;
4297 spec->gen.automute_hook = alc_update_headset_mode;
4298 spec->gen.hp_automute_hook = alc_update_headset_jack_cb;
4299}
4300
4301static void alc_fixup_headset_mode(struct hda_codec *codec,
4302 const struct hda_fixup *fix, int action)
4303{
4304 struct alc_spec *spec = codec->spec;
4305
4306 switch (action) {
4307 case HDA_FIXUP_ACT_PRE_PROBE:
4308 spec->parse_flags |= HDA_PINCFG_HEADSET_MIC | HDA_PINCFG_HEADPHONE_MIC;
4309 break;
4310 case HDA_FIXUP_ACT_PROBE:
4311 alc_probe_headset_mode(codec);
4312 break;
4313 case HDA_FIXUP_ACT_INIT:
4314 spec->current_headset_mode = 0;
4315 alc_update_headset_mode(codec);
4316 break;
4317 }
4318}
4319
4320static void alc_fixup_headset_mode_no_hp_mic(struct hda_codec *codec,
4321 const struct hda_fixup *fix, int action)
4322{
4323 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
4324 struct alc_spec *spec = codec->spec;
4325 spec->parse_flags |= HDA_PINCFG_HEADSET_MIC;
4326 }
4327 else
4328 alc_fixup_headset_mode(codec, fix, action);
4329}
4330
Kailang Yang31278992014-03-03 15:27:22 +08004331static void alc255_set_default_jack_type(struct hda_codec *codec)
4332{
4333 /* Set to iphone type */
Kailang Yange69e7e02016-05-30 15:58:28 +08004334 static struct coef_fw alc255fw[] = {
Takashi Iwai54db6c32014-08-18 15:11:19 +02004335 WRITE_COEF(0x1b, 0x880b),
4336 WRITE_COEF(0x45, 0xd089),
4337 WRITE_COEF(0x1b, 0x080b),
4338 WRITE_COEF(0x46, 0x0004),
4339 WRITE_COEF(0x1b, 0x0c0b),
4340 {}
4341 };
Kailang Yange69e7e02016-05-30 15:58:28 +08004342 static struct coef_fw alc256fw[] = {
4343 WRITE_COEF(0x1b, 0x884b),
4344 WRITE_COEF(0x45, 0xd089),
4345 WRITE_COEF(0x1b, 0x084b),
4346 WRITE_COEF(0x46, 0x0004),
4347 WRITE_COEF(0x1b, 0x0c4b),
4348 {}
4349 };
4350 switch (codec->core.vendor_id) {
4351 case 0x10ec0255:
4352 alc_process_coef_fw(codec, alc255fw);
4353 break;
Kailang Yang61ae3fb2017-10-20 15:06:34 +08004354 case 0x10ec0236:
Kailang Yange69e7e02016-05-30 15:58:28 +08004355 case 0x10ec0256:
4356 alc_process_coef_fw(codec, alc256fw);
4357 break;
4358 }
Kailang Yang31278992014-03-03 15:27:22 +08004359 msleep(30);
4360}
4361
Kailang Yang9a22a8f2013-11-08 15:54:49 +08004362static void alc_fixup_headset_mode_alc255(struct hda_codec *codec,
4363 const struct hda_fixup *fix, int action)
4364{
4365 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Kailang Yang31278992014-03-03 15:27:22 +08004366 alc255_set_default_jack_type(codec);
Kailang Yang9a22a8f2013-11-08 15:54:49 +08004367 }
4368 alc_fixup_headset_mode(codec, fix, action);
4369}
4370
Kailang Yang31278992014-03-03 15:27:22 +08004371static void alc_fixup_headset_mode_alc255_no_hp_mic(struct hda_codec *codec,
4372 const struct hda_fixup *fix, int action)
4373{
4374 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
4375 struct alc_spec *spec = codec->spec;
4376 spec->parse_flags |= HDA_PINCFG_HEADSET_MIC;
4377 alc255_set_default_jack_type(codec);
4378 }
4379 else
4380 alc_fixup_headset_mode(codec, fix, action);
4381}
4382
Kailang Yange1e62b92015-04-08 16:01:22 +08004383static void alc288_update_headset_jack_cb(struct hda_codec *codec,
4384 struct hda_jack_callback *jack)
4385{
4386 struct alc_spec *spec = codec->spec;
4387 int present;
4388
4389 alc_update_headset_jack_cb(codec, jack);
4390 /* Headset Mic enable or disable, only for Dell Dino */
4391 present = spec->gen.hp_jack_present ? 0x40 : 0;
4392 snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DATA,
4393 present);
4394}
4395
4396static void alc_fixup_headset_mode_dell_alc288(struct hda_codec *codec,
4397 const struct hda_fixup *fix, int action)
4398{
4399 alc_fixup_headset_mode(codec, fix, action);
4400 if (action == HDA_FIXUP_ACT_PROBE) {
4401 struct alc_spec *spec = codec->spec;
4402 spec->gen.hp_automute_hook = alc288_update_headset_jack_cb;
4403 }
4404}
4405
Hui Wang493a52a2014-01-14 14:07:36 +08004406static void alc_fixup_auto_mute_via_amp(struct hda_codec *codec,
4407 const struct hda_fixup *fix, int action)
4408{
4409 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
4410 struct alc_spec *spec = codec->spec;
4411 spec->gen.auto_mute_via_amp = 1;
4412 }
4413}
4414
Takashi Iwai9b745ab2014-03-07 08:37:19 +01004415static void alc_no_shutup(struct hda_codec *codec)
4416{
4417}
4418
4419static void alc_fixup_no_shutup(struct hda_codec *codec,
4420 const struct hda_fixup *fix, int action)
4421{
Gabriele Mazzottad47162b2016-12-24 19:50:00 +01004422 if (action == HDA_FIXUP_ACT_PROBE) {
Takashi Iwai9b745ab2014-03-07 08:37:19 +01004423 struct alc_spec *spec = codec->spec;
4424 spec->shutup = alc_no_shutup;
4425 }
4426}
4427
Gabriele Mazzotta5e6db662014-05-20 18:58:26 +02004428static void alc_fixup_disable_aamix(struct hda_codec *codec,
4429 const struct hda_fixup *fix, int action)
4430{
4431 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
4432 struct alc_spec *spec = codec->spec;
4433 /* Disable AA-loopback as it causes white noise */
4434 spec->gen.mixer_nid = 0;
4435 }
4436}
4437
Takashi Iwai7f57d802015-09-24 17:36:51 +02004438/* fixup for Thinkpad docks: add dock pins, avoid HP parser fixup */
4439static void alc_fixup_tpt440_dock(struct hda_codec *codec,
4440 const struct hda_fixup *fix, int action)
4441{
4442 static const struct hda_pintbl pincfgs[] = {
4443 { 0x16, 0x21211010 }, /* dock headphone */
4444 { 0x19, 0x21a11010 }, /* dock mic */
4445 { }
4446 };
4447 struct alc_spec *spec = codec->spec;
4448
4449 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwai157f0b72015-12-10 23:30:43 +01004450 spec->shutup = alc_no_shutup; /* reduce click noise */
Takashi Iwai70a09762015-12-15 14:59:58 +01004451 spec->reboot_notify = alc_d3_at_reboot; /* reduce noise */
Takashi Iwai7f57d802015-09-24 17:36:51 +02004452 spec->parse_flags = HDA_PINCFG_NO_HP_FIXUP;
4453 codec->power_save_node = 0; /* avoid click noises */
4454 snd_hda_apply_pincfgs(codec, pincfgs);
4455 }
4456}
4457
Gabriele Mazzotta9476d3692015-04-19 19:00:40 +02004458static void alc_shutup_dell_xps13(struct hda_codec *codec)
Gabriele Mazzotta033b0a72014-05-26 17:11:46 +02004459{
4460 struct alc_spec *spec = codec->spec;
Gabriele Mazzotta9476d3692015-04-19 19:00:40 +02004461 int hp_pin = spec->gen.autocfg.hp_pins[0];
Gabriele Mazzotta033b0a72014-05-26 17:11:46 +02004462
Gabriele Mazzotta9476d3692015-04-19 19:00:40 +02004463 /* Prevent pop noises when headphones are plugged in */
4464 snd_hda_codec_write(codec, hp_pin, 0,
4465 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
4466 msleep(20);
Gabriele Mazzotta033b0a72014-05-26 17:11:46 +02004467}
4468
4469static void alc_fixup_dell_xps13(struct hda_codec *codec,
4470 const struct hda_fixup *fix, int action)
4471{
Takashi Iwai3e1b0c42015-04-27 10:43:22 +02004472 struct alc_spec *spec = codec->spec;
4473 struct hda_input_mux *imux = &spec->gen.input_mux;
4474 int i;
Gabriele Mazzottaf38663a2014-08-08 19:34:40 +02004475
Takashi Iwai3e1b0c42015-04-27 10:43:22 +02004476 switch (action) {
4477 case HDA_FIXUP_ACT_PRE_PROBE:
4478 /* mic pin 0x19 must be initialized with Vref Hi-Z, otherwise
4479 * it causes a click noise at start up
4480 */
4481 snd_hda_codec_set_pin_target(codec, 0x19, PIN_VREFHIZ);
4482 break;
4483 case HDA_FIXUP_ACT_PROBE:
Gabriele Mazzotta9476d3692015-04-19 19:00:40 +02004484 spec->shutup = alc_shutup_dell_xps13;
Gabriele Mazzottaf38663a2014-08-08 19:34:40 +02004485
4486 /* Make the internal mic the default input source. */
4487 for (i = 0; i < imux->num_items; i++) {
4488 if (spec->gen.imux_pins[i] == 0x12) {
4489 spec->gen.cur_mux[0] = i;
4490 break;
4491 }
4492 }
Takashi Iwai3e1b0c42015-04-27 10:43:22 +02004493 break;
Gabriele Mazzotta033b0a72014-05-26 17:11:46 +02004494 }
4495}
4496
David Henningsson1f8b46c2015-05-12 14:38:15 +02004497static void alc_fixup_headset_mode_alc662(struct hda_codec *codec,
4498 const struct hda_fixup *fix, int action)
4499{
4500 struct alc_spec *spec = codec->spec;
4501
4502 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
4503 spec->parse_flags |= HDA_PINCFG_HEADSET_MIC;
4504 spec->gen.hp_mic = 1; /* Mic-in is same pin as headphone */
David Henningssonb40eda62015-05-28 09:40:23 +02004505
4506 /* Disable boost for mic-in permanently. (This code is only called
4507 from quirks that guarantee that the headphone is at NID 0x1b.) */
4508 snd_hda_codec_write(codec, 0x1b, 0, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000);
4509 snd_hda_override_wcaps(codec, 0x1b, get_wcaps(codec, 0x1b) & ~AC_WCAP_IN_AMP);
David Henningsson1f8b46c2015-05-12 14:38:15 +02004510 } else
4511 alc_fixup_headset_mode(codec, fix, action);
4512}
4513
David Henningsson73bdd592013-04-15 15:44:14 +02004514static void alc_fixup_headset_mode_alc668(struct hda_codec *codec,
4515 const struct hda_fixup *fix, int action)
4516{
4517 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
David Henningsson73bdd592013-04-15 15:44:14 +02004518 alc_write_coef_idx(codec, 0xc4, 0x8000);
Takashi Iwai98b24882014-08-18 13:47:50 +02004519 alc_update_coef_idx(codec, 0xc2, ~0xfe, 0);
David Henningsson73bdd592013-04-15 15:44:14 +02004520 snd_hda_set_pin_ctl_cache(codec, 0x18, 0);
4521 }
4522 alc_fixup_headset_mode(codec, fix, action);
4523}
4524
Chih-Chung Changbde7bc62013-08-05 16:38:42 +08004525/* Returns the nid of the external mic input pin, or 0 if it cannot be found. */
4526static int find_ext_mic_pin(struct hda_codec *codec)
4527{
4528 struct alc_spec *spec = codec->spec;
4529 struct auto_pin_cfg *cfg = &spec->gen.autocfg;
4530 hda_nid_t nid;
4531 unsigned int defcfg;
4532 int i;
4533
4534 for (i = 0; i < cfg->num_inputs; i++) {
4535 if (cfg->inputs[i].type != AUTO_PIN_MIC)
4536 continue;
4537 nid = cfg->inputs[i].pin;
4538 defcfg = snd_hda_codec_get_pincfg(codec, nid);
4539 if (snd_hda_get_input_pin_attr(defcfg) == INPUT_PIN_ATTR_INT)
4540 continue;
4541 return nid;
4542 }
4543
4544 return 0;
4545}
4546
Dylan Reid08a978d2012-11-18 22:56:40 -08004547static void alc271_hp_gate_mic_jack(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01004548 const struct hda_fixup *fix,
Dylan Reid08a978d2012-11-18 22:56:40 -08004549 int action)
4550{
4551 struct alc_spec *spec = codec->spec;
4552
Takashi Iwai0db75792013-01-23 13:57:20 +01004553 if (action == HDA_FIXUP_ACT_PROBE) {
Chih-Chung Changbde7bc62013-08-05 16:38:42 +08004554 int mic_pin = find_ext_mic_pin(codec);
4555 int hp_pin = spec->gen.autocfg.hp_pins[0];
4556
4557 if (snd_BUG_ON(!mic_pin || !hp_pin))
Takashi Iwai0db75792013-01-23 13:57:20 +01004558 return;
Chih-Chung Changbde7bc62013-08-05 16:38:42 +08004559 snd_hda_jack_set_gating_jack(codec, mic_pin, hp_pin);
Takashi Iwai0db75792013-01-23 13:57:20 +01004560 }
Dylan Reid08a978d2012-11-18 22:56:40 -08004561}
David Henningsson693b6132012-06-22 19:12:10 +02004562
David Henningsson3e0d6112013-04-22 14:30:14 +02004563static void alc269_fixup_limit_int_mic_boost(struct hda_codec *codec,
4564 const struct hda_fixup *fix,
4565 int action)
4566{
4567 struct alc_spec *spec = codec->spec;
4568 struct auto_pin_cfg *cfg = &spec->gen.autocfg;
4569 int i;
4570
4571 /* The mic boosts on level 2 and 3 are too noisy
4572 on the internal mic input.
4573 Therefore limit the boost to 0 or 1. */
4574
4575 if (action != HDA_FIXUP_ACT_PROBE)
4576 return;
4577
4578 for (i = 0; i < cfg->num_inputs; i++) {
4579 hda_nid_t nid = cfg->inputs[i].pin;
4580 unsigned int defcfg;
4581 if (cfg->inputs[i].type != AUTO_PIN_MIC)
4582 continue;
4583 defcfg = snd_hda_codec_get_pincfg(codec, nid);
4584 if (snd_hda_get_input_pin_attr(defcfg) != INPUT_PIN_ATTR_INT)
4585 continue;
4586
4587 snd_hda_override_amp_caps(codec, nid, HDA_INPUT,
4588 (0x00 << AC_AMPCAP_OFFSET_SHIFT) |
4589 (0x01 << AC_AMPCAP_NUM_STEPS_SHIFT) |
4590 (0x2f << AC_AMPCAP_STEP_SIZE_SHIFT) |
4591 (0 << AC_AMPCAP_MUTE_SHIFT));
4592 }
4593}
4594
Kailang Yangcd217a62013-08-22 10:15:24 +02004595static void alc283_hp_automute_hook(struct hda_codec *codec,
Takashi Iwai1a4f69d2014-09-11 15:22:46 +02004596 struct hda_jack_callback *jack)
Kailang Yangcd217a62013-08-22 10:15:24 +02004597{
4598 struct alc_spec *spec = codec->spec;
4599 int vref;
4600
4601 msleep(200);
4602 snd_hda_gen_hp_automute(codec, jack);
4603
4604 vref = spec->gen.hp_jack_present ? PIN_VREF80 : 0;
4605
4606 msleep(600);
4607 snd_hda_codec_write(codec, 0x19, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
4608 vref);
4609}
4610
Kailang Yangcd217a62013-08-22 10:15:24 +02004611static void alc283_fixup_chromebook(struct hda_codec *codec,
4612 const struct hda_fixup *fix, int action)
4613{
4614 struct alc_spec *spec = codec->spec;
Kailang Yangcd217a62013-08-22 10:15:24 +02004615
4616 switch (action) {
4617 case HDA_FIXUP_ACT_PRE_PROBE:
Kailang Yang0202e992013-12-02 15:20:15 +08004618 snd_hda_override_wcaps(codec, 0x03, 0);
Takashi Iwaid2e92702013-10-30 07:50:53 +01004619 /* Disable AA-loopback as it causes white noise */
4620 spec->gen.mixer_nid = 0;
Kailang Yang38070212013-11-01 15:57:35 +08004621 break;
4622 case HDA_FIXUP_ACT_INIT:
Kailang Yangde9481c2014-04-07 16:41:52 +08004623 /* MIC2-VREF control */
4624 /* Set to manual mode */
Takashi Iwai98b24882014-08-18 13:47:50 +02004625 alc_update_coef_idx(codec, 0x06, 0x000c, 0);
Kailang Yang0202e992013-12-02 15:20:15 +08004626 /* Enable Line1 input control by verb */
Takashi Iwai98b24882014-08-18 13:47:50 +02004627 alc_update_coef_idx(codec, 0x1a, 0, 1 << 4);
Kailang Yang0202e992013-12-02 15:20:15 +08004628 break;
4629 }
4630}
4631
4632static void alc283_fixup_sense_combo_jack(struct hda_codec *codec,
4633 const struct hda_fixup *fix, int action)
4634{
4635 struct alc_spec *spec = codec->spec;
Kailang Yang0202e992013-12-02 15:20:15 +08004636
4637 switch (action) {
4638 case HDA_FIXUP_ACT_PRE_PROBE:
Kailang Yangcd217a62013-08-22 10:15:24 +02004639 spec->gen.hp_automute_hook = alc283_hp_automute_hook;
4640 break;
4641 case HDA_FIXUP_ACT_INIT:
4642 /* MIC2-VREF control */
4643 /* Set to manual mode */
Takashi Iwai98b24882014-08-18 13:47:50 +02004644 alc_update_coef_idx(codec, 0x06, 0x000c, 0);
Kailang Yangcd217a62013-08-22 10:15:24 +02004645 break;
4646 }
4647}
4648
Takashi Iwai7bba2152013-09-06 15:45:38 +02004649/* mute tablet speaker pin (0x14) via dock plugging in addition */
4650static void asus_tx300_automute(struct hda_codec *codec)
4651{
4652 struct alc_spec *spec = codec->spec;
4653 snd_hda_gen_update_outputs(codec);
4654 if (snd_hda_jack_detect(codec, 0x1b))
4655 spec->gen.mute_bits |= (1ULL << 0x14);
4656}
4657
4658static void alc282_fixup_asus_tx300(struct hda_codec *codec,
4659 const struct hda_fixup *fix, int action)
4660{
4661 struct alc_spec *spec = codec->spec;
4662 /* TX300 needs to set up GPIO2 for the speaker amp */
4663 static const struct hda_verb gpio2_verbs[] = {
4664 { 0x01, AC_VERB_SET_GPIO_MASK, 0x04 },
4665 { 0x01, AC_VERB_SET_GPIO_DIRECTION, 0x04 },
4666 { 0x01, AC_VERB_SET_GPIO_DATA, 0x04 },
4667 {}
4668 };
4669 static const struct hda_pintbl dock_pins[] = {
4670 { 0x1b, 0x21114000 }, /* dock speaker pin */
4671 {}
4672 };
4673 struct snd_kcontrol *kctl;
4674
4675 switch (action) {
4676 case HDA_FIXUP_ACT_PRE_PROBE:
4677 snd_hda_add_verbs(codec, gpio2_verbs);
4678 snd_hda_apply_pincfgs(codec, dock_pins);
4679 spec->gen.auto_mute_via_amp = 1;
4680 spec->gen.automute_hook = asus_tx300_automute;
4681 snd_hda_jack_detect_enable_callback(codec, 0x1b,
Takashi Iwai7bba2152013-09-06 15:45:38 +02004682 snd_hda_gen_hp_automute);
4683 break;
4684 case HDA_FIXUP_ACT_BUILD:
4685 /* this is a bit tricky; give more sane names for the main
4686 * (tablet) speaker and the dock speaker, respectively
4687 */
4688 kctl = snd_hda_find_mixer_ctl(codec, "Speaker Playback Switch");
4689 if (kctl)
4690 strcpy(kctl->id.name, "Dock Speaker Playback Switch");
4691 kctl = snd_hda_find_mixer_ctl(codec, "Bass Speaker Playback Switch");
4692 if (kctl)
4693 strcpy(kctl->id.name, "Speaker Playback Switch");
4694 break;
4695 }
4696}
4697
David Henningsson338cae52013-10-07 10:39:59 +02004698static void alc290_fixup_mono_speakers(struct hda_codec *codec,
4699 const struct hda_fixup *fix, int action)
4700{
David Henningsson0f4881d2013-12-20 16:08:13 +01004701 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
4702 /* DAC node 0x03 is giving mono output. We therefore want to
4703 make sure 0x14 (front speaker) and 0x15 (headphones) use the
4704 stereo DAC, while leaving 0x17 (bass speaker) for node 0x03. */
4705 hda_nid_t conn1[2] = { 0x0c };
4706 snd_hda_override_conn_list(codec, 0x14, 1, conn1);
4707 snd_hda_override_conn_list(codec, 0x15, 1, conn1);
4708 }
David Henningsson338cae52013-10-07 10:39:59 +02004709}
4710
Hui Wangdd9aa332016-08-01 10:20:32 +08004711static void alc298_fixup_speaker_volume(struct hda_codec *codec,
4712 const struct hda_fixup *fix, int action)
4713{
4714 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
4715 /* The speaker is routed to the Node 0x06 by a mistake, as a result
4716 we can't adjust the speaker's volume since this node does not has
4717 Amp-out capability. we change the speaker's route to:
4718 Node 0x02 (Audio Output) -> Node 0x0c (Audio Mixer) -> Node 0x17 (
4719 Pin Complex), since Node 0x02 has Amp-out caps, we can adjust
4720 speaker's volume now. */
4721
4722 hda_nid_t conn1[1] = { 0x0c };
4723 snd_hda_override_conn_list(codec, 0x17, 1, conn1);
4724 }
4725}
4726
Keith Packard98973f22015-07-15 12:14:39 -07004727/* Hook to update amp GPIO4 for automute */
4728static void alc280_hp_gpio4_automute_hook(struct hda_codec *codec,
4729 struct hda_jack_callback *jack)
4730{
4731 struct alc_spec *spec = codec->spec;
4732
4733 snd_hda_gen_hp_automute(codec, jack);
4734 /* mute_led_polarity is set to 0, so we pass inverted value here */
4735 alc_update_gpio_led(codec, 0x10, !spec->gen.hp_jack_present);
4736}
4737
4738/* Manage GPIOs for HP EliteBook Folio 9480m.
4739 *
4740 * GPIO4 is the headphone amplifier power control
4741 * GPIO3 is the audio output mute indicator LED
4742 */
4743
4744static void alc280_fixup_hp_9480m(struct hda_codec *codec,
4745 const struct hda_fixup *fix,
4746 int action)
4747{
4748 struct alc_spec *spec = codec->spec;
4749 static const struct hda_verb gpio_init[] = {
4750 { 0x01, AC_VERB_SET_GPIO_MASK, 0x18 },
4751 { 0x01, AC_VERB_SET_GPIO_DIRECTION, 0x18 },
4752 {}
4753 };
4754
4755 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
4756 /* Set the hooks to turn the headphone amp on/off
4757 * as needed
4758 */
4759 spec->gen.vmaster_mute.hook = alc_fixup_gpio_mute_hook;
4760 spec->gen.hp_automute_hook = alc280_hp_gpio4_automute_hook;
4761
4762 /* The GPIOs are currently off */
4763 spec->gpio_led = 0;
4764
4765 /* GPIO3 is connected to the output mute LED,
4766 * high is on, low is off
4767 */
4768 spec->mute_led_polarity = 0;
4769 spec->gpio_mute_led_mask = 0x08;
4770
4771 /* Initialize GPIO configuration */
4772 snd_hda_add_verbs(codec, gpio_init);
4773 }
4774}
4775
Takashi Iwaib317b032014-01-08 11:44:21 +01004776/* for hda_fixup_thinkpad_acpi() */
4777#include "thinkpad_helper.c"
David Henningssonb67ae3f2013-11-05 13:11:37 +01004778
Hui Wang00ef9942014-07-31 11:52:38 +08004779/* for dell wmi mic mute led */
4780#include "dell_wmi_helper.c"
4781
Takashi Iwai1d045db2011-07-07 18:23:21 +02004782enum {
4783 ALC269_FIXUP_SONY_VAIO,
4784 ALC275_FIXUP_SONY_VAIO_GPIO2,
4785 ALC269_FIXUP_DELL_M101Z,
4786 ALC269_FIXUP_SKU_IGNORE,
4787 ALC269_FIXUP_ASUS_G73JW,
4788 ALC269_FIXUP_LENOVO_EAPD,
4789 ALC275_FIXUP_SONY_HWEQ,
Takashi Iwaie9bd7d52014-05-21 11:06:49 +02004790 ALC275_FIXUP_SONY_DISABLE_AAMIX,
Takashi Iwai1d045db2011-07-07 18:23:21 +02004791 ALC271_FIXUP_DMIC,
Takashi Iwai017f2a12011-07-09 14:42:25 +02004792 ALC269_FIXUP_PCM_44K,
Takashi Iwaiadabb3e2011-08-03 07:48:37 +02004793 ALC269_FIXUP_STEREO_DMIC,
David Henningsson7c478f02013-10-11 10:18:46 +02004794 ALC269_FIXUP_HEADSET_MIC,
Takashi Iwai24519912011-08-16 15:08:49 +02004795 ALC269_FIXUP_QUANTA_MUTE,
4796 ALC269_FIXUP_LIFEBOOK,
David Henningsson2041d562014-06-13 11:15:44 +02004797 ALC269_FIXUP_LIFEBOOK_EXTMIC,
Takashi Iwaicc7016a2015-04-08 20:47:55 +02004798 ALC269_FIXUP_LIFEBOOK_HP_PIN,
Takashi Iwai4df3fd12015-06-29 08:38:02 +02004799 ALC269_FIXUP_LIFEBOOK_NO_HP_TO_LINEOUT,
Takashi Iwaia4297b52011-08-23 18:40:12 +02004800 ALC269_FIXUP_AMIC,
4801 ALC269_FIXUP_DMIC,
4802 ALC269VB_FIXUP_AMIC,
4803 ALC269VB_FIXUP_DMIC,
Takashi Iwai08fb0d02013-01-10 17:33:58 +01004804 ALC269_FIXUP_HP_MUTE_LED,
David Henningssond06ac142013-02-18 11:41:55 +01004805 ALC269_FIXUP_HP_MUTE_LED_MIC1,
Takashi Iwai08fb0d02013-01-10 17:33:58 +01004806 ALC269_FIXUP_HP_MUTE_LED_MIC2,
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01004807 ALC269_FIXUP_HP_GPIO_LED,
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08004808 ALC269_FIXUP_HP_GPIO_MIC1_LED,
4809 ALC269_FIXUP_HP_LINE1_MIC1_LED,
David Henningsson693b6132012-06-22 19:12:10 +02004810 ALC269_FIXUP_INV_DMIC,
David Henningsson108cc102012-07-20 10:37:25 +02004811 ALC269_FIXUP_LENOVO_DOCK,
Takashi Iwai9b745ab2014-03-07 08:37:19 +01004812 ALC269_FIXUP_NO_SHUTUP,
David Henningsson88cfcf82013-10-11 10:18:45 +02004813 ALC286_FIXUP_SONY_MIC_NO_PRESENCE,
David Henningsson108cc102012-07-20 10:37:25 +02004814 ALC269_FIXUP_PINCFG_NO_HP_TO_LINEOUT,
David Henningsson73bdd592013-04-15 15:44:14 +02004815 ALC269_FIXUP_DELL1_MIC_NO_PRESENCE,
4816 ALC269_FIXUP_DELL2_MIC_NO_PRESENCE,
David Henningsson338cae52013-10-07 10:39:59 +02004817 ALC269_FIXUP_DELL3_MIC_NO_PRESENCE,
David Henningsson73bdd592013-04-15 15:44:14 +02004818 ALC269_FIXUP_HEADSET_MODE,
4819 ALC269_FIXUP_HEADSET_MODE_NO_HP_MIC,
Takashi Iwai78197172015-06-27 10:21:13 +02004820 ALC269_FIXUP_ASPIRE_HEADSET_MIC,
David Henningssond240d1d2013-04-15 12:50:02 +02004821 ALC269_FIXUP_ASUS_X101_FUNC,
4822 ALC269_FIXUP_ASUS_X101_VERB,
4823 ALC269_FIXUP_ASUS_X101,
Dylan Reid08a978d2012-11-18 22:56:40 -08004824 ALC271_FIXUP_AMIC_MIC2,
4825 ALC271_FIXUP_HP_GATE_MIC_JACK,
Oleksij Rempelb1e89722013-12-04 20:50:53 +01004826 ALC271_FIXUP_HP_GATE_MIC_JACK_E1_572,
Dylan Reid42397002013-04-05 14:58:22 -07004827 ALC269_FIXUP_ACER_AC700,
David Henningsson3e0d6112013-04-22 14:30:14 +02004828 ALC269_FIXUP_LIMIT_INT_MIC_BOOST,
Oleksij Rempel2cede302013-11-27 17:12:03 +01004829 ALC269VB_FIXUP_ASUS_ZENBOOK,
Takashi Iwai23870832013-11-29 14:13:12 +01004830 ALC269VB_FIXUP_ASUS_ZENBOOK_UX31A,
David Henningsson8e35cd42013-11-06 11:20:01 +01004831 ALC269_FIXUP_LIMIT_INT_MIC_BOOST_MUTE_LED,
Anisse Astier02b504d2013-06-03 11:53:10 +02004832 ALC269VB_FIXUP_ORDISSIMO_EVE2,
Kailang Yangcd217a62013-08-22 10:15:24 +02004833 ALC283_FIXUP_CHROME_BOOK,
Kailang Yang0202e992013-12-02 15:20:15 +08004834 ALC283_FIXUP_SENSE_COMBO_JACK,
Takashi Iwai7bba2152013-09-06 15:45:38 +02004835 ALC282_FIXUP_ASUS_TX300,
Kailang Yang1bb3e062013-09-27 13:10:25 +02004836 ALC283_FIXUP_INT_MIC,
David Henningsson338cae52013-10-07 10:39:59 +02004837 ALC290_FIXUP_MONO_SPEAKERS,
David Henningsson0f4881d2013-12-20 16:08:13 +01004838 ALC290_FIXUP_MONO_SPEAKERS_HSJACK,
4839 ALC290_FIXUP_SUBWOOFER,
4840 ALC290_FIXUP_SUBWOOFER_HSJACK,
David Henningssonb67ae3f2013-11-05 13:11:37 +01004841 ALC269_FIXUP_THINKPAD_ACPI,
David Henningsson56f27012016-01-11 09:33:14 +01004842 ALC269_FIXUP_DMIC_THINKPAD_ACPI,
Kailang Yang9a22a8f2013-11-08 15:54:49 +08004843 ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Kailang Yang31278992014-03-03 15:27:22 +08004844 ALC255_FIXUP_DELL2_MIC_NO_PRESENCE,
Kailang Yang9a22a8f2013-11-08 15:54:49 +08004845 ALC255_FIXUP_HEADSET_MODE,
Kailang Yang31278992014-03-03 15:27:22 +08004846 ALC255_FIXUP_HEADSET_MODE_NO_HP_MIC,
Kailang Yanga22aa262014-04-23 17:34:28 +08004847 ALC293_FIXUP_DELL1_MIC_NO_PRESENCE,
Takashi Iwai1c37c222014-05-06 17:34:42 +02004848 ALC292_FIXUP_TPT440_DOCK,
Takashi Iwai9a811232015-12-09 15:17:43 +01004849 ALC292_FIXUP_TPT440,
Anisse Astierabaa22742016-08-24 09:14:13 +02004850 ALC283_FIXUP_HEADSET_MIC,
Hui Wang00ef9942014-07-31 11:52:38 +08004851 ALC255_FIXUP_DELL_WMI_MIC_MUTE_LED,
Takashi Iwai1a22e772014-08-27 08:19:05 +02004852 ALC282_FIXUP_ASPIRE_V5_PINS,
David Henningsson7a5255f2014-10-30 08:26:01 +01004853 ALC280_FIXUP_HP_GPIO4,
Kailang Yangeaa8e5e2014-11-21 15:49:11 +08004854 ALC286_FIXUP_HP_GPIO_LED,
David Henningsson33f4acd2015-01-07 15:50:13 +01004855 ALC280_FIXUP_HP_GPIO2_MIC_HOTKEY,
TienFu Chenb4b33f92015-01-20 15:06:21 +01004856 ALC280_FIXUP_HP_DOCK_PINS,
Jaroslav Kysela52c33232017-03-09 13:29:13 +01004857 ALC269_FIXUP_HP_DOCK_GPIO_MIC1_LED,
Keith Packard98973f22015-07-15 12:14:39 -07004858 ALC280_FIXUP_HP_9480M,
Kailang Yange1e62b92015-04-08 16:01:22 +08004859 ALC288_FIXUP_DELL_HEADSET_MODE,
4860 ALC288_FIXUP_DELL1_MIC_NO_PRESENCE,
4861 ALC288_FIXUP_DELL_XPS_13_GPIO6,
Hui Wang831bfdf2015-06-26 12:35:17 +08004862 ALC288_FIXUP_DELL_XPS_13,
4863 ALC288_FIXUP_DISABLE_AAMIX,
Takashi Iwai8b99aba2015-06-15 11:59:32 +02004864 ALC292_FIXUP_DELL_E7X,
4865 ALC292_FIXUP_DISABLE_AAMIX,
David Henningssonc04017e2015-12-15 14:44:03 +01004866 ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK,
Kailang Yang977e6272015-05-18 15:31:20 +08004867 ALC298_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wang8aabccd2017-03-31 10:31:40 +08004868 ALC298_FIXUP_DELL_AIO_MIC_NO_PRESENCE,
Kailang Yang6ed11312015-10-26 15:37:39 +08004869 ALC275_FIXUP_DELL_XPS,
Hui Wang8c697292015-11-24 11:08:18 +08004870 ALC256_FIXUP_DELL_XPS_13_HEADPHONE_NOISE,
Hui Wang23adc192015-12-08 12:27:18 +08004871 ALC293_FIXUP_LENOVO_SPK_NOISE,
Kailang3694cb22015-12-28 11:35:24 +08004872 ALC233_FIXUP_LENOVO_LINE2_MIC_HOTKEY,
Kai-Heng Feng3b43b712016-02-25 15:19:38 +08004873 ALC255_FIXUP_DELL_SPK_NOISE,
David Henningsson2ae95572016-02-25 09:37:05 +01004874 ALC225_FIXUP_DELL1_MIC_NO_PRESENCE,
Takashi Iwaif8839822016-02-25 14:31:59 +01004875 ALC280_FIXUP_HP_HEADSET_MIC,
Hui Wange549d192016-04-01 11:00:15 +08004876 ALC221_FIXUP_HP_FRONT_MIC,
Sven Eckelmannc636b952016-04-11 16:55:26 +02004877 ALC292_FIXUP_TPT460,
Hui Wangdd9aa332016-08-01 10:20:32 +08004878 ALC298_FIXUP_SPK_VOLUME,
Kai-Heng Fengfd06c772016-08-30 15:36:34 +08004879 ALC256_FIXUP_DELL_INSPIRON_7559_SUBWOOFER,
Takashi Iwai1d045db2011-07-07 18:23:21 +02004880};
4881
Takashi Iwai1727a772013-01-10 09:52:52 +01004882static const struct hda_fixup alc269_fixups[] = {
Takashi Iwai1d045db2011-07-07 18:23:21 +02004883 [ALC269_FIXUP_SONY_VAIO] = {
Takashi Iwaifd108212013-01-10 10:18:14 +01004884 .type = HDA_FIXUP_PINCTLS,
4885 .v.pins = (const struct hda_pintbl[]) {
4886 {0x19, PIN_VREFGRD},
Takashi Iwai1d045db2011-07-07 18:23:21 +02004887 {}
4888 }
4889 },
4890 [ALC275_FIXUP_SONY_VAIO_GPIO2] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01004891 .type = HDA_FIXUP_VERBS,
Takashi Iwai1d045db2011-07-07 18:23:21 +02004892 .v.verbs = (const struct hda_verb[]) {
4893 {0x01, AC_VERB_SET_GPIO_MASK, 0x04},
4894 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x04},
4895 {0x01, AC_VERB_SET_GPIO_DATA, 0x00},
4896 { }
4897 },
4898 .chained = true,
4899 .chain_id = ALC269_FIXUP_SONY_VAIO
4900 },
4901 [ALC269_FIXUP_DELL_M101Z] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01004902 .type = HDA_FIXUP_VERBS,
Takashi Iwai1d045db2011-07-07 18:23:21 +02004903 .v.verbs = (const struct hda_verb[]) {
4904 /* Enables internal speaker */
4905 {0x20, AC_VERB_SET_COEF_INDEX, 13},
4906 {0x20, AC_VERB_SET_PROC_COEF, 0x4040},
4907 {}
4908 }
4909 },
4910 [ALC269_FIXUP_SKU_IGNORE] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01004911 .type = HDA_FIXUP_FUNC,
Takashi Iwai23d30f22012-05-07 17:17:32 +02004912 .v.func = alc_fixup_sku_ignore,
Takashi Iwai1d045db2011-07-07 18:23:21 +02004913 },
4914 [ALC269_FIXUP_ASUS_G73JW] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01004915 .type = HDA_FIXUP_PINS,
4916 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02004917 { 0x17, 0x99130111 }, /* subwoofer */
4918 { }
4919 }
4920 },
4921 [ALC269_FIXUP_LENOVO_EAPD] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01004922 .type = HDA_FIXUP_VERBS,
Takashi Iwai1d045db2011-07-07 18:23:21 +02004923 .v.verbs = (const struct hda_verb[]) {
4924 {0x14, AC_VERB_SET_EAPD_BTLENABLE, 0},
4925 {}
4926 }
4927 },
4928 [ALC275_FIXUP_SONY_HWEQ] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01004929 .type = HDA_FIXUP_FUNC,
Takashi Iwai1d045db2011-07-07 18:23:21 +02004930 .v.func = alc269_fixup_hweq,
4931 .chained = true,
4932 .chain_id = ALC275_FIXUP_SONY_VAIO_GPIO2
4933 },
Takashi Iwaie9bd7d52014-05-21 11:06:49 +02004934 [ALC275_FIXUP_SONY_DISABLE_AAMIX] = {
4935 .type = HDA_FIXUP_FUNC,
4936 .v.func = alc_fixup_disable_aamix,
4937 .chained = true,
4938 .chain_id = ALC269_FIXUP_SONY_VAIO
4939 },
Takashi Iwai1d045db2011-07-07 18:23:21 +02004940 [ALC271_FIXUP_DMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01004941 .type = HDA_FIXUP_FUNC,
Takashi Iwai1d045db2011-07-07 18:23:21 +02004942 .v.func = alc271_fixup_dmic,
4943 },
Takashi Iwai017f2a12011-07-09 14:42:25 +02004944 [ALC269_FIXUP_PCM_44K] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01004945 .type = HDA_FIXUP_FUNC,
Takashi Iwai017f2a12011-07-09 14:42:25 +02004946 .v.func = alc269_fixup_pcm_44k,
David Henningsson012e7eb2012-08-08 08:43:37 +02004947 .chained = true,
4948 .chain_id = ALC269_FIXUP_QUANTA_MUTE
Takashi Iwai017f2a12011-07-09 14:42:25 +02004949 },
Takashi Iwaiadabb3e2011-08-03 07:48:37 +02004950 [ALC269_FIXUP_STEREO_DMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01004951 .type = HDA_FIXUP_FUNC,
Takashi Iwaiadabb3e2011-08-03 07:48:37 +02004952 .v.func = alc269_fixup_stereo_dmic,
4953 },
David Henningsson7c478f02013-10-11 10:18:46 +02004954 [ALC269_FIXUP_HEADSET_MIC] = {
4955 .type = HDA_FIXUP_FUNC,
4956 .v.func = alc269_fixup_headset_mic,
4957 },
Takashi Iwai24519912011-08-16 15:08:49 +02004958 [ALC269_FIXUP_QUANTA_MUTE] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01004959 .type = HDA_FIXUP_FUNC,
Takashi Iwai24519912011-08-16 15:08:49 +02004960 .v.func = alc269_fixup_quanta_mute,
4961 },
4962 [ALC269_FIXUP_LIFEBOOK] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01004963 .type = HDA_FIXUP_PINS,
4964 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai24519912011-08-16 15:08:49 +02004965 { 0x1a, 0x2101103f }, /* dock line-out */
4966 { 0x1b, 0x23a11040 }, /* dock mic-in */
4967 { }
4968 },
4969 .chained = true,
4970 .chain_id = ALC269_FIXUP_QUANTA_MUTE
4971 },
David Henningsson2041d562014-06-13 11:15:44 +02004972 [ALC269_FIXUP_LIFEBOOK_EXTMIC] = {
4973 .type = HDA_FIXUP_PINS,
4974 .v.pins = (const struct hda_pintbl[]) {
4975 { 0x19, 0x01a1903c }, /* headset mic, with jack detect */
4976 { }
4977 },
4978 },
Takashi Iwaicc7016a2015-04-08 20:47:55 +02004979 [ALC269_FIXUP_LIFEBOOK_HP_PIN] = {
4980 .type = HDA_FIXUP_PINS,
4981 .v.pins = (const struct hda_pintbl[]) {
4982 { 0x21, 0x0221102f }, /* HP out */
4983 { }
4984 },
4985 },
Takashi Iwai4df3fd12015-06-29 08:38:02 +02004986 [ALC269_FIXUP_LIFEBOOK_NO_HP_TO_LINEOUT] = {
4987 .type = HDA_FIXUP_FUNC,
4988 .v.func = alc269_fixup_pincfg_no_hp_to_lineout,
4989 },
Takashi Iwaia4297b52011-08-23 18:40:12 +02004990 [ALC269_FIXUP_AMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01004991 .type = HDA_FIXUP_PINS,
4992 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaia4297b52011-08-23 18:40:12 +02004993 { 0x14, 0x99130110 }, /* speaker */
4994 { 0x15, 0x0121401f }, /* HP out */
4995 { 0x18, 0x01a19c20 }, /* mic */
4996 { 0x19, 0x99a3092f }, /* int-mic */
4997 { }
4998 },
4999 },
5000 [ALC269_FIXUP_DMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005001 .type = HDA_FIXUP_PINS,
5002 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaia4297b52011-08-23 18:40:12 +02005003 { 0x12, 0x99a3092f }, /* int-mic */
5004 { 0x14, 0x99130110 }, /* speaker */
5005 { 0x15, 0x0121401f }, /* HP out */
5006 { 0x18, 0x01a19c20 }, /* mic */
5007 { }
5008 },
5009 },
5010 [ALC269VB_FIXUP_AMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005011 .type = HDA_FIXUP_PINS,
5012 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaia4297b52011-08-23 18:40:12 +02005013 { 0x14, 0x99130110 }, /* speaker */
5014 { 0x18, 0x01a19c20 }, /* mic */
5015 { 0x19, 0x99a3092f }, /* int-mic */
5016 { 0x21, 0x0121401f }, /* HP out */
5017 { }
5018 },
5019 },
David Henningsson2267ea92012-01-03 08:45:56 +01005020 [ALC269VB_FIXUP_DMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005021 .type = HDA_FIXUP_PINS,
5022 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaia4297b52011-08-23 18:40:12 +02005023 { 0x12, 0x99a3092f }, /* int-mic */
5024 { 0x14, 0x99130110 }, /* speaker */
5025 { 0x18, 0x01a19c20 }, /* mic */
5026 { 0x21, 0x0121401f }, /* HP out */
5027 { }
5028 },
5029 },
Takashi Iwai08fb0d02013-01-10 17:33:58 +01005030 [ALC269_FIXUP_HP_MUTE_LED] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005031 .type = HDA_FIXUP_FUNC,
Takashi Iwai08fb0d02013-01-10 17:33:58 +01005032 .v.func = alc269_fixup_hp_mute_led,
David Henningsson6d3cd5d2013-01-07 12:03:47 +01005033 },
David Henningssond06ac142013-02-18 11:41:55 +01005034 [ALC269_FIXUP_HP_MUTE_LED_MIC1] = {
5035 .type = HDA_FIXUP_FUNC,
5036 .v.func = alc269_fixup_hp_mute_led_mic1,
5037 },
Takashi Iwai08fb0d02013-01-10 17:33:58 +01005038 [ALC269_FIXUP_HP_MUTE_LED_MIC2] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005039 .type = HDA_FIXUP_FUNC,
Takashi Iwai08fb0d02013-01-10 17:33:58 +01005040 .v.func = alc269_fixup_hp_mute_led_mic2,
Takashi Iwai420b0fe2012-03-12 12:35:27 +01005041 },
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01005042 [ALC269_FIXUP_HP_GPIO_LED] = {
5043 .type = HDA_FIXUP_FUNC,
5044 .v.func = alc269_fixup_hp_gpio_led,
5045 },
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08005046 [ALC269_FIXUP_HP_GPIO_MIC1_LED] = {
5047 .type = HDA_FIXUP_FUNC,
5048 .v.func = alc269_fixup_hp_gpio_mic1_led,
5049 },
5050 [ALC269_FIXUP_HP_LINE1_MIC1_LED] = {
5051 .type = HDA_FIXUP_FUNC,
5052 .v.func = alc269_fixup_hp_line1_mic1_led,
5053 },
David Henningsson693b6132012-06-22 19:12:10 +02005054 [ALC269_FIXUP_INV_DMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005055 .type = HDA_FIXUP_FUNC,
David Henningsson9d36a7d2014-10-07 10:18:42 +02005056 .v.func = alc_fixup_inv_dmic,
David Henningsson693b6132012-06-22 19:12:10 +02005057 },
Takashi Iwai9b745ab2014-03-07 08:37:19 +01005058 [ALC269_FIXUP_NO_SHUTUP] = {
5059 .type = HDA_FIXUP_FUNC,
5060 .v.func = alc_fixup_no_shutup,
5061 },
David Henningsson108cc102012-07-20 10:37:25 +02005062 [ALC269_FIXUP_LENOVO_DOCK] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005063 .type = HDA_FIXUP_PINS,
5064 .v.pins = (const struct hda_pintbl[]) {
David Henningsson108cc102012-07-20 10:37:25 +02005065 { 0x19, 0x23a11040 }, /* dock mic */
5066 { 0x1b, 0x2121103f }, /* dock headphone */
5067 { }
5068 },
5069 .chained = true,
5070 .chain_id = ALC269_FIXUP_PINCFG_NO_HP_TO_LINEOUT
5071 },
5072 [ALC269_FIXUP_PINCFG_NO_HP_TO_LINEOUT] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005073 .type = HDA_FIXUP_FUNC,
David Henningsson108cc102012-07-20 10:37:25 +02005074 .v.func = alc269_fixup_pincfg_no_hp_to_lineout,
David Henningsson52129002013-11-19 10:25:53 +01005075 .chained = true,
5076 .chain_id = ALC269_FIXUP_THINKPAD_ACPI,
David Henningsson108cc102012-07-20 10:37:25 +02005077 },
David Henningsson73bdd592013-04-15 15:44:14 +02005078 [ALC269_FIXUP_DELL1_MIC_NO_PRESENCE] = {
5079 .type = HDA_FIXUP_PINS,
5080 .v.pins = (const struct hda_pintbl[]) {
5081 { 0x19, 0x01a1913c }, /* use as headset mic, without its own jack detect */
5082 { 0x1a, 0x01a1913d }, /* use as headphone mic, without its own jack detect */
5083 { }
5084 },
5085 .chained = true,
5086 .chain_id = ALC269_FIXUP_HEADSET_MODE
5087 },
5088 [ALC269_FIXUP_DELL2_MIC_NO_PRESENCE] = {
5089 .type = HDA_FIXUP_PINS,
5090 .v.pins = (const struct hda_pintbl[]) {
5091 { 0x16, 0x21014020 }, /* dock line out */
5092 { 0x19, 0x21a19030 }, /* dock mic */
5093 { 0x1a, 0x01a1913c }, /* use as headset mic, without its own jack detect */
5094 { }
5095 },
5096 .chained = true,
5097 .chain_id = ALC269_FIXUP_HEADSET_MODE_NO_HP_MIC
5098 },
David Henningsson338cae52013-10-07 10:39:59 +02005099 [ALC269_FIXUP_DELL3_MIC_NO_PRESENCE] = {
5100 .type = HDA_FIXUP_PINS,
5101 .v.pins = (const struct hda_pintbl[]) {
5102 { 0x1a, 0x01a1913c }, /* use as headset mic, without its own jack detect */
5103 { }
5104 },
5105 .chained = true,
5106 .chain_id = ALC269_FIXUP_HEADSET_MODE_NO_HP_MIC
5107 },
David Henningsson73bdd592013-04-15 15:44:14 +02005108 [ALC269_FIXUP_HEADSET_MODE] = {
5109 .type = HDA_FIXUP_FUNC,
5110 .v.func = alc_fixup_headset_mode,
Hui Wang6676f302014-11-18 17:57:41 +08005111 .chained = true,
5112 .chain_id = ALC255_FIXUP_DELL_WMI_MIC_MUTE_LED
David Henningsson73bdd592013-04-15 15:44:14 +02005113 },
5114 [ALC269_FIXUP_HEADSET_MODE_NO_HP_MIC] = {
5115 .type = HDA_FIXUP_FUNC,
5116 .v.func = alc_fixup_headset_mode_no_hp_mic,
5117 },
Takashi Iwai78197172015-06-27 10:21:13 +02005118 [ALC269_FIXUP_ASPIRE_HEADSET_MIC] = {
5119 .type = HDA_FIXUP_PINS,
5120 .v.pins = (const struct hda_pintbl[]) {
5121 { 0x19, 0x01a1913c }, /* headset mic w/o jack detect */
5122 { }
5123 },
5124 .chained = true,
5125 .chain_id = ALC269_FIXUP_HEADSET_MODE,
5126 },
David Henningsson88cfcf82013-10-11 10:18:45 +02005127 [ALC286_FIXUP_SONY_MIC_NO_PRESENCE] = {
5128 .type = HDA_FIXUP_PINS,
5129 .v.pins = (const struct hda_pintbl[]) {
5130 { 0x18, 0x01a1913c }, /* use as headset mic, without its own jack detect */
5131 { }
5132 },
David Henningssonfbc78ad2013-10-11 13:46:04 +02005133 .chained = true,
5134 .chain_id = ALC269_FIXUP_HEADSET_MIC
David Henningsson88cfcf82013-10-11 10:18:45 +02005135 },
David Henningssond240d1d2013-04-15 12:50:02 +02005136 [ALC269_FIXUP_ASUS_X101_FUNC] = {
5137 .type = HDA_FIXUP_FUNC,
5138 .v.func = alc269_fixup_x101_headset_mic,
5139 },
5140 [ALC269_FIXUP_ASUS_X101_VERB] = {
5141 .type = HDA_FIXUP_VERBS,
5142 .v.verbs = (const struct hda_verb[]) {
5143 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
5144 {0x20, AC_VERB_SET_COEF_INDEX, 0x08},
5145 {0x20, AC_VERB_SET_PROC_COEF, 0x0310},
5146 { }
5147 },
5148 .chained = true,
5149 .chain_id = ALC269_FIXUP_ASUS_X101_FUNC
5150 },
5151 [ALC269_FIXUP_ASUS_X101] = {
5152 .type = HDA_FIXUP_PINS,
5153 .v.pins = (const struct hda_pintbl[]) {
5154 { 0x18, 0x04a1182c }, /* Headset mic */
5155 { }
5156 },
5157 .chained = true,
5158 .chain_id = ALC269_FIXUP_ASUS_X101_VERB
5159 },
Dylan Reid08a978d2012-11-18 22:56:40 -08005160 [ALC271_FIXUP_AMIC_MIC2] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005161 .type = HDA_FIXUP_PINS,
5162 .v.pins = (const struct hda_pintbl[]) {
Dylan Reid08a978d2012-11-18 22:56:40 -08005163 { 0x14, 0x99130110 }, /* speaker */
5164 { 0x19, 0x01a19c20 }, /* mic */
5165 { 0x1b, 0x99a7012f }, /* int-mic */
5166 { 0x21, 0x0121401f }, /* HP out */
5167 { }
5168 },
5169 },
5170 [ALC271_FIXUP_HP_GATE_MIC_JACK] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005171 .type = HDA_FIXUP_FUNC,
Dylan Reid08a978d2012-11-18 22:56:40 -08005172 .v.func = alc271_hp_gate_mic_jack,
5173 .chained = true,
5174 .chain_id = ALC271_FIXUP_AMIC_MIC2,
5175 },
Oleksij Rempelb1e89722013-12-04 20:50:53 +01005176 [ALC271_FIXUP_HP_GATE_MIC_JACK_E1_572] = {
5177 .type = HDA_FIXUP_FUNC,
5178 .v.func = alc269_fixup_limit_int_mic_boost,
5179 .chained = true,
5180 .chain_id = ALC271_FIXUP_HP_GATE_MIC_JACK,
5181 },
Dylan Reid42397002013-04-05 14:58:22 -07005182 [ALC269_FIXUP_ACER_AC700] = {
5183 .type = HDA_FIXUP_PINS,
5184 .v.pins = (const struct hda_pintbl[]) {
5185 { 0x12, 0x99a3092f }, /* int-mic */
5186 { 0x14, 0x99130110 }, /* speaker */
5187 { 0x18, 0x03a11c20 }, /* mic */
5188 { 0x1e, 0x0346101e }, /* SPDIF1 */
5189 { 0x21, 0x0321101f }, /* HP out */
5190 { }
5191 },
5192 .chained = true,
5193 .chain_id = ALC271_FIXUP_DMIC,
5194 },
David Henningsson3e0d6112013-04-22 14:30:14 +02005195 [ALC269_FIXUP_LIMIT_INT_MIC_BOOST] = {
5196 .type = HDA_FIXUP_FUNC,
5197 .v.func = alc269_fixup_limit_int_mic_boost,
David Henningsson27937692013-11-18 11:51:47 +01005198 .chained = true,
5199 .chain_id = ALC269_FIXUP_THINKPAD_ACPI,
David Henningsson3e0d6112013-04-22 14:30:14 +02005200 },
Oleksij Rempel2cede302013-11-27 17:12:03 +01005201 [ALC269VB_FIXUP_ASUS_ZENBOOK] = {
5202 .type = HDA_FIXUP_FUNC,
5203 .v.func = alc269_fixup_limit_int_mic_boost,
5204 .chained = true,
5205 .chain_id = ALC269VB_FIXUP_DMIC,
5206 },
Takashi Iwai23870832013-11-29 14:13:12 +01005207 [ALC269VB_FIXUP_ASUS_ZENBOOK_UX31A] = {
5208 .type = HDA_FIXUP_VERBS,
5209 .v.verbs = (const struct hda_verb[]) {
5210 /* class-D output amp +5dB */
5211 { 0x20, AC_VERB_SET_COEF_INDEX, 0x12 },
5212 { 0x20, AC_VERB_SET_PROC_COEF, 0x2800 },
5213 {}
5214 },
5215 .chained = true,
5216 .chain_id = ALC269VB_FIXUP_ASUS_ZENBOOK,
5217 },
David Henningsson8e35cd42013-11-06 11:20:01 +01005218 [ALC269_FIXUP_LIMIT_INT_MIC_BOOST_MUTE_LED] = {
5219 .type = HDA_FIXUP_FUNC,
5220 .v.func = alc269_fixup_limit_int_mic_boost,
5221 .chained = true,
5222 .chain_id = ALC269_FIXUP_HP_MUTE_LED_MIC1,
5223 },
Anisse Astier02b504d2013-06-03 11:53:10 +02005224 [ALC269VB_FIXUP_ORDISSIMO_EVE2] = {
5225 .type = HDA_FIXUP_PINS,
5226 .v.pins = (const struct hda_pintbl[]) {
5227 { 0x12, 0x99a3092f }, /* int-mic */
5228 { 0x18, 0x03a11d20 }, /* mic */
5229 { 0x19, 0x411111f0 }, /* Unused bogus pin */
5230 { }
5231 },
5232 },
Kailang Yangcd217a62013-08-22 10:15:24 +02005233 [ALC283_FIXUP_CHROME_BOOK] = {
5234 .type = HDA_FIXUP_FUNC,
5235 .v.func = alc283_fixup_chromebook,
5236 },
Kailang Yang0202e992013-12-02 15:20:15 +08005237 [ALC283_FIXUP_SENSE_COMBO_JACK] = {
5238 .type = HDA_FIXUP_FUNC,
5239 .v.func = alc283_fixup_sense_combo_jack,
5240 .chained = true,
5241 .chain_id = ALC283_FIXUP_CHROME_BOOK,
5242 },
Takashi Iwai7bba2152013-09-06 15:45:38 +02005243 [ALC282_FIXUP_ASUS_TX300] = {
5244 .type = HDA_FIXUP_FUNC,
5245 .v.func = alc282_fixup_asus_tx300,
5246 },
Kailang Yang1bb3e062013-09-27 13:10:25 +02005247 [ALC283_FIXUP_INT_MIC] = {
5248 .type = HDA_FIXUP_VERBS,
5249 .v.verbs = (const struct hda_verb[]) {
5250 {0x20, AC_VERB_SET_COEF_INDEX, 0x1a},
5251 {0x20, AC_VERB_SET_PROC_COEF, 0x0011},
5252 { }
5253 },
5254 .chained = true,
5255 .chain_id = ALC269_FIXUP_LIMIT_INT_MIC_BOOST
5256 },
David Henningsson0f4881d2013-12-20 16:08:13 +01005257 [ALC290_FIXUP_SUBWOOFER_HSJACK] = {
5258 .type = HDA_FIXUP_PINS,
5259 .v.pins = (const struct hda_pintbl[]) {
5260 { 0x17, 0x90170112 }, /* subwoofer */
5261 { }
5262 },
5263 .chained = true,
5264 .chain_id = ALC290_FIXUP_MONO_SPEAKERS_HSJACK,
5265 },
5266 [ALC290_FIXUP_SUBWOOFER] = {
5267 .type = HDA_FIXUP_PINS,
5268 .v.pins = (const struct hda_pintbl[]) {
5269 { 0x17, 0x90170112 }, /* subwoofer */
5270 { }
5271 },
5272 .chained = true,
5273 .chain_id = ALC290_FIXUP_MONO_SPEAKERS,
5274 },
David Henningsson338cae52013-10-07 10:39:59 +02005275 [ALC290_FIXUP_MONO_SPEAKERS] = {
5276 .type = HDA_FIXUP_FUNC,
5277 .v.func = alc290_fixup_mono_speakers,
David Henningsson0f4881d2013-12-20 16:08:13 +01005278 },
5279 [ALC290_FIXUP_MONO_SPEAKERS_HSJACK] = {
5280 .type = HDA_FIXUP_FUNC,
5281 .v.func = alc290_fixup_mono_speakers,
David Henningsson338cae52013-10-07 10:39:59 +02005282 .chained = true,
5283 .chain_id = ALC269_FIXUP_DELL3_MIC_NO_PRESENCE,
5284 },
David Henningssonb67ae3f2013-11-05 13:11:37 +01005285 [ALC269_FIXUP_THINKPAD_ACPI] = {
5286 .type = HDA_FIXUP_FUNC,
Takashi Iwaib317b032014-01-08 11:44:21 +01005287 .v.func = hda_fixup_thinkpad_acpi,
Takashi Iwai09da1112016-09-08 12:15:02 +02005288 .chained = true,
5289 .chain_id = ALC269_FIXUP_SKU_IGNORE,
David Henningssonb67ae3f2013-11-05 13:11:37 +01005290 },
David Henningsson56f27012016-01-11 09:33:14 +01005291 [ALC269_FIXUP_DMIC_THINKPAD_ACPI] = {
5292 .type = HDA_FIXUP_FUNC,
5293 .v.func = alc_fixup_inv_dmic,
5294 .chained = true,
5295 .chain_id = ALC269_FIXUP_THINKPAD_ACPI,
5296 },
Kailang Yang9a22a8f2013-11-08 15:54:49 +08005297 [ALC255_FIXUP_DELL1_MIC_NO_PRESENCE] = {
5298 .type = HDA_FIXUP_PINS,
5299 .v.pins = (const struct hda_pintbl[]) {
5300 { 0x19, 0x01a1913c }, /* use as headset mic, without its own jack detect */
5301 { 0x1a, 0x01a1913d }, /* use as headphone mic, without its own jack detect */
5302 { }
5303 },
5304 .chained = true,
5305 .chain_id = ALC255_FIXUP_HEADSET_MODE
5306 },
Kailang Yang31278992014-03-03 15:27:22 +08005307 [ALC255_FIXUP_DELL2_MIC_NO_PRESENCE] = {
5308 .type = HDA_FIXUP_PINS,
5309 .v.pins = (const struct hda_pintbl[]) {
5310 { 0x19, 0x01a1913c }, /* use as headset mic, without its own jack detect */
5311 { }
5312 },
5313 .chained = true,
5314 .chain_id = ALC255_FIXUP_HEADSET_MODE_NO_HP_MIC
5315 },
Kailang Yang9a22a8f2013-11-08 15:54:49 +08005316 [ALC255_FIXUP_HEADSET_MODE] = {
5317 .type = HDA_FIXUP_FUNC,
5318 .v.func = alc_fixup_headset_mode_alc255,
Hui Wang4a83d422014-11-18 17:57:40 +08005319 .chained = true,
5320 .chain_id = ALC255_FIXUP_DELL_WMI_MIC_MUTE_LED
Kailang Yang9a22a8f2013-11-08 15:54:49 +08005321 },
Kailang Yang31278992014-03-03 15:27:22 +08005322 [ALC255_FIXUP_HEADSET_MODE_NO_HP_MIC] = {
5323 .type = HDA_FIXUP_FUNC,
5324 .v.func = alc_fixup_headset_mode_alc255_no_hp_mic,
5325 },
Kailang Yanga22aa262014-04-23 17:34:28 +08005326 [ALC293_FIXUP_DELL1_MIC_NO_PRESENCE] = {
5327 .type = HDA_FIXUP_PINS,
5328 .v.pins = (const struct hda_pintbl[]) {
5329 { 0x18, 0x01a1913d }, /* use as headphone mic, without its own jack detect */
5330 { 0x1a, 0x01a1913c }, /* use as headset mic, without its own jack detect */
5331 { }
5332 },
5333 .chained = true,
5334 .chain_id = ALC269_FIXUP_HEADSET_MODE
5335 },
Takashi Iwai1c37c222014-05-06 17:34:42 +02005336 [ALC292_FIXUP_TPT440_DOCK] = {
David Henningssonec56af62015-06-24 10:46:33 +02005337 .type = HDA_FIXUP_FUNC,
Takashi Iwai7f57d802015-09-24 17:36:51 +02005338 .v.func = alc_fixup_tpt440_dock,
Takashi Iwai1c37c222014-05-06 17:34:42 +02005339 .chained = true,
5340 .chain_id = ALC269_FIXUP_LIMIT_INT_MIC_BOOST
5341 },
Takashi Iwai9a811232015-12-09 15:17:43 +01005342 [ALC292_FIXUP_TPT440] = {
5343 .type = HDA_FIXUP_FUNC,
Takashi Iwai157f0b72015-12-10 23:30:43 +01005344 .v.func = alc_fixup_disable_aamix,
Takashi Iwai9a811232015-12-09 15:17:43 +01005345 .chained = true,
5346 .chain_id = ALC292_FIXUP_TPT440_DOCK,
5347 },
Anisse Astierabaa22742016-08-24 09:14:13 +02005348 [ALC283_FIXUP_HEADSET_MIC] = {
Daniel Drake9dc12862014-07-22 10:58:29 +01005349 .type = HDA_FIXUP_PINS,
5350 .v.pins = (const struct hda_pintbl[]) {
5351 { 0x19, 0x04a110f0 },
5352 { },
5353 },
5354 },
Hui Wang00ef9942014-07-31 11:52:38 +08005355 [ALC255_FIXUP_DELL_WMI_MIC_MUTE_LED] = {
5356 .type = HDA_FIXUP_FUNC,
5357 .v.func = alc_fixup_dell_wmi,
Hui Wang00ef9942014-07-31 11:52:38 +08005358 },
Takashi Iwai1a22e772014-08-27 08:19:05 +02005359 [ALC282_FIXUP_ASPIRE_V5_PINS] = {
5360 .type = HDA_FIXUP_PINS,
5361 .v.pins = (const struct hda_pintbl[]) {
5362 { 0x12, 0x90a60130 },
5363 { 0x14, 0x90170110 },
5364 { 0x17, 0x40000008 },
5365 { 0x18, 0x411111f0 },
Mateusz Sylwestrzak04206942015-07-19 17:38:56 +02005366 { 0x19, 0x01a1913c },
Takashi Iwai1a22e772014-08-27 08:19:05 +02005367 { 0x1a, 0x411111f0 },
5368 { 0x1b, 0x411111f0 },
5369 { 0x1d, 0x40f89b2d },
5370 { 0x1e, 0x411111f0 },
5371 { 0x21, 0x0321101f },
5372 { },
5373 },
5374 },
David Henningsson7a5255f2014-10-30 08:26:01 +01005375 [ALC280_FIXUP_HP_GPIO4] = {
5376 .type = HDA_FIXUP_FUNC,
5377 .v.func = alc280_fixup_hp_gpio4,
5378 },
Kailang Yangeaa8e5e2014-11-21 15:49:11 +08005379 [ALC286_FIXUP_HP_GPIO_LED] = {
5380 .type = HDA_FIXUP_FUNC,
5381 .v.func = alc286_fixup_hp_gpio_led,
5382 },
David Henningsson33f4acd2015-01-07 15:50:13 +01005383 [ALC280_FIXUP_HP_GPIO2_MIC_HOTKEY] = {
5384 .type = HDA_FIXUP_FUNC,
5385 .v.func = alc280_fixup_hp_gpio2_mic_hotkey,
5386 },
TienFu Chenb4b33f92015-01-20 15:06:21 +01005387 [ALC280_FIXUP_HP_DOCK_PINS] = {
5388 .type = HDA_FIXUP_PINS,
5389 .v.pins = (const struct hda_pintbl[]) {
5390 { 0x1b, 0x21011020 }, /* line-out */
5391 { 0x1a, 0x01a1903c }, /* headset mic */
5392 { 0x18, 0x2181103f }, /* line-in */
5393 { },
5394 },
5395 .chained = true,
5396 .chain_id = ALC280_FIXUP_HP_GPIO4
5397 },
Jaroslav Kysela52c33232017-03-09 13:29:13 +01005398 [ALC269_FIXUP_HP_DOCK_GPIO_MIC1_LED] = {
5399 .type = HDA_FIXUP_PINS,
5400 .v.pins = (const struct hda_pintbl[]) {
5401 { 0x1b, 0x21011020 }, /* line-out */
5402 { 0x18, 0x2181103f }, /* line-in */
5403 { },
5404 },
5405 .chained = true,
5406 .chain_id = ALC269_FIXUP_HP_GPIO_MIC1_LED
5407 },
Keith Packard98973f22015-07-15 12:14:39 -07005408 [ALC280_FIXUP_HP_9480M] = {
5409 .type = HDA_FIXUP_FUNC,
5410 .v.func = alc280_fixup_hp_9480m,
5411 },
Kailang Yange1e62b92015-04-08 16:01:22 +08005412 [ALC288_FIXUP_DELL_HEADSET_MODE] = {
5413 .type = HDA_FIXUP_FUNC,
5414 .v.func = alc_fixup_headset_mode_dell_alc288,
5415 .chained = true,
5416 .chain_id = ALC255_FIXUP_DELL_WMI_MIC_MUTE_LED
5417 },
5418 [ALC288_FIXUP_DELL1_MIC_NO_PRESENCE] = {
5419 .type = HDA_FIXUP_PINS,
5420 .v.pins = (const struct hda_pintbl[]) {
5421 { 0x18, 0x01a1913c }, /* use as headset mic, without its own jack detect */
5422 { 0x1a, 0x01a1913d }, /* use as headphone mic, without its own jack detect */
5423 { }
5424 },
5425 .chained = true,
5426 .chain_id = ALC288_FIXUP_DELL_HEADSET_MODE
5427 },
5428 [ALC288_FIXUP_DELL_XPS_13_GPIO6] = {
5429 .type = HDA_FIXUP_VERBS,
5430 .v.verbs = (const struct hda_verb[]) {
5431 {0x01, AC_VERB_SET_GPIO_MASK, 0x40},
5432 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x40},
5433 {0x01, AC_VERB_SET_GPIO_DATA, 0x00},
5434 { }
5435 },
5436 .chained = true,
5437 .chain_id = ALC288_FIXUP_DELL1_MIC_NO_PRESENCE
5438 },
Hui Wang831bfdf2015-06-26 12:35:17 +08005439 [ALC288_FIXUP_DISABLE_AAMIX] = {
5440 .type = HDA_FIXUP_FUNC,
5441 .v.func = alc_fixup_disable_aamix,
5442 .chained = true,
5443 .chain_id = ALC288_FIXUP_DELL_XPS_13_GPIO6
5444 },
5445 [ALC288_FIXUP_DELL_XPS_13] = {
5446 .type = HDA_FIXUP_FUNC,
5447 .v.func = alc_fixup_dell_xps13,
5448 .chained = true,
5449 .chain_id = ALC288_FIXUP_DISABLE_AAMIX
5450 },
Takashi Iwai8b99aba2015-06-15 11:59:32 +02005451 [ALC292_FIXUP_DISABLE_AAMIX] = {
5452 .type = HDA_FIXUP_FUNC,
5453 .v.func = alc_fixup_disable_aamix,
Hui Wang831bfdf2015-06-26 12:35:17 +08005454 .chained = true,
5455 .chain_id = ALC269_FIXUP_DELL2_MIC_NO_PRESENCE
Takashi Iwai8b99aba2015-06-15 11:59:32 +02005456 },
David Henningssonc04017e2015-12-15 14:44:03 +01005457 [ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK] = {
5458 .type = HDA_FIXUP_FUNC,
5459 .v.func = alc_fixup_disable_aamix,
5460 .chained = true,
5461 .chain_id = ALC293_FIXUP_DELL1_MIC_NO_PRESENCE
5462 },
Takashi Iwai8b99aba2015-06-15 11:59:32 +02005463 [ALC292_FIXUP_DELL_E7X] = {
5464 .type = HDA_FIXUP_FUNC,
5465 .v.func = alc_fixup_dell_xps13,
5466 .chained = true,
5467 .chain_id = ALC292_FIXUP_DISABLE_AAMIX
5468 },
Kailang Yang977e6272015-05-18 15:31:20 +08005469 [ALC298_FIXUP_DELL1_MIC_NO_PRESENCE] = {
5470 .type = HDA_FIXUP_PINS,
5471 .v.pins = (const struct hda_pintbl[]) {
5472 { 0x18, 0x01a1913c }, /* use as headset mic, without its own jack detect */
5473 { 0x1a, 0x01a1913d }, /* use as headphone mic, without its own jack detect */
5474 { }
5475 },
5476 .chained = true,
5477 .chain_id = ALC269_FIXUP_HEADSET_MODE
5478 },
Hui Wang8aabccd2017-03-31 10:31:40 +08005479 [ALC298_FIXUP_DELL_AIO_MIC_NO_PRESENCE] = {
5480 .type = HDA_FIXUP_PINS,
5481 .v.pins = (const struct hda_pintbl[]) {
5482 { 0x18, 0x01a1913c }, /* use as headset mic, without its own jack detect */
5483 { }
5484 },
5485 .chained = true,
5486 .chain_id = ALC269_FIXUP_HEADSET_MODE
5487 },
Kailang Yang6ed11312015-10-26 15:37:39 +08005488 [ALC275_FIXUP_DELL_XPS] = {
5489 .type = HDA_FIXUP_VERBS,
5490 .v.verbs = (const struct hda_verb[]) {
5491 /* Enables internal speaker */
5492 {0x20, AC_VERB_SET_COEF_INDEX, 0x1f},
5493 {0x20, AC_VERB_SET_PROC_COEF, 0x00c0},
5494 {0x20, AC_VERB_SET_COEF_INDEX, 0x30},
5495 {0x20, AC_VERB_SET_PROC_COEF, 0x00b1},
5496 {}
5497 }
5498 },
Hui Wang8c697292015-11-24 11:08:18 +08005499 [ALC256_FIXUP_DELL_XPS_13_HEADPHONE_NOISE] = {
5500 .type = HDA_FIXUP_VERBS,
5501 .v.verbs = (const struct hda_verb[]) {
5502 /* Disable pass-through path for FRONT 14h */
5503 {0x20, AC_VERB_SET_COEF_INDEX, 0x36},
5504 {0x20, AC_VERB_SET_PROC_COEF, 0x1737},
5505 {}
5506 },
5507 .chained = true,
5508 .chain_id = ALC255_FIXUP_DELL1_MIC_NO_PRESENCE
5509 },
Hui Wang23adc192015-12-08 12:27:18 +08005510 [ALC293_FIXUP_LENOVO_SPK_NOISE] = {
5511 .type = HDA_FIXUP_FUNC,
5512 .v.func = alc_fixup_disable_aamix,
5513 .chained = true,
5514 .chain_id = ALC269_FIXUP_THINKPAD_ACPI
5515 },
Kailang3694cb22015-12-28 11:35:24 +08005516 [ALC233_FIXUP_LENOVO_LINE2_MIC_HOTKEY] = {
5517 .type = HDA_FIXUP_FUNC,
5518 .v.func = alc233_fixup_lenovo_line2_mic_hotkey,
5519 },
Kai-Heng Feng3b43b712016-02-25 15:19:38 +08005520 [ALC255_FIXUP_DELL_SPK_NOISE] = {
5521 .type = HDA_FIXUP_FUNC,
5522 .v.func = alc_fixup_disable_aamix,
5523 .chained = true,
5524 .chain_id = ALC255_FIXUP_DELL1_MIC_NO_PRESENCE
5525 },
David Henningsson2ae95572016-02-25 09:37:05 +01005526 [ALC225_FIXUP_DELL1_MIC_NO_PRESENCE] = {
5527 .type = HDA_FIXUP_VERBS,
5528 .v.verbs = (const struct hda_verb[]) {
5529 /* Disable pass-through path for FRONT 14h */
5530 { 0x20, AC_VERB_SET_COEF_INDEX, 0x36 },
5531 { 0x20, AC_VERB_SET_PROC_COEF, 0x57d7 },
5532 {}
5533 },
5534 .chained = true,
5535 .chain_id = ALC269_FIXUP_DELL1_MIC_NO_PRESENCE
5536 },
Takashi Iwaif8839822016-02-25 14:31:59 +01005537 [ALC280_FIXUP_HP_HEADSET_MIC] = {
5538 .type = HDA_FIXUP_FUNC,
5539 .v.func = alc_fixup_disable_aamix,
5540 .chained = true,
5541 .chain_id = ALC269_FIXUP_HEADSET_MIC,
5542 },
Hui Wange549d192016-04-01 11:00:15 +08005543 [ALC221_FIXUP_HP_FRONT_MIC] = {
5544 .type = HDA_FIXUP_PINS,
5545 .v.pins = (const struct hda_pintbl[]) {
5546 { 0x19, 0x02a19020 }, /* Front Mic */
5547 { }
5548 },
5549 },
Sven Eckelmannc636b952016-04-11 16:55:26 +02005550 [ALC292_FIXUP_TPT460] = {
5551 .type = HDA_FIXUP_FUNC,
5552 .v.func = alc_fixup_tpt440_dock,
5553 .chained = true,
5554 .chain_id = ALC293_FIXUP_LENOVO_SPK_NOISE,
5555 },
Hui Wangdd9aa332016-08-01 10:20:32 +08005556 [ALC298_FIXUP_SPK_VOLUME] = {
5557 .type = HDA_FIXUP_FUNC,
5558 .v.func = alc298_fixup_speaker_volume,
Hui Wang59ec4b52016-08-04 15:28:04 +08005559 .chained = true,
Hui Wang8aabccd2017-03-31 10:31:40 +08005560 .chain_id = ALC298_FIXUP_DELL_AIO_MIC_NO_PRESENCE,
Hui Wangdd9aa332016-08-01 10:20:32 +08005561 },
Kai-Heng Fengfd06c772016-08-30 15:36:34 +08005562 [ALC256_FIXUP_DELL_INSPIRON_7559_SUBWOOFER] = {
5563 .type = HDA_FIXUP_PINS,
5564 .v.pins = (const struct hda_pintbl[]) {
5565 { 0x1b, 0x90170151 },
5566 { }
5567 },
5568 .chained = true,
5569 .chain_id = ALC255_FIXUP_DELL1_MIC_NO_PRESENCE
5570 },
Takashi Iwai1d045db2011-07-07 18:23:21 +02005571};
5572
5573static const struct snd_pci_quirk alc269_fixup_tbl[] = {
Marius Knausta6b92b62014-03-03 01:48:58 +01005574 SND_PCI_QUIRK(0x1025, 0x0283, "Acer TravelMate 8371", ALC269_FIXUP_INV_DMIC),
David Henningsson693b6132012-06-22 19:12:10 +02005575 SND_PCI_QUIRK(0x1025, 0x029b, "Acer 1810TZ", ALC269_FIXUP_INV_DMIC),
5576 SND_PCI_QUIRK(0x1025, 0x0349, "Acer AOD260", ALC269_FIXUP_INV_DMIC),
David Henningssonaaedfb42013-08-16 14:09:02 +02005577 SND_PCI_QUIRK(0x1025, 0x047c, "Acer AC700", ALC269_FIXUP_ACER_AC700),
Takashi Iwai78197172015-06-27 10:21:13 +02005578 SND_PCI_QUIRK(0x1025, 0x072d, "Acer Aspire V5-571G", ALC269_FIXUP_ASPIRE_HEADSET_MIC),
5579 SND_PCI_QUIRK(0x1025, 0x080d, "Acer Aspire V5-122P", ALC269_FIXUP_ASPIRE_HEADSET_MIC),
David Henningssonaaedfb42013-08-16 14:09:02 +02005580 SND_PCI_QUIRK(0x1025, 0x0740, "Acer AO725", ALC271_FIXUP_HP_GATE_MIC_JACK),
5581 SND_PCI_QUIRK(0x1025, 0x0742, "Acer AO756", ALC271_FIXUP_HP_GATE_MIC_JACK),
Simon South02322ac2016-03-02 23:10:44 -05005582 SND_PCI_QUIRK(0x1025, 0x0762, "Acer Aspire E1-472", ALC271_FIXUP_HP_GATE_MIC_JACK_E1_572),
Oleksij Rempelb1e89722013-12-04 20:50:53 +01005583 SND_PCI_QUIRK(0x1025, 0x0775, "Acer Aspire E1-572", ALC271_FIXUP_HP_GATE_MIC_JACK_E1_572),
Takashi Iwai1a22e772014-08-27 08:19:05 +02005584 SND_PCI_QUIRK(0x1025, 0x079b, "Acer Aspire V5-573G", ALC282_FIXUP_ASPIRE_V5_PINS),
Takashi Iwaib9c2fa52015-11-19 16:39:50 +01005585 SND_PCI_QUIRK(0x1025, 0x106d, "Acer Cloudbook 14", ALC283_FIXUP_CHROME_BOOK),
David Henningssonaaedfb42013-08-16 14:09:02 +02005586 SND_PCI_QUIRK(0x1028, 0x0470, "Dell M101z", ALC269_FIXUP_DELL_M101Z),
Kailang Yang6ed11312015-10-26 15:37:39 +08005587 SND_PCI_QUIRK(0x1028, 0x054b, "Dell XPS one 2710", ALC275_FIXUP_DELL_XPS),
Takashi Iwai86f799b2015-11-14 17:46:31 +01005588 SND_PCI_QUIRK(0x1028, 0x05bd, "Dell Latitude E6440", ALC292_FIXUP_DELL_E7X),
Takashi Iwaicf521032016-01-15 12:59:25 +01005589 SND_PCI_QUIRK(0x1028, 0x05be, "Dell Latitude E6540", ALC292_FIXUP_DELL_E7X),
Takashi Iwai8b99aba2015-06-15 11:59:32 +02005590 SND_PCI_QUIRK(0x1028, 0x05ca, "Dell Latitude E7240", ALC292_FIXUP_DELL_E7X),
5591 SND_PCI_QUIRK(0x1028, 0x05cb, "Dell Latitude E7440", ALC292_FIXUP_DELL_E7X),
David Henningsson0f4881d2013-12-20 16:08:13 +01005592 SND_PCI_QUIRK(0x1028, 0x05da, "Dell Vostro 5460", ALC290_FIXUP_SUBWOOFER),
David Henningsson73bdd592013-04-15 15:44:14 +02005593 SND_PCI_QUIRK(0x1028, 0x05f4, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE),
5594 SND_PCI_QUIRK(0x1028, 0x05f5, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE),
5595 SND_PCI_QUIRK(0x1028, 0x05f6, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE),
David Henningsson0f4881d2013-12-20 16:08:13 +01005596 SND_PCI_QUIRK(0x1028, 0x0615, "Dell Vostro 5470", ALC290_FIXUP_SUBWOOFER_HSJACK),
5597 SND_PCI_QUIRK(0x1028, 0x0616, "Dell Vostro 5470", ALC290_FIXUP_SUBWOOFER_HSJACK),
Takashi Iwai98070572016-01-12 21:06:39 +01005598 SND_PCI_QUIRK(0x1028, 0x062c, "Dell Latitude E5550", ALC292_FIXUP_DELL_E7X),
Takashi Iwai42755542015-06-29 10:56:53 +02005599 SND_PCI_QUIRK(0x1028, 0x062e, "Dell Latitude E7450", ALC292_FIXUP_DELL_E7X),
David Henningsson0f4881d2013-12-20 16:08:13 +01005600 SND_PCI_QUIRK(0x1028, 0x0638, "Dell Inspiron 5439", ALC290_FIXUP_MONO_SPEAKERS_HSJACK),
Kailang Yanga22aa262014-04-23 17:34:28 +08005601 SND_PCI_QUIRK(0x1028, 0x064a, "Dell", ALC293_FIXUP_DELL1_MIC_NO_PRESENCE),
5602 SND_PCI_QUIRK(0x1028, 0x064b, "Dell", ALC293_FIXUP_DELL1_MIC_NO_PRESENCE),
Hui Wang831bfdf2015-06-26 12:35:17 +08005603 SND_PCI_QUIRK(0x1028, 0x0665, "Dell XPS 13", ALC288_FIXUP_DELL_XPS_13),
Bastien Noceraafecb142016-04-18 11:10:42 +02005604 SND_PCI_QUIRK(0x1028, 0x0669, "Dell Optiplex 9020m", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE),
Takashi Iwai3a05d122015-07-29 09:04:52 +02005605 SND_PCI_QUIRK(0x1028, 0x069a, "Dell Vostro 5480", ALC290_FIXUP_SUBWOOFER_HSJACK),
Kailang Yang8b724152014-12-17 17:08:59 +08005606 SND_PCI_QUIRK(0x1028, 0x06c7, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE),
Kailang Yangb7343042014-12-02 14:50:45 +08005607 SND_PCI_QUIRK(0x1028, 0x06d9, "Dell", ALC293_FIXUP_DELL1_MIC_NO_PRESENCE),
5608 SND_PCI_QUIRK(0x1028, 0x06da, "Dell", ALC293_FIXUP_DELL1_MIC_NO_PRESENCE),
David Henningssonc04017e2015-12-15 14:44:03 +01005609 SND_PCI_QUIRK(0x1028, 0x06db, "Dell", ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK),
5610 SND_PCI_QUIRK(0x1028, 0x06dd, "Dell", ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK),
5611 SND_PCI_QUIRK(0x1028, 0x06de, "Dell", ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK),
5612 SND_PCI_QUIRK(0x1028, 0x06df, "Dell", ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK),
5613 SND_PCI_QUIRK(0x1028, 0x06e0, "Dell", ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK),
Kai-Heng Feng423cd782016-05-20 15:47:23 +08005614 SND_PCI_QUIRK(0x1028, 0x0704, "Dell XPS 13 9350", ALC256_FIXUP_DELL_XPS_13_HEADPHONE_NOISE),
Kai-Heng Fengfd06c772016-08-30 15:36:34 +08005615 SND_PCI_QUIRK(0x1028, 0x0706, "Dell Inspiron 7559", ALC256_FIXUP_DELL_INSPIRON_7559_SUBWOOFER),
Kai-Heng Feng3b43b712016-02-25 15:19:38 +08005616 SND_PCI_QUIRK(0x1028, 0x0725, "Dell Inspiron 3162", ALC255_FIXUP_DELL_SPK_NOISE),
Kai-Heng Feng423cd782016-05-20 15:47:23 +08005617 SND_PCI_QUIRK(0x1028, 0x075b, "Dell XPS 13 9360", ALC256_FIXUP_DELL_XPS_13_HEADPHONE_NOISE),
Hui Wangdd9aa332016-08-01 10:20:32 +08005618 SND_PCI_QUIRK(0x1028, 0x075d, "Dell AIO", ALC298_FIXUP_SPK_VOLUME),
Takashi Iwai2abe6202017-02-28 17:27:57 +01005619 SND_PCI_QUIRK(0x1028, 0x0798, "Dell Inspiron 17 7000 Gaming", ALC256_FIXUP_DELL_INSPIRON_7559_SUBWOOFER),
Takashi Iwaifae704d2018-01-10 08:34:28 +01005620 SND_PCI_QUIRK(0x1028, 0x082a, "Dell XPS 13 9360", ALC256_FIXUP_DELL_XPS_13_HEADPHONE_NOISE),
Kailang Yanga22aa262014-04-23 17:34:28 +08005621 SND_PCI_QUIRK(0x1028, 0x164a, "Dell", ALC293_FIXUP_DELL1_MIC_NO_PRESENCE),
5622 SND_PCI_QUIRK(0x1028, 0x164b, "Dell", ALC293_FIXUP_DELL1_MIC_NO_PRESENCE),
Takashi Iwai08fb0d02013-01-10 17:33:58 +01005623 SND_PCI_QUIRK(0x103c, 0x1586, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC2),
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01005624 SND_PCI_QUIRK(0x103c, 0x18e6, "HP", ALC269_FIXUP_HP_GPIO_LED),
David Henningsson8e35cd42013-11-06 11:20:01 +01005625 SND_PCI_QUIRK(0x103c, 0x218b, "HP", ALC269_FIXUP_LIMIT_INT_MIC_BOOST_MUTE_LED),
David Henningsson33f4acd2015-01-07 15:50:13 +01005626 SND_PCI_QUIRK(0x103c, 0x225f, "HP", ALC280_FIXUP_HP_GPIO2_MIC_HOTKEY),
Kailang Yangc60666b2014-02-21 16:23:35 +08005627 /* ALC282 */
Hui Wang7976eb42015-02-13 11:14:41 +08005628 SND_PCI_QUIRK(0x103c, 0x21f9, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yang8a02b162014-06-13 17:16:31 +08005629 SND_PCI_QUIRK(0x103c, 0x2210, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yang8a02b162014-06-13 17:16:31 +08005630 SND_PCI_QUIRK(0x103c, 0x2214, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08005631 SND_PCI_QUIRK(0x103c, 0x2236, "HP", ALC269_FIXUP_HP_LINE1_MIC1_LED),
5632 SND_PCI_QUIRK(0x103c, 0x2237, "HP", ALC269_FIXUP_HP_LINE1_MIC1_LED),
5633 SND_PCI_QUIRK(0x103c, 0x2238, "HP", ALC269_FIXUP_HP_LINE1_MIC1_LED),
5634 SND_PCI_QUIRK(0x103c, 0x2239, "HP", ALC269_FIXUP_HP_LINE1_MIC1_LED),
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08005635 SND_PCI_QUIRK(0x103c, 0x224b, "HP", ALC269_FIXUP_HP_LINE1_MIC1_LED),
Kailang Yangc60666b2014-02-21 16:23:35 +08005636 SND_PCI_QUIRK(0x103c, 0x2268, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangc60666b2014-02-21 16:23:35 +08005637 SND_PCI_QUIRK(0x103c, 0x226a, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
5638 SND_PCI_QUIRK(0x103c, 0x226b, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yang8a02b162014-06-13 17:16:31 +08005639 SND_PCI_QUIRK(0x103c, 0x226e, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangeaa8e5e2014-11-21 15:49:11 +08005640 SND_PCI_QUIRK(0x103c, 0x2271, "HP", ALC286_FIXUP_HP_GPIO_LED),
TienFu Chenb4b33f92015-01-20 15:06:21 +01005641 SND_PCI_QUIRK(0x103c, 0x2272, "HP", ALC280_FIXUP_HP_DOCK_PINS),
TienFu Chen3271cb22015-02-10 09:09:41 +01005642 SND_PCI_QUIRK(0x103c, 0x2273, "HP", ALC280_FIXUP_HP_DOCK_PINS),
Kailang Yangc60666b2014-02-21 16:23:35 +08005643 SND_PCI_QUIRK(0x103c, 0x229e, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangc60666b2014-02-21 16:23:35 +08005644 SND_PCI_QUIRK(0x103c, 0x22b2, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
5645 SND_PCI_QUIRK(0x103c, 0x22b7, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
5646 SND_PCI_QUIRK(0x103c, 0x22bf, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangc60666b2014-02-21 16:23:35 +08005647 SND_PCI_QUIRK(0x103c, 0x22cf, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Keith Packard98973f22015-07-15 12:14:39 -07005648 SND_PCI_QUIRK(0x103c, 0x22db, "HP", ALC280_FIXUP_HP_9480M),
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08005649 SND_PCI_QUIRK(0x103c, 0x22dc, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
5650 SND_PCI_QUIRK(0x103c, 0x22fb, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
Kailang Yangc60666b2014-02-21 16:23:35 +08005651 /* ALC290 */
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08005652 SND_PCI_QUIRK(0x103c, 0x221b, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08005653 SND_PCI_QUIRK(0x103c, 0x2221, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08005654 SND_PCI_QUIRK(0x103c, 0x2225, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08005655 SND_PCI_QUIRK(0x103c, 0x2253, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
5656 SND_PCI_QUIRK(0x103c, 0x2254, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
5657 SND_PCI_QUIRK(0x103c, 0x2255, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
5658 SND_PCI_QUIRK(0x103c, 0x2256, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
5659 SND_PCI_QUIRK(0x103c, 0x2257, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08005660 SND_PCI_QUIRK(0x103c, 0x2259, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
Jaroslav Kysela52c33232017-03-09 13:29:13 +01005661 SND_PCI_QUIRK(0x103c, 0x225a, "HP", ALC269_FIXUP_HP_DOCK_GPIO_MIC1_LED),
Kailang Yangc60666b2014-02-21 16:23:35 +08005662 SND_PCI_QUIRK(0x103c, 0x2260, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangc60666b2014-02-21 16:23:35 +08005663 SND_PCI_QUIRK(0x103c, 0x2263, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
5664 SND_PCI_QUIRK(0x103c, 0x2264, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
5665 SND_PCI_QUIRK(0x103c, 0x2265, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08005666 SND_PCI_QUIRK(0x103c, 0x2272, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
5667 SND_PCI_QUIRK(0x103c, 0x2273, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08005668 SND_PCI_QUIRK(0x103c, 0x2278, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
Kailang Yangc60666b2014-02-21 16:23:35 +08005669 SND_PCI_QUIRK(0x103c, 0x227f, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangc60666b2014-02-21 16:23:35 +08005670 SND_PCI_QUIRK(0x103c, 0x2282, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangc60666b2014-02-21 16:23:35 +08005671 SND_PCI_QUIRK(0x103c, 0x228b, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangc60666b2014-02-21 16:23:35 +08005672 SND_PCI_QUIRK(0x103c, 0x228e, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
5673 SND_PCI_QUIRK(0x103c, 0x22c5, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangc60666b2014-02-21 16:23:35 +08005674 SND_PCI_QUIRK(0x103c, 0x22c7, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
5675 SND_PCI_QUIRK(0x103c, 0x22c8, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangc60666b2014-02-21 16:23:35 +08005676 SND_PCI_QUIRK(0x103c, 0x22c4, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yang8a02b162014-06-13 17:16:31 +08005677 SND_PCI_QUIRK(0x103c, 0x2334, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
5678 SND_PCI_QUIRK(0x103c, 0x2335, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
5679 SND_PCI_QUIRK(0x103c, 0x2336, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
5680 SND_PCI_QUIRK(0x103c, 0x2337, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Takashi Iwaif8839822016-02-25 14:31:59 +01005681 SND_PCI_QUIRK(0x103c, 0x221c, "HP EliteBook 755 G2", ALC280_FIXUP_HP_HEADSET_MIC),
Hui Wange549d192016-04-01 11:00:15 +08005682 SND_PCI_QUIRK(0x103c, 0x8256, "HP", ALC221_FIXUP_HP_FRONT_MIC),
Takashi Iwai7bba2152013-09-06 15:45:38 +02005683 SND_PCI_QUIRK(0x1043, 0x103f, "ASUS TX300", ALC282_FIXUP_ASUS_TX300),
David Henningsson3e0d6112013-04-22 14:30:14 +02005684 SND_PCI_QUIRK(0x1043, 0x106d, "Asus K53BE", ALC269_FIXUP_LIMIT_INT_MIC_BOOST),
5685 SND_PCI_QUIRK(0x1043, 0x115d, "Asus 1015E", ALC269_FIXUP_LIMIT_INT_MIC_BOOST),
Oleksij Rempel2cede302013-11-27 17:12:03 +01005686 SND_PCI_QUIRK(0x1043, 0x1427, "Asus Zenbook UX31E", ALC269VB_FIXUP_ASUS_ZENBOOK),
Takashi Iwai23870832013-11-29 14:13:12 +01005687 SND_PCI_QUIRK(0x1043, 0x1517, "Asus Zenbook UX31A", ALC269VB_FIXUP_ASUS_ZENBOOK_UX31A),
David Henningsson3e0d6112013-04-22 14:30:14 +02005688 SND_PCI_QUIRK(0x1043, 0x16e3, "ASUS UX50", ALC269_FIXUP_STEREO_DMIC),
Takashi Iwai017f2a12011-07-09 14:42:25 +02005689 SND_PCI_QUIRK(0x1043, 0x1a13, "Asus G73Jw", ALC269_FIXUP_ASUS_G73JW),
David Henningsson693b6132012-06-22 19:12:10 +02005690 SND_PCI_QUIRK(0x1043, 0x1b13, "Asus U41SV", ALC269_FIXUP_INV_DMIC),
David Henningsson3e0d6112013-04-22 14:30:14 +02005691 SND_PCI_QUIRK(0x1043, 0x1c23, "Asus X55U", ALC269_FIXUP_LIMIT_INT_MIC_BOOST),
Takashi Iwaiadabb3e2011-08-03 07:48:37 +02005692 SND_PCI_QUIRK(0x1043, 0x831a, "ASUS P901", ALC269_FIXUP_STEREO_DMIC),
5693 SND_PCI_QUIRK(0x1043, 0x834a, "ASUS S101", ALC269_FIXUP_STEREO_DMIC),
5694 SND_PCI_QUIRK(0x1043, 0x8398, "ASUS P1005", ALC269_FIXUP_STEREO_DMIC),
5695 SND_PCI_QUIRK(0x1043, 0x83ce, "ASUS P1005", ALC269_FIXUP_STEREO_DMIC),
David Henningssond240d1d2013-04-15 12:50:02 +02005696 SND_PCI_QUIRK(0x1043, 0x8516, "ASUS X101CH", ALC269_FIXUP_ASUS_X101),
Takashi Iwaif88abaa2014-02-07 12:07:59 +01005697 SND_PCI_QUIRK(0x104d, 0x90b5, "Sony VAIO Pro 11", ALC286_FIXUP_SONY_MIC_NO_PRESENCE),
David Henningsson88cfcf82013-10-11 10:18:45 +02005698 SND_PCI_QUIRK(0x104d, 0x90b6, "Sony VAIO Pro 13", ALC286_FIXUP_SONY_MIC_NO_PRESENCE),
Takashi Iwai1d045db2011-07-07 18:23:21 +02005699 SND_PCI_QUIRK(0x104d, 0x9073, "Sony VAIO", ALC275_FIXUP_SONY_VAIO_GPIO2),
5700 SND_PCI_QUIRK(0x104d, 0x907b, "Sony VAIO", ALC275_FIXUP_SONY_HWEQ),
5701 SND_PCI_QUIRK(0x104d, 0x9084, "Sony VAIO", ALC275_FIXUP_SONY_HWEQ),
Takashi Iwaie9bd7d52014-05-21 11:06:49 +02005702 SND_PCI_QUIRK(0x104d, 0x9099, "Sony VAIO S13", ALC275_FIXUP_SONY_DISABLE_AAMIX),
Takashi Iwai24519912011-08-16 15:08:49 +02005703 SND_PCI_QUIRK(0x10cf, 0x1475, "Lifebook", ALC269_FIXUP_LIFEBOOK),
Takashi Iwai4df3fd12015-06-29 08:38:02 +02005704 SND_PCI_QUIRK(0x10cf, 0x159f, "Lifebook E780", ALC269_FIXUP_LIFEBOOK_NO_HP_TO_LINEOUT),
Takashi Iwaicc7016a2015-04-08 20:47:55 +02005705 SND_PCI_QUIRK(0x10cf, 0x15dc, "Lifebook T731", ALC269_FIXUP_LIFEBOOK_HP_PIN),
Takashi Iwai88776f32015-05-01 09:20:34 +02005706 SND_PCI_QUIRK(0x10cf, 0x1757, "Lifebook E752", ALC269_FIXUP_LIFEBOOK_HP_PIN),
David Henningsson2041d562014-06-13 11:15:44 +02005707 SND_PCI_QUIRK(0x10cf, 0x1845, "Lifebook U904", ALC269_FIXUP_LIFEBOOK_EXTMIC),
David Henningssona33cc482014-10-07 10:18:41 +02005708 SND_PCI_QUIRK(0x144d, 0xc109, "Samsung Ativ book 9 (NP900X3G)", ALC269_FIXUP_INV_DMIC),
Anisse Astierabaa22742016-08-24 09:14:13 +02005709 SND_PCI_QUIRK(0x1458, 0xfa53, "Gigabyte BXBT-2807", ALC283_FIXUP_HEADSET_MIC),
5710 SND_PCI_QUIRK(0x1462, 0xb120, "MSI Cubi MS-B120", ALC283_FIXUP_HEADSET_MIC),
Takashi Iwai1d045db2011-07-07 18:23:21 +02005711 SND_PCI_QUIRK(0x17aa, 0x20f2, "Thinkpad SL410/510", ALC269_FIXUP_SKU_IGNORE),
5712 SND_PCI_QUIRK(0x17aa, 0x215e, "Thinkpad L512", ALC269_FIXUP_SKU_IGNORE),
5713 SND_PCI_QUIRK(0x17aa, 0x21b8, "Thinkpad Edge 14", ALC269_FIXUP_SKU_IGNORE),
5714 SND_PCI_QUIRK(0x17aa, 0x21ca, "Thinkpad L412", ALC269_FIXUP_SKU_IGNORE),
5715 SND_PCI_QUIRK(0x17aa, 0x21e9, "Thinkpad Edge 15", ALC269_FIXUP_SKU_IGNORE),
Takashi Iwai707fba32012-08-02 09:04:39 +02005716 SND_PCI_QUIRK(0x17aa, 0x21f6, "Thinkpad T530", ALC269_FIXUP_LENOVO_DOCK),
Felix Kaechelec8415a42012-08-06 23:02:01 +02005717 SND_PCI_QUIRK(0x17aa, 0x21fa, "Thinkpad X230", ALC269_FIXUP_LENOVO_DOCK),
Stefán Freyr84f98fd2012-10-19 22:46:00 +02005718 SND_PCI_QUIRK(0x17aa, 0x21f3, "Thinkpad T430", ALC269_FIXUP_LENOVO_DOCK),
Philipp A. Mohrenweiser4407be62012-08-06 13:14:18 +02005719 SND_PCI_QUIRK(0x17aa, 0x21fb, "Thinkpad T430s", ALC269_FIXUP_LENOVO_DOCK),
David Henningsson108cc102012-07-20 10:37:25 +02005720 SND_PCI_QUIRK(0x17aa, 0x2203, "Thinkpad X230 Tablet", ALC269_FIXUP_LENOVO_DOCK),
David Henningssonaaedfb42013-08-16 14:09:02 +02005721 SND_PCI_QUIRK(0x17aa, 0x2208, "Thinkpad T431s", ALC269_FIXUP_LENOVO_DOCK),
Takashi Iwai9a811232015-12-09 15:17:43 +01005722 SND_PCI_QUIRK(0x17aa, 0x220c, "Thinkpad T440s", ALC292_FIXUP_TPT440),
Takashi Iwai1c37c222014-05-06 17:34:42 +02005723 SND_PCI_QUIRK(0x17aa, 0x220e, "Thinkpad T440p", ALC292_FIXUP_TPT440_DOCK),
Takashi Iwaia12137e2014-06-27 12:14:35 +02005724 SND_PCI_QUIRK(0x17aa, 0x2210, "Thinkpad T540p", ALC292_FIXUP_TPT440_DOCK),
Rick Sherman59a51a62015-08-18 21:04:30 -05005725 SND_PCI_QUIRK(0x17aa, 0x2211, "Thinkpad W541", ALC292_FIXUP_TPT440_DOCK),
Takashi Iwai6d169412014-10-07 17:27:02 +02005726 SND_PCI_QUIRK(0x17aa, 0x2212, "Thinkpad T440", ALC292_FIXUP_TPT440_DOCK),
Lukas Bossard7c215392014-10-29 18:31:07 +01005727 SND_PCI_QUIRK(0x17aa, 0x2214, "Thinkpad X240", ALC292_FIXUP_TPT440_DOCK),
David Henningssona4a9e082013-08-16 14:09:01 +02005728 SND_PCI_QUIRK(0x17aa, 0x2215, "Thinkpad", ALC269_FIXUP_LIMIT_INT_MIC_BOOST),
Takashi Iwaib6903c02015-12-10 12:20:20 +01005729 SND_PCI_QUIRK(0x17aa, 0x2218, "Thinkpad X1 Carbon 2nd", ALC292_FIXUP_TPT440_DOCK),
Laura Abbottd05ea7d2015-10-02 11:09:54 -07005730 SND_PCI_QUIRK(0x17aa, 0x2223, "ThinkPad T550", ALC292_FIXUP_TPT440_DOCK),
Yves-Alexis Perezc0278662015-04-11 09:31:35 +02005731 SND_PCI_QUIRK(0x17aa, 0x2226, "ThinkPad X250", ALC292_FIXUP_TPT440_DOCK),
Torsten Hilbrichdab38e42016-06-07 13:14:21 +02005732 SND_PCI_QUIRK(0x17aa, 0x2231, "Thinkpad T560", ALC292_FIXUP_TPT460),
Sven Eckelmannc636b952016-04-11 16:55:26 +02005733 SND_PCI_QUIRK(0x17aa, 0x2233, "Thinkpad", ALC292_FIXUP_TPT460),
Kailang3694cb22015-12-28 11:35:24 +08005734 SND_PCI_QUIRK(0x17aa, 0x30bb, "ThinkCentre AIO", ALC233_FIXUP_LENOVO_LINE2_MIC_HOTKEY),
Hui Wang6ef2f682016-03-11 12:04:02 +08005735 SND_PCI_QUIRK(0x17aa, 0x30e2, "ThinkCentre AIO", ALC233_FIXUP_LENOVO_LINE2_MIC_HOTKEY),
Hui Wangacb06ff2017-02-27 10:11:47 +08005736 SND_PCI_QUIRK(0x17aa, 0x3112, "ThinkCentre AIO", ALC233_FIXUP_LENOVO_LINE2_MIC_HOTKEY),
David Henningsson56f27012016-01-11 09:33:14 +01005737 SND_PCI_QUIRK(0x17aa, 0x3902, "Lenovo E50-80", ALC269_FIXUP_DMIC_THINKPAD_ACPI),
Takashi Iwaifedb2242014-11-13 07:11:38 +01005738 SND_PCI_QUIRK(0x17aa, 0x3977, "IdeaPad S210", ALC283_FIXUP_INT_MIC),
Takashi Iwai9b745ab2014-03-07 08:37:19 +01005739 SND_PCI_QUIRK(0x17aa, 0x3978, "IdeaPad Y410P", ALC269_FIXUP_NO_SHUTUP),
David Henningssona4a9e082013-08-16 14:09:01 +02005740 SND_PCI_QUIRK(0x17aa, 0x5013, "Thinkpad", ALC269_FIXUP_LIMIT_INT_MIC_BOOST),
Kailang Yang1bb3e062013-09-27 13:10:25 +02005741 SND_PCI_QUIRK(0x17aa, 0x501a, "Thinkpad", ALC283_FIXUP_INT_MIC),
Takashi Iwaic497d9f2014-10-08 12:14:40 +02005742 SND_PCI_QUIRK(0x17aa, 0x501e, "Thinkpad L440", ALC292_FIXUP_TPT440_DOCK),
David Henningssoncd5302c2013-08-19 12:22:33 +02005743 SND_PCI_QUIRK(0x17aa, 0x5026, "Thinkpad", ALC269_FIXUP_LIMIT_INT_MIC_BOOST),
Jo-Philipp Wichf2aa1112015-04-13 12:47:26 +02005744 SND_PCI_QUIRK(0x17aa, 0x5034, "Thinkpad T450", ALC292_FIXUP_TPT440_DOCK),
Sebastian Wicki80b311d2015-03-23 17:23:11 +01005745 SND_PCI_QUIRK(0x17aa, 0x5036, "Thinkpad T450s", ALC292_FIXUP_TPT440_DOCK),
Ansgar Hegerfeld09ea9972015-05-14 12:31:32 +02005746 SND_PCI_QUIRK(0x17aa, 0x503c, "Thinkpad L450", ALC292_FIXUP_TPT440_DOCK),
Conrad Kostecki037e1192016-04-26 10:08:10 +02005747 SND_PCI_QUIRK(0x17aa, 0x504a, "ThinkPad X260", ALC292_FIXUP_TPT440_DOCK),
Hui Wang23adc192015-12-08 12:27:18 +08005748 SND_PCI_QUIRK(0x17aa, 0x504b, "Thinkpad", ALC293_FIXUP_LENOVO_SPK_NOISE),
Jaroslav Kysela0f087ee2016-06-24 15:13:16 +02005749 SND_PCI_QUIRK(0x17aa, 0x5050, "Thinkpad T560p", ALC292_FIXUP_TPT460),
Torsten Hilbrich9cd25742016-07-05 10:40:22 +02005750 SND_PCI_QUIRK(0x17aa, 0x5051, "Thinkpad L460", ALC292_FIXUP_TPT460),
Jaroslav Kysela0f087ee2016-06-24 15:13:16 +02005751 SND_PCI_QUIRK(0x17aa, 0x5053, "Thinkpad T460", ALC292_FIXUP_TPT460),
David Henningssoncd5302c2013-08-19 12:22:33 +02005752 SND_PCI_QUIRK(0x17aa, 0x5109, "Thinkpad", ALC269_FIXUP_LIMIT_INT_MIC_BOOST),
David Henningsson012e7eb2012-08-08 08:43:37 +02005753 SND_PCI_QUIRK(0x17aa, 0x3bf8, "Quanta FL1", ALC269_FIXUP_PCM_44K),
Takashi Iwai1d045db2011-07-07 18:23:21 +02005754 SND_PCI_QUIRK(0x17aa, 0x9e54, "LENOVO NB", ALC269_FIXUP_LENOVO_EAPD),
Anisse Astier02b504d2013-06-03 11:53:10 +02005755 SND_PCI_QUIRK(0x1b7d, 0xa831, "Ordissimo EVE2 ", ALC269VB_FIXUP_ORDISSIMO_EVE2), /* Also known as Malata PC-B1303 */
Takashi Iwaia4297b52011-08-23 18:40:12 +02005756
Takashi Iwaia7f3eed2012-02-16 13:03:18 +01005757#if 0
Takashi Iwaia4297b52011-08-23 18:40:12 +02005758 /* Below is a quirk table taken from the old code.
5759 * Basically the device should work as is without the fixup table.
5760 * If BIOS doesn't give a proper info, enable the corresponding
5761 * fixup entry.
Jesper Juhl7d7eb9e2012-04-12 22:11:25 +02005762 */
Takashi Iwaia4297b52011-08-23 18:40:12 +02005763 SND_PCI_QUIRK(0x1043, 0x8330, "ASUS Eeepc P703 P900A",
5764 ALC269_FIXUP_AMIC),
5765 SND_PCI_QUIRK(0x1043, 0x1013, "ASUS N61Da", ALC269_FIXUP_AMIC),
Takashi Iwaia4297b52011-08-23 18:40:12 +02005766 SND_PCI_QUIRK(0x1043, 0x1143, "ASUS B53f", ALC269_FIXUP_AMIC),
5767 SND_PCI_QUIRK(0x1043, 0x1133, "ASUS UJ20ft", ALC269_FIXUP_AMIC),
5768 SND_PCI_QUIRK(0x1043, 0x1183, "ASUS K72DR", ALC269_FIXUP_AMIC),
5769 SND_PCI_QUIRK(0x1043, 0x11b3, "ASUS K52DR", ALC269_FIXUP_AMIC),
5770 SND_PCI_QUIRK(0x1043, 0x11e3, "ASUS U33Jc", ALC269_FIXUP_AMIC),
5771 SND_PCI_QUIRK(0x1043, 0x1273, "ASUS UL80Jt", ALC269_FIXUP_AMIC),
5772 SND_PCI_QUIRK(0x1043, 0x1283, "ASUS U53Jc", ALC269_FIXUP_AMIC),
5773 SND_PCI_QUIRK(0x1043, 0x12b3, "ASUS N82JV", ALC269_FIXUP_AMIC),
5774 SND_PCI_QUIRK(0x1043, 0x12d3, "ASUS N61Jv", ALC269_FIXUP_AMIC),
5775 SND_PCI_QUIRK(0x1043, 0x13a3, "ASUS UL30Vt", ALC269_FIXUP_AMIC),
5776 SND_PCI_QUIRK(0x1043, 0x1373, "ASUS G73JX", ALC269_FIXUP_AMIC),
5777 SND_PCI_QUIRK(0x1043, 0x1383, "ASUS UJ30Jc", ALC269_FIXUP_AMIC),
5778 SND_PCI_QUIRK(0x1043, 0x13d3, "ASUS N61JA", ALC269_FIXUP_AMIC),
5779 SND_PCI_QUIRK(0x1043, 0x1413, "ASUS UL50", ALC269_FIXUP_AMIC),
5780 SND_PCI_QUIRK(0x1043, 0x1443, "ASUS UL30", ALC269_FIXUP_AMIC),
5781 SND_PCI_QUIRK(0x1043, 0x1453, "ASUS M60Jv", ALC269_FIXUP_AMIC),
5782 SND_PCI_QUIRK(0x1043, 0x1483, "ASUS UL80", ALC269_FIXUP_AMIC),
5783 SND_PCI_QUIRK(0x1043, 0x14f3, "ASUS F83Vf", ALC269_FIXUP_AMIC),
5784 SND_PCI_QUIRK(0x1043, 0x14e3, "ASUS UL20", ALC269_FIXUP_AMIC),
5785 SND_PCI_QUIRK(0x1043, 0x1513, "ASUS UX30", ALC269_FIXUP_AMIC),
5786 SND_PCI_QUIRK(0x1043, 0x1593, "ASUS N51Vn", ALC269_FIXUP_AMIC),
5787 SND_PCI_QUIRK(0x1043, 0x15a3, "ASUS N60Jv", ALC269_FIXUP_AMIC),
5788 SND_PCI_QUIRK(0x1043, 0x15b3, "ASUS N60Dp", ALC269_FIXUP_AMIC),
5789 SND_PCI_QUIRK(0x1043, 0x15c3, "ASUS N70De", ALC269_FIXUP_AMIC),
5790 SND_PCI_QUIRK(0x1043, 0x15e3, "ASUS F83T", ALC269_FIXUP_AMIC),
5791 SND_PCI_QUIRK(0x1043, 0x1643, "ASUS M60J", ALC269_FIXUP_AMIC),
5792 SND_PCI_QUIRK(0x1043, 0x1653, "ASUS U50", ALC269_FIXUP_AMIC),
5793 SND_PCI_QUIRK(0x1043, 0x1693, "ASUS F50N", ALC269_FIXUP_AMIC),
5794 SND_PCI_QUIRK(0x1043, 0x16a3, "ASUS F5Q", ALC269_FIXUP_AMIC),
5795 SND_PCI_QUIRK(0x1043, 0x1723, "ASUS P80", ALC269_FIXUP_AMIC),
5796 SND_PCI_QUIRK(0x1043, 0x1743, "ASUS U80", ALC269_FIXUP_AMIC),
5797 SND_PCI_QUIRK(0x1043, 0x1773, "ASUS U20A", ALC269_FIXUP_AMIC),
5798 SND_PCI_QUIRK(0x1043, 0x1883, "ASUS F81Se", ALC269_FIXUP_AMIC),
5799 SND_PCI_QUIRK(0x152d, 0x1778, "Quanta ON1", ALC269_FIXUP_DMIC),
5800 SND_PCI_QUIRK(0x17aa, 0x3be9, "Quanta Wistron", ALC269_FIXUP_AMIC),
5801 SND_PCI_QUIRK(0x17aa, 0x3bf8, "Quanta FL1", ALC269_FIXUP_AMIC),
5802 SND_PCI_QUIRK(0x17ff, 0x059a, "Quanta EL3", ALC269_FIXUP_DMIC),
5803 SND_PCI_QUIRK(0x17ff, 0x059b, "Quanta JR1", ALC269_FIXUP_DMIC),
5804#endif
5805 {}
5806};
5807
David Henningsson214eef72014-07-22 14:09:35 +02005808static const struct snd_pci_quirk alc269_fixup_vendor_tbl[] = {
5809 SND_PCI_QUIRK_VENDOR(0x1025, "Acer Aspire", ALC271_FIXUP_DMIC),
5810 SND_PCI_QUIRK_VENDOR(0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED),
5811 SND_PCI_QUIRK_VENDOR(0x104d, "Sony VAIO", ALC269_FIXUP_SONY_VAIO),
5812 SND_PCI_QUIRK_VENDOR(0x17aa, "Thinkpad", ALC269_FIXUP_THINKPAD_ACPI),
5813 {}
5814};
5815
Takashi Iwai1727a772013-01-10 09:52:52 +01005816static const struct hda_model_fixup alc269_fixup_models[] = {
Takashi Iwaia4297b52011-08-23 18:40:12 +02005817 {.id = ALC269_FIXUP_AMIC, .name = "laptop-amic"},
5818 {.id = ALC269_FIXUP_DMIC, .name = "laptop-dmic"},
Takashi Iwai6e72aa52012-06-25 10:52:25 +02005819 {.id = ALC269_FIXUP_STEREO_DMIC, .name = "alc269-dmic"},
5820 {.id = ALC271_FIXUP_DMIC, .name = "alc271-dmic"},
5821 {.id = ALC269_FIXUP_INV_DMIC, .name = "inv-dmic"},
David Henningsson7c478f02013-10-11 10:18:46 +02005822 {.id = ALC269_FIXUP_HEADSET_MIC, .name = "headset-mic"},
Takashi Iwaib0169512015-05-19 10:20:13 +02005823 {.id = ALC269_FIXUP_HEADSET_MODE, .name = "headset-mode"},
5824 {.id = ALC269_FIXUP_HEADSET_MODE_NO_HP_MIC, .name = "headset-mode-no-hp-mic"},
David Henningsson108cc102012-07-20 10:37:25 +02005825 {.id = ALC269_FIXUP_LENOVO_DOCK, .name = "lenovo-dock"},
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01005826 {.id = ALC269_FIXUP_HP_GPIO_LED, .name = "hp-gpio-led"},
Jaroslav Kysela52c33232017-03-09 13:29:13 +01005827 {.id = ALC269_FIXUP_HP_DOCK_GPIO_MIC1_LED, .name = "hp-dock-gpio-mic1-led"},
David Henningssone32aa852013-06-17 11:04:02 +02005828 {.id = ALC269_FIXUP_DELL1_MIC_NO_PRESENCE, .name = "dell-headset-multi"},
5829 {.id = ALC269_FIXUP_DELL2_MIC_NO_PRESENCE, .name = "dell-headset-dock"},
Kailang Yangbe8ef162014-04-08 15:19:49 +08005830 {.id = ALC283_FIXUP_CHROME_BOOK, .name = "alc283-dac-wcaps"},
Kailang Yang0202e992013-12-02 15:20:15 +08005831 {.id = ALC283_FIXUP_SENSE_COMBO_JACK, .name = "alc283-sense-combo"},
Takashi Iwai1c37c222014-05-06 17:34:42 +02005832 {.id = ALC292_FIXUP_TPT440_DOCK, .name = "tpt440-dock"},
Takashi Iwai9a811232015-12-09 15:17:43 +01005833 {.id = ALC292_FIXUP_TPT440, .name = "tpt440"},
Sven Eckelmannc636b952016-04-11 16:55:26 +02005834 {.id = ALC292_FIXUP_TPT460, .name = "tpt460"},
Takashi Iwai1d045db2011-07-07 18:23:21 +02005835 {}
5836};
Kailang Yangcfc5a842016-02-03 15:20:39 +08005837#define ALC225_STANDARD_PINS \
Kailang Yangcfc5a842016-02-03 15:20:39 +08005838 {0x21, 0x04211020}
Takashi Iwai1d045db2011-07-07 18:23:21 +02005839
Hui Wange8191a82015-04-24 13:39:59 +08005840#define ALC256_STANDARD_PINS \
5841 {0x12, 0x90a60140}, \
5842 {0x14, 0x90170110}, \
Hui Wange8191a82015-04-24 13:39:59 +08005843 {0x21, 0x02211020}
5844
David Henningssonfea185e2014-09-03 10:23:04 +02005845#define ALC282_STANDARD_PINS \
Hui Wang11580292015-08-03 11:03:49 +08005846 {0x14, 0x90170110}
Kailang Yange1e62b92015-04-08 16:01:22 +08005847
David Henningssonfea185e2014-09-03 10:23:04 +02005848#define ALC290_STANDARD_PINS \
Hui Wang11580292015-08-03 11:03:49 +08005849 {0x12, 0x99a30130}
David Henningssonfea185e2014-09-03 10:23:04 +02005850
5851#define ALC292_STANDARD_PINS \
5852 {0x14, 0x90170110}, \
Hui Wang11580292015-08-03 11:03:49 +08005853 {0x15, 0x0221401f}
Kailang Yang977e6272015-05-18 15:31:20 +08005854
Hui Wang3f6409702016-09-11 11:26:16 +08005855#define ALC295_STANDARD_PINS \
5856 {0x12, 0xb7a60130}, \
5857 {0x14, 0x90170110}, \
Hui Wang3f6409702016-09-11 11:26:16 +08005858 {0x21, 0x04211020}
5859
Woodrow Shen703867e2015-08-05 12:34:12 +08005860#define ALC298_STANDARD_PINS \
5861 {0x12, 0x90a60130}, \
5862 {0x21, 0x03211020}
5863
Hui Wange1918932014-05-26 16:22:44 +08005864static const struct snd_hda_pin_quirk alc269_pin_fixup_tbl[] = {
David Henningsson2ae95572016-02-25 09:37:05 +01005865 SND_HDA_PIN_QUIRK(0x10ec0225, 0x1028, "Dell", ALC225_FIXUP_DELL1_MIC_NO_PRESENCE,
Kailang Yangcfc5a842016-02-03 15:20:39 +08005866 ALC225_STANDARD_PINS,
Hui Wang8a132092016-07-08 14:26:57 +08005867 {0x12, 0xb7a60130},
Kailang Yangcfc5a842016-02-03 15:20:39 +08005868 {0x14, 0x901701a0}),
David Henningsson2ae95572016-02-25 09:37:05 +01005869 SND_HDA_PIN_QUIRK(0x10ec0225, 0x1028, "Dell", ALC225_FIXUP_DELL1_MIC_NO_PRESENCE,
Kailang Yangcfc5a842016-02-03 15:20:39 +08005870 ALC225_STANDARD_PINS,
Hui Wang8a132092016-07-08 14:26:57 +08005871 {0x12, 0xb7a60130},
Kailang Yangcfc5a842016-02-03 15:20:39 +08005872 {0x14, 0x901701b0}),
Hui Wang8a132092016-07-08 14:26:57 +08005873 SND_HDA_PIN_QUIRK(0x10ec0225, 0x1028, "Dell", ALC225_FIXUP_DELL1_MIC_NO_PRESENCE,
5874 ALC225_STANDARD_PINS,
5875 {0x12, 0xb7a60150},
5876 {0x14, 0x901701a0}),
5877 SND_HDA_PIN_QUIRK(0x10ec0225, 0x1028, "Dell", ALC225_FIXUP_DELL1_MIC_NO_PRESENCE,
5878 ALC225_STANDARD_PINS,
5879 {0x12, 0xb7a60150},
5880 {0x14, 0x901701b0}),
5881 SND_HDA_PIN_QUIRK(0x10ec0225, 0x1028, "Dell", ALC225_FIXUP_DELL1_MIC_NO_PRESENCE,
5882 ALC225_STANDARD_PINS,
5883 {0x12, 0xb7a60130},
5884 {0x1b, 0x90170110}),
Hui Wang41f804d2017-10-24 16:53:34 +08005885 SND_HDA_PIN_QUIRK(0x10ec0236, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
5886 {0x12, 0x90a60140},
5887 {0x14, 0x90170110},
5888 {0x21, 0x02211020}),
5889 SND_HDA_PIN_QUIRK(0x10ec0236, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
5890 {0x12, 0x90a60140},
5891 {0x14, 0x90170150},
5892 {0x21, 0x02211020}),
Hui Wangc77900e2014-09-03 11:31:07 +08005893 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL2_MIC_NO_PRESENCE,
Hui Wangc77900e2014-09-03 11:31:07 +08005894 {0x14, 0x90170110},
Hui Wangc77900e2014-09-03 11:31:07 +08005895 {0x21, 0x02211020}),
David Henningsson76c21322014-06-24 14:46:54 +02005896 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wang86c72d12016-05-25 12:12:32 +08005897 {0x14, 0x90170130},
5898 {0x21, 0x02211040}),
5899 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
David Henningsson76c21322014-06-24 14:46:54 +02005900 {0x12, 0x90a60140},
5901 {0x14, 0x90170110},
David Henningsson76c21322014-06-24 14:46:54 +02005902 {0x21, 0x02211020}),
5903 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
5904 {0x12, 0x90a60160},
5905 {0x14, 0x90170120},
David Henningsson76c21322014-06-24 14:46:54 +02005906 {0x21, 0x02211030}),
5907 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wang392c9da2016-09-26 10:59:38 +08005908 {0x14, 0x90170110},
5909 {0x1b, 0x02011020},
5910 {0x21, 0x0221101f}),
5911 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wang6aecd872016-10-20 14:03:33 +08005912 {0x14, 0x90170110},
5913 {0x1b, 0x01011020},
5914 {0x21, 0x0221101f}),
5915 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
David Henningssoncba59972015-07-22 10:00:25 +02005916 {0x14, 0x90170130},
David Henningssoncba59972015-07-22 10:00:25 +02005917 {0x1b, 0x01014020},
David Henningssoncba59972015-07-22 10:00:25 +02005918 {0x21, 0x0221103f}),
5919 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wang59ec4b52016-08-04 15:28:04 +08005920 {0x14, 0x90170130},
Hui Wang6aecd872016-10-20 14:03:33 +08005921 {0x1b, 0x01011020},
5922 {0x21, 0x0221103f}),
5923 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
5924 {0x14, 0x90170130},
Hui Wang59ec4b52016-08-04 15:28:04 +08005925 {0x1b, 0x02011020},
5926 {0x21, 0x0221103f}),
5927 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Woodrow Shene9c28e12015-07-25 10:36:16 +08005928 {0x14, 0x90170150},
Woodrow Shene9c28e12015-07-25 10:36:16 +08005929 {0x1b, 0x02011020},
Woodrow Shene9c28e12015-07-25 10:36:16 +08005930 {0x21, 0x0221105f}),
5931 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Woodrow Shene9c28e12015-07-25 10:36:16 +08005932 {0x14, 0x90170110},
Woodrow Shene9c28e12015-07-25 10:36:16 +08005933 {0x1b, 0x01014020},
Woodrow Shene9c28e12015-07-25 10:36:16 +08005934 {0x21, 0x0221101f}),
5935 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
David Henningsson76c21322014-06-24 14:46:54 +02005936 {0x12, 0x90a60160},
5937 {0x14, 0x90170120},
5938 {0x17, 0x90170140},
David Henningsson76c21322014-06-24 14:46:54 +02005939 {0x21, 0x0321102f}),
5940 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
5941 {0x12, 0x90a60160},
5942 {0x14, 0x90170130},
David Henningsson76c21322014-06-24 14:46:54 +02005943 {0x21, 0x02211040}),
5944 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
5945 {0x12, 0x90a60160},
5946 {0x14, 0x90170140},
David Henningsson76c21322014-06-24 14:46:54 +02005947 {0x21, 0x02211050}),
5948 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
5949 {0x12, 0x90a60170},
5950 {0x14, 0x90170120},
David Henningsson76c21322014-06-24 14:46:54 +02005951 {0x21, 0x02211030}),
5952 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
5953 {0x12, 0x90a60170},
5954 {0x14, 0x90170130},
David Henningsson76c21322014-06-24 14:46:54 +02005955 {0x21, 0x02211040}),
Hui Wang70658b92015-03-06 14:03:57 +08005956 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wang70658b92015-03-06 14:03:57 +08005957 {0x12, 0x90a60170},
Hui Wang0a1f90a2016-01-13 11:51:38 +08005958 {0x14, 0x90171130},
5959 {0x21, 0x02211040}),
5960 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
5961 {0x12, 0x90a60170},
Hui Wang70658b92015-03-06 14:03:57 +08005962 {0x14, 0x90170140},
Hui Wang70658b92015-03-06 14:03:57 +08005963 {0x21, 0x02211050}),
David Henningsson9b5a4e32015-05-11 14:04:14 +02005964 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell Inspiron 5548", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
David Henningsson9b5a4e32015-05-11 14:04:14 +02005965 {0x12, 0x90a60180},
5966 {0x14, 0x90170130},
David Henningsson9b5a4e32015-05-11 14:04:14 +02005967 {0x21, 0x02211040}),
AceLan Kaof90d83b2016-06-03 14:45:25 +08005968 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell Inspiron 5565", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
5969 {0x12, 0x90a60180},
5970 {0x14, 0x90170120},
5971 {0x21, 0x02211030}),
Hui Wang0119d5d2016-11-23 16:05:38 +08005972 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
5973 {0x1b, 0x01011020},
5974 {0x21, 0x02211010}),
Kailang Yang7081adf2015-03-30 17:05:37 +08005975 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wang65ca46e2017-12-22 11:17:45 +08005976 {0x12, 0x90a60130},
5977 {0x14, 0x90170110},
5978 {0x1b, 0x01011020},
5979 {0x21, 0x0221101f}),
5980 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Kailang Yang81a12312015-05-28 16:48:22 +08005981 {0x12, 0x90a60160},
5982 {0x14, 0x90170120},
Kailang Yang81a12312015-05-28 16:48:22 +08005983 {0x21, 0x02211030}),
5984 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Woodrow Shenf83c3292016-06-24 15:58:34 +08005985 {0x12, 0x90a60170},
5986 {0x14, 0x90170120},
5987 {0x21, 0x02211030}),
Shrirang Bagul311042d2016-08-29 15:19:27 +08005988 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1028, "Dell Inspiron 5468", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
5989 {0x12, 0x90a60180},
5990 {0x14, 0x90170120},
5991 {0x21, 0x02211030}),
Woodrow Shenf83c3292016-06-24 15:58:34 +08005992 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wang3f6409702016-09-11 11:26:16 +08005993 {0x12, 0xb7a60130},
5994 {0x14, 0x90170110},
5995 {0x21, 0x02211020}),
David Henningssoncf51eb92014-10-30 08:26:02 +01005996 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
5997 ALC256_STANDARD_PINS),
Hui Wang11580292015-08-03 11:03:49 +08005998 SND_HDA_PIN_QUIRK(0x10ec0280, 0x103c, "HP", ALC280_FIXUP_HP_GPIO4,
Hui Wang02796612014-09-03 11:31:11 +08005999 {0x12, 0x90a60130},
6000 {0x14, 0x90170110},
Hui Wang02796612014-09-03 11:31:11 +08006001 {0x15, 0x0421101f},
6002 {0x1a, 0x04a11020}),
Hui Wang02796612014-09-03 11:31:11 +08006003 SND_HDA_PIN_QUIRK(0x10ec0280, 0x103c, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED,
Hui Wang02796612014-09-03 11:31:11 +08006004 {0x12, 0x90a60140},
Hui Wang11580292015-08-03 11:03:49 +08006005 {0x14, 0x90170110},
David Henningsson42304472014-07-22 11:42:17 +02006006 {0x15, 0x0421101f},
David Henningssonaec856d2014-09-03 10:23:05 +02006007 {0x18, 0x02811030},
David Henningsson42304472014-07-22 11:42:17 +02006008 {0x1a, 0x04a1103f},
David Henningsson42304472014-07-22 11:42:17 +02006009 {0x1b, 0x02011020}),
David Henningsson42304472014-07-22 11:42:17 +02006010 SND_HDA_PIN_QUIRK(0x10ec0282, 0x103c, "HP 15 Touchsmart", ALC269_FIXUP_HP_MUTE_LED_MIC1,
Hui Wang2c609992014-09-03 11:31:08 +08006011 ALC282_STANDARD_PINS,
David Henningssonaec856d2014-09-03 10:23:05 +02006012 {0x12, 0x99a30130},
Hui Wang2c609992014-09-03 11:31:08 +08006013 {0x19, 0x03a11020},
Hui Wang2c609992014-09-03 11:31:08 +08006014 {0x21, 0x0321101f}),
Hui Wang2c609992014-09-03 11:31:08 +08006015 SND_HDA_PIN_QUIRK(0x10ec0282, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
6016 ALC282_STANDARD_PINS,
David Henningssonaec856d2014-09-03 10:23:05 +02006017 {0x12, 0x99a30130},
Hui Wang2c609992014-09-03 11:31:08 +08006018 {0x19, 0x03a11020},
Hui Wang2c609992014-09-03 11:31:08 +08006019 {0x21, 0x03211040}),
Hui Wang2c609992014-09-03 11:31:08 +08006020 SND_HDA_PIN_QUIRK(0x10ec0282, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
6021 ALC282_STANDARD_PINS,
David Henningssonaec856d2014-09-03 10:23:05 +02006022 {0x12, 0x99a30130},
Hui Wang2c609992014-09-03 11:31:08 +08006023 {0x19, 0x03a11030},
Hui Wang2c609992014-09-03 11:31:08 +08006024 {0x21, 0x03211020}),
Hui Wang2c609992014-09-03 11:31:08 +08006025 SND_HDA_PIN_QUIRK(0x10ec0282, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
Hui Wang200afc02014-09-03 11:31:10 +08006026 ALC282_STANDARD_PINS,
David Henningssonaec856d2014-09-03 10:23:05 +02006027 {0x12, 0x99a30130},
Hui Wang200afc02014-09-03 11:31:10 +08006028 {0x19, 0x04a11020},
Hui Wang200afc02014-09-03 11:31:10 +08006029 {0x21, 0x0421101f}),
Hui Wang200afc02014-09-03 11:31:10 +08006030 SND_HDA_PIN_QUIRK(0x10ec0282, 0x103c, "HP", ALC269_FIXUP_HP_LINE1_MIC1_LED,
David Henningsson76c21322014-06-24 14:46:54 +02006031 ALC282_STANDARD_PINS,
David Henningssonaec856d2014-09-03 10:23:05 +02006032 {0x12, 0x90a60140},
David Henningsson76c21322014-06-24 14:46:54 +02006033 {0x19, 0x04a11030},
David Henningsson76c21322014-06-24 14:46:54 +02006034 {0x21, 0x04211020}),
6035 SND_HDA_PIN_QUIRK(0x10ec0283, 0x1028, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE,
6036 ALC282_STANDARD_PINS,
6037 {0x12, 0x90a60130},
David Henningsson76c21322014-06-24 14:46:54 +02006038 {0x21, 0x0321101f}),
Hui Wangbc262172014-09-03 11:31:05 +08006039 SND_HDA_PIN_QUIRK(0x10ec0283, 0x1028, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE,
David Henningssonaec856d2014-09-03 10:23:05 +02006040 {0x12, 0x90a60160},
Hui Wangbc262172014-09-03 11:31:05 +08006041 {0x14, 0x90170120},
Hui Wangbc262172014-09-03 11:31:05 +08006042 {0x21, 0x02211030}),
Hui Wangbc262172014-09-03 11:31:05 +08006043 SND_HDA_PIN_QUIRK(0x10ec0283, 0x1028, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE,
Kailang Yange1e62b92015-04-08 16:01:22 +08006044 ALC282_STANDARD_PINS,
Kailang Yange1e62b92015-04-08 16:01:22 +08006045 {0x12, 0x90a60130},
Kailang Yange1e62b92015-04-08 16:01:22 +08006046 {0x19, 0x03a11020},
Kailang Yange1e62b92015-04-08 16:01:22 +08006047 {0x21, 0x0321101f}),
Hui Wange4442bc2014-09-03 11:31:09 +08006048 SND_HDA_PIN_QUIRK(0x10ec0288, 0x1028, "Dell", ALC288_FIXUP_DELL_XPS_13_GPIO6,
David Henningssonaec856d2014-09-03 10:23:05 +02006049 {0x12, 0x90a60120},
Hui Wange4442bc2014-09-03 11:31:09 +08006050 {0x14, 0x90170110},
Hui Wange4442bc2014-09-03 11:31:09 +08006051 {0x21, 0x0321101f}),
Hui Wang11580292015-08-03 11:03:49 +08006052 SND_HDA_PIN_QUIRK(0x10ec0290, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
Hui Wange4442bc2014-09-03 11:31:09 +08006053 ALC290_STANDARD_PINS,
David Henningssonaec856d2014-09-03 10:23:05 +02006054 {0x15, 0x04211040},
Hui Wange4442bc2014-09-03 11:31:09 +08006055 {0x18, 0x90170112},
Hui Wange4442bc2014-09-03 11:31:09 +08006056 {0x1a, 0x04a11020}),
Hui Wang11580292015-08-03 11:03:49 +08006057 SND_HDA_PIN_QUIRK(0x10ec0290, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
Hui Wange4442bc2014-09-03 11:31:09 +08006058 ALC290_STANDARD_PINS,
David Henningssonaec856d2014-09-03 10:23:05 +02006059 {0x15, 0x04211040},
Hui Wange4442bc2014-09-03 11:31:09 +08006060 {0x18, 0x90170110},
Hui Wang11580292015-08-03 11:03:49 +08006061 {0x1a, 0x04a11020}),
Hui Wange4442bc2014-09-03 11:31:09 +08006062 SND_HDA_PIN_QUIRK(0x10ec0290, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
David Henningssonaec856d2014-09-03 10:23:05 +02006063 ALC290_STANDARD_PINS,
Hui Wange4442bc2014-09-03 11:31:09 +08006064 {0x15, 0x0421101f},
Hui Wang11580292015-08-03 11:03:49 +08006065 {0x1a, 0x04a11020}),
Hui Wange4442bc2014-09-03 11:31:09 +08006066 SND_HDA_PIN_QUIRK(0x10ec0290, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
David Henningssonaec856d2014-09-03 10:23:05 +02006067 ALC290_STANDARD_PINS,
Hui Wange4442bc2014-09-03 11:31:09 +08006068 {0x15, 0x04211020},
6069 {0x1a, 0x04a11040}),
Hui Wang11580292015-08-03 11:03:49 +08006070 SND_HDA_PIN_QUIRK(0x10ec0290, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
Hui Wange4442bc2014-09-03 11:31:09 +08006071 ALC290_STANDARD_PINS,
David Henningssonaec856d2014-09-03 10:23:05 +02006072 {0x14, 0x90170110},
Hui Wange4442bc2014-09-03 11:31:09 +08006073 {0x15, 0x04211020},
6074 {0x1a, 0x04a11040}),
Hui Wang11580292015-08-03 11:03:49 +08006075 SND_HDA_PIN_QUIRK(0x10ec0290, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
Hui Wange4442bc2014-09-03 11:31:09 +08006076 ALC290_STANDARD_PINS,
David Henningssonaec856d2014-09-03 10:23:05 +02006077 {0x14, 0x90170110},
Hui Wange4442bc2014-09-03 11:31:09 +08006078 {0x15, 0x04211020},
6079 {0x1a, 0x04a11020}),
Hui Wang11580292015-08-03 11:03:49 +08006080 SND_HDA_PIN_QUIRK(0x10ec0290, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
Hui Wange8818fa2014-09-03 11:31:04 +08006081 ALC290_STANDARD_PINS,
David Henningssonaec856d2014-09-03 10:23:05 +02006082 {0x14, 0x90170110},
Hui Wange8818fa2014-09-03 11:31:04 +08006083 {0x15, 0x0421101f},
Hui Wange8818fa2014-09-03 11:31:04 +08006084 {0x1a, 0x04a11020}),
Hui Wang11580292015-08-03 11:03:49 +08006085 SND_HDA_PIN_QUIRK(0x10ec0292, 0x1028, "Dell", ALC269_FIXUP_DELL2_MIC_NO_PRESENCE,
Hui Wange8818fa2014-09-03 11:31:04 +08006086 ALC292_STANDARD_PINS,
David Henningssonaec856d2014-09-03 10:23:05 +02006087 {0x12, 0x90a60140},
Hui Wange8818fa2014-09-03 11:31:04 +08006088 {0x16, 0x01014020},
Hui Wange8818fa2014-09-03 11:31:04 +08006089 {0x19, 0x01a19030}),
6090 SND_HDA_PIN_QUIRK(0x10ec0292, 0x1028, "Dell", ALC269_FIXUP_DELL2_MIC_NO_PRESENCE,
Hui Wang11580292015-08-03 11:03:49 +08006091 ALC292_STANDARD_PINS,
David Henningsson76c21322014-06-24 14:46:54 +02006092 {0x12, 0x90a60140},
David Henningssonaec856d2014-09-03 10:23:05 +02006093 {0x16, 0x01014020},
Hui Wang11580292015-08-03 11:03:49 +08006094 {0x18, 0x02a19031},
David Henningsson76c21322014-06-24 14:46:54 +02006095 {0x19, 0x01a1903e}),
David Henningssonaec856d2014-09-03 10:23:05 +02006096 SND_HDA_PIN_QUIRK(0x10ec0292, 0x1028, "Dell", ALC269_FIXUP_DELL3_MIC_NO_PRESENCE,
David Henningsson76c21322014-06-24 14:46:54 +02006097 ALC292_STANDARD_PINS,
David Henningsson76c21322014-06-24 14:46:54 +02006098 {0x12, 0x90a60140}),
Hui Wang11580292015-08-03 11:03:49 +08006099 SND_HDA_PIN_QUIRK(0x10ec0293, 0x1028, "Dell", ALC293_FIXUP_DELL1_MIC_NO_PRESENCE,
David Henningssone03fdbd2014-06-27 08:41:59 +02006100 ALC292_STANDARD_PINS,
David Henningssonaec856d2014-09-03 10:23:05 +02006101 {0x13, 0x90a60140},
Hui Wang11580292015-08-03 11:03:49 +08006102 {0x16, 0x21014020},
Kailang Yang977e6272015-05-18 15:31:20 +08006103 {0x19, 0x21a19030}),
Woodrow Shen703867e2015-08-05 12:34:12 +08006104 SND_HDA_PIN_QUIRK(0x10ec0293, 0x1028, "Dell", ALC293_FIXUP_DELL1_MIC_NO_PRESENCE,
Takashi Iwai9f502ff2015-08-05 12:09:45 +02006105 ALC292_STANDARD_PINS,
6106 {0x13, 0x90a60140}),
Hui Wang3f6409702016-09-11 11:26:16 +08006107 SND_HDA_PIN_QUIRK(0x10ec0295, 0x1028, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wangf771d5b2016-10-18 10:59:09 +08006108 ALC295_STANDARD_PINS,
6109 {0x17, 0x21014020},
6110 {0x18, 0x21a19030}),
6111 SND_HDA_PIN_QUIRK(0x10ec0295, 0x1028, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE,
6112 ALC295_STANDARD_PINS,
6113 {0x17, 0x21014040},
6114 {0x18, 0x21a19050}),
Hui Wang0fe87712017-03-23 10:00:25 +08006115 SND_HDA_PIN_QUIRK(0x10ec0295, 0x1028, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE,
6116 ALC295_STANDARD_PINS),
Kailang Yang977e6272015-05-18 15:31:20 +08006117 SND_HDA_PIN_QUIRK(0x10ec0298, 0x1028, "Dell", ALC298_FIXUP_DELL1_MIC_NO_PRESENCE,
Woodrow Shen703867e2015-08-05 12:34:12 +08006118 ALC298_STANDARD_PINS,
Takashi Iwai9f502ff2015-08-05 12:09:45 +02006119 {0x17, 0x90170110}),
6120 SND_HDA_PIN_QUIRK(0x10ec0298, 0x1028, "Dell", ALC298_FIXUP_DELL1_MIC_NO_PRESENCE,
6121 ALC298_STANDARD_PINS,
Woodrow Shen703867e2015-08-05 12:34:12 +08006122 {0x17, 0x90170140}),
6123 SND_HDA_PIN_QUIRK(0x10ec0298, 0x1028, "Dell", ALC298_FIXUP_DELL1_MIC_NO_PRESENCE,
6124 ALC298_STANDARD_PINS,
Takashi Iwai9f502ff2015-08-05 12:09:45 +02006125 {0x17, 0x90170150}),
Kai-Heng Fengebc3e952017-02-16 15:26:54 +08006126 SND_HDA_PIN_QUIRK(0x10ec0298, 0x1028, "Dell", ALC298_FIXUP_SPK_VOLUME,
6127 {0x12, 0xb7a60140},
6128 {0x13, 0xb7a60150},
6129 {0x17, 0x90170110},
6130 {0x1a, 0x03011020},
6131 {0x21, 0x03211030}),
Hui Wange1918932014-05-26 16:22:44 +08006132 {}
6133};
Takashi Iwai1d045db2011-07-07 18:23:21 +02006134
Takashi Iwai546bb672012-03-07 08:37:19 +01006135static void alc269_fill_coef(struct hda_codec *codec)
Takashi Iwai1d045db2011-07-07 18:23:21 +02006136{
Kailang Yang526af6e2012-03-07 08:25:20 +01006137 struct alc_spec *spec = codec->spec;
Takashi Iwai1d045db2011-07-07 18:23:21 +02006138 int val;
6139
Kailang Yang526af6e2012-03-07 08:25:20 +01006140 if (spec->codec_variant != ALC269_TYPE_ALC269VB)
Takashi Iwai546bb672012-03-07 08:37:19 +01006141 return;
Kailang Yang526af6e2012-03-07 08:25:20 +01006142
Takashi Iwai1bb7e432011-10-17 16:50:59 +02006143 if ((alc_get_coef0(codec) & 0x00ff) < 0x015) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02006144 alc_write_coef_idx(codec, 0xf, 0x960b);
6145 alc_write_coef_idx(codec, 0xe, 0x8817);
6146 }
6147
Takashi Iwai1bb7e432011-10-17 16:50:59 +02006148 if ((alc_get_coef0(codec) & 0x00ff) == 0x016) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02006149 alc_write_coef_idx(codec, 0xf, 0x960b);
6150 alc_write_coef_idx(codec, 0xe, 0x8814);
6151 }
6152
Takashi Iwai1bb7e432011-10-17 16:50:59 +02006153 if ((alc_get_coef0(codec) & 0x00ff) == 0x017) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02006154 /* Power up output pin */
Takashi Iwai98b24882014-08-18 13:47:50 +02006155 alc_update_coef_idx(codec, 0x04, 0, 1<<11);
Takashi Iwai1d045db2011-07-07 18:23:21 +02006156 }
6157
Takashi Iwai1bb7e432011-10-17 16:50:59 +02006158 if ((alc_get_coef0(codec) & 0x00ff) == 0x018) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02006159 val = alc_read_coef_idx(codec, 0xd);
Takashi Iwaif3ee07d2014-08-15 17:35:00 +02006160 if (val != -1 && (val & 0x0c00) >> 10 != 0x1) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02006161 /* Capless ramp up clock control */
6162 alc_write_coef_idx(codec, 0xd, val | (1<<10));
6163 }
6164 val = alc_read_coef_idx(codec, 0x17);
Takashi Iwaif3ee07d2014-08-15 17:35:00 +02006165 if (val != -1 && (val & 0x01c0) >> 6 != 0x4) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02006166 /* Class D power on reset */
6167 alc_write_coef_idx(codec, 0x17, val | (1<<7));
6168 }
6169 }
6170
Takashi Iwai98b24882014-08-18 13:47:50 +02006171 /* HP */
6172 alc_update_coef_idx(codec, 0x4, 0, 1<<11);
Takashi Iwai1d045db2011-07-07 18:23:21 +02006173}
6174
6175/*
6176 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02006177static int patch_alc269(struct hda_codec *codec)
6178{
6179 struct alc_spec *spec;
Takashi Iwai3de95172012-05-07 18:03:15 +02006180 int err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02006181
Takashi Iwai3de95172012-05-07 18:03:15 +02006182 err = alc_alloc_spec(codec, 0x0b);
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02006183 if (err < 0)
Takashi Iwai3de95172012-05-07 18:03:15 +02006184 return err;
6185
6186 spec = codec->spec;
Takashi Iwai08c189f2012-12-19 15:22:24 +01006187 spec->gen.shared_mic_vref_pin = 0x18;
Takashi Iwai8b99aba2015-06-15 11:59:32 +02006188 codec->power_save_node = 1;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02006189
Takashi Iwai225068a2015-05-29 10:42:14 +02006190#ifdef CONFIG_PM
6191 codec->patch_ops.suspend = alc269_suspend;
6192 codec->patch_ops.resume = alc269_resume;
6193#endif
6194 spec->shutup = alc269_shutup;
6195
Takashi Iwai1727a772013-01-10 09:52:52 +01006196 snd_hda_pick_fixup(codec, alc269_fixup_models,
Herton Ronaldo Krzesinski9f720bb2012-09-27 10:38:14 -03006197 alc269_fixup_tbl, alc269_fixups);
Hui Wange1918932014-05-26 16:22:44 +08006198 snd_hda_pick_pin_fixup(codec, alc269_pin_fixup_tbl, alc269_fixups);
David Henningsson214eef72014-07-22 14:09:35 +02006199 snd_hda_pick_fixup(codec, NULL, alc269_fixup_vendor_tbl,
6200 alc269_fixups);
Takashi Iwai1727a772013-01-10 09:52:52 +01006201 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
Herton Ronaldo Krzesinski9f720bb2012-09-27 10:38:14 -03006202
6203 alc_auto_parse_customize_define(codec);
6204
Takashi Iwai7504b6c2013-03-18 11:25:51 +01006205 if (has_cdefine_beep(codec))
6206 spec->gen.beep_nid = 0x01;
6207
Takashi Iwai7639a062015-03-03 10:07:24 +01006208 switch (codec->core.vendor_id) {
Kailang Yang065380f2013-01-10 10:25:48 +01006209 case 0x10ec0269:
Takashi Iwai1d045db2011-07-07 18:23:21 +02006210 spec->codec_variant = ALC269_TYPE_ALC269VA;
Takashi Iwai1bb7e432011-10-17 16:50:59 +02006211 switch (alc_get_coef0(codec) & 0x00f0) {
6212 case 0x0010:
Takashi Iwai5100cd02014-02-15 10:03:19 +01006213 if (codec->bus->pci &&
6214 codec->bus->pci->subsystem_vendor == 0x1025 &&
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02006215 spec->cdefine.platform_type == 1)
Takashi Iwai20ca0c32011-10-17 16:00:35 +02006216 err = alc_codec_rename(codec, "ALC271X");
Takashi Iwai1d045db2011-07-07 18:23:21 +02006217 spec->codec_variant = ALC269_TYPE_ALC269VB;
Takashi Iwai1bb7e432011-10-17 16:50:59 +02006218 break;
6219 case 0x0020:
Takashi Iwai5100cd02014-02-15 10:03:19 +01006220 if (codec->bus->pci &&
6221 codec->bus->pci->subsystem_vendor == 0x17aa &&
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02006222 codec->bus->pci->subsystem_device == 0x21f3)
Takashi Iwai20ca0c32011-10-17 16:00:35 +02006223 err = alc_codec_rename(codec, "ALC3202");
Takashi Iwai1d045db2011-07-07 18:23:21 +02006224 spec->codec_variant = ALC269_TYPE_ALC269VC;
Takashi Iwai1bb7e432011-10-17 16:50:59 +02006225 break;
Kailang Yangadcc70b2012-05-25 08:08:38 +02006226 case 0x0030:
6227 spec->codec_variant = ALC269_TYPE_ALC269VD;
6228 break;
Takashi Iwai1bb7e432011-10-17 16:50:59 +02006229 default:
Takashi Iwai1d045db2011-07-07 18:23:21 +02006230 alc_fix_pll_init(codec, 0x20, 0x04, 15);
Takashi Iwai1bb7e432011-10-17 16:50:59 +02006231 }
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02006232 if (err < 0)
6233 goto error;
Takashi Iwai546bb672012-03-07 08:37:19 +01006234 spec->init_hook = alc269_fill_coef;
Takashi Iwai1d045db2011-07-07 18:23:21 +02006235 alc269_fill_coef(codec);
Kailang Yang065380f2013-01-10 10:25:48 +01006236 break;
6237
6238 case 0x10ec0280:
6239 case 0x10ec0290:
6240 spec->codec_variant = ALC269_TYPE_ALC280;
6241 break;
6242 case 0x10ec0282:
Kailang Yang065380f2013-01-10 10:25:48 +01006243 spec->codec_variant = ALC269_TYPE_ALC282;
Kailang Yang7b5c7a02014-03-18 16:15:54 +08006244 spec->shutup = alc282_shutup;
6245 spec->init_hook = alc282_init;
Kailang Yang065380f2013-01-10 10:25:48 +01006246 break;
Kailang Yang2af02be2013-08-22 10:03:50 +02006247 case 0x10ec0233:
6248 case 0x10ec0283:
6249 spec->codec_variant = ALC269_TYPE_ALC283;
6250 spec->shutup = alc283_shutup;
6251 spec->init_hook = alc283_init;
6252 break;
Kailang Yang065380f2013-01-10 10:25:48 +01006253 case 0x10ec0284:
6254 case 0x10ec0292:
6255 spec->codec_variant = ALC269_TYPE_ALC284;
6256 break;
Kailang Yang161ebf22013-10-24 11:36:33 +02006257 case 0x10ec0285:
6258 case 0x10ec0293:
6259 spec->codec_variant = ALC269_TYPE_ALC285;
6260 break;
Kailang Yang7fc7d042013-04-25 11:04:43 +02006261 case 0x10ec0286:
Kailang Yang7c665932014-04-14 15:09:44 +08006262 case 0x10ec0288:
Kailang Yang7fc7d042013-04-25 11:04:43 +02006263 spec->codec_variant = ALC269_TYPE_ALC286;
Kailang Yangf7ae9ba2014-06-30 16:10:37 +08006264 spec->shutup = alc286_shutup;
Kailang Yang7fc7d042013-04-25 11:04:43 +02006265 break;
Kailang Yang506b62c2014-12-18 17:07:44 +08006266 case 0x10ec0298:
6267 spec->codec_variant = ALC269_TYPE_ALC298;
6268 break;
Kailang Yang1d04c9d2013-10-24 11:35:18 +02006269 case 0x10ec0255:
6270 spec->codec_variant = ALC269_TYPE_ALC255;
6271 break;
Kailang Yang61ae3fb2017-10-20 15:06:34 +08006272 case 0x10ec0236:
Kailang Yang4344aec2014-12-17 17:39:05 +08006273 case 0x10ec0256:
6274 spec->codec_variant = ALC269_TYPE_ALC256;
David Henningsson7d1b6e22015-04-21 10:48:46 +02006275 spec->gen.mixer_nid = 0; /* ALC256 does not have any loopback mixer path */
Kailang Yangd32b6662015-04-23 15:10:53 +08006276 alc_update_coef_idx(codec, 0x36, 1 << 13, 1 << 5); /* Switch pcbeep path to Line in path*/
Kailang Yang4344aec2014-12-17 17:39:05 +08006277 break;
Kailang Yang42314302016-02-03 15:03:50 +08006278 case 0x10ec0225:
Kailang Yang7d727862016-05-24 16:46:07 +08006279 case 0x10ec0295:
Kailang Yangf6e94c22017-01-04 14:49:07 +08006280 case 0x10ec0299:
Kailang Yang42314302016-02-03 15:03:50 +08006281 spec->codec_variant = ALC269_TYPE_ALC225;
6282 break;
Kailang Yangdcd4f0d2016-05-04 15:50:18 +08006283 case 0x10ec0234:
6284 case 0x10ec0274:
6285 case 0x10ec0294:
6286 spec->codec_variant = ALC269_TYPE_ALC294;
6287 break;
Kailang Yang6fbae352016-05-30 16:44:20 +08006288 case 0x10ec0700:
6289 case 0x10ec0701:
6290 case 0x10ec0703:
6291 spec->codec_variant = ALC269_TYPE_ALC700;
6292 spec->gen.mixer_nid = 0; /* ALC700 does not have any loopback mixer path */
Kailang Yang193fd092017-11-22 15:21:32 +08006293 alc_update_coef_idx(codec, 0x4a, 1 << 15, 0); /* Combo jack auto trigger control */
Kailang Yang6fbae352016-05-30 16:44:20 +08006294 break;
6295
Takashi Iwai1d045db2011-07-07 18:23:21 +02006296 }
6297
Kailang Yangad60d502013-06-28 12:03:01 +02006298 if (snd_hda_codec_read(codec, 0x51, 0, AC_VERB_PARAMETERS, 0) == 0x10ec5505) {
Kailang Yang97a26572013-11-29 00:35:26 -05006299 spec->has_alc5505_dsp = 1;
Kailang Yangad60d502013-06-28 12:03:01 +02006300 spec->init_hook = alc5505_dsp_init;
6301 }
6302
Takashi Iwaia4297b52011-08-23 18:40:12 +02006303 /* automatic parse from the BIOS config */
6304 err = alc269_parse_auto_config(codec);
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02006305 if (err < 0)
6306 goto error;
Takashi Iwai1d045db2011-07-07 18:23:21 +02006307
David Henningsson7d1b6e22015-04-21 10:48:46 +02006308 if (!spec->gen.no_analog && spec->gen.beep_nid && spec->gen.mixer_nid)
6309 set_beep_amp(spec, spec->gen.mixer_nid, 0x04, HDA_INPUT);
Takashi Iwai1d045db2011-07-07 18:23:21 +02006310
Takashi Iwai1727a772013-01-10 09:52:52 +01006311 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
Takashi Iwai589876e2012-02-20 15:47:55 +01006312
Takashi Iwai1d045db2011-07-07 18:23:21 +02006313 return 0;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02006314
6315 error:
6316 alc_free(codec);
6317 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02006318}
6319
6320/*
6321 * ALC861
6322 */
6323
Takashi Iwai1d045db2011-07-07 18:23:21 +02006324static int alc861_parse_auto_config(struct hda_codec *codec)
6325{
Takashi Iwai1d045db2011-07-07 18:23:21 +02006326 static const hda_nid_t alc861_ignore[] = { 0x1d, 0 };
Takashi Iwai3e6179b2011-07-08 16:55:13 +02006327 static const hda_nid_t alc861_ssids[] = { 0x0e, 0x0f, 0x0b, 0 };
6328 return alc_parse_auto_config(codec, alc861_ignore, alc861_ssids);
Takashi Iwai1d045db2011-07-07 18:23:21 +02006329}
6330
Takashi Iwai1d045db2011-07-07 18:23:21 +02006331/* Pin config fixes */
6332enum {
Takashi Iwaie652f4c2012-02-13 11:56:25 +01006333 ALC861_FIXUP_FSC_AMILO_PI1505,
6334 ALC861_FIXUP_AMP_VREF_0F,
6335 ALC861_FIXUP_NO_JACK_DETECT,
6336 ALC861_FIXUP_ASUS_A6RP,
Takashi Iwai6ddf0fd2013-11-29 12:47:34 +01006337 ALC660_FIXUP_ASUS_W7J,
Takashi Iwai1d045db2011-07-07 18:23:21 +02006338};
6339
Takashi Iwai31150f22012-01-30 10:54:08 +01006340/* On some laptops, VREF of pin 0x0f is abused for controlling the main amp */
6341static void alc861_fixup_asus_amp_vref_0f(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01006342 const struct hda_fixup *fix, int action)
Takashi Iwai31150f22012-01-30 10:54:08 +01006343{
6344 struct alc_spec *spec = codec->spec;
6345 unsigned int val;
6346
Takashi Iwai1727a772013-01-10 09:52:52 +01006347 if (action != HDA_FIXUP_ACT_INIT)
Takashi Iwai31150f22012-01-30 10:54:08 +01006348 return;
Takashi Iwaid3f02d62013-01-10 10:12:22 +01006349 val = snd_hda_codec_get_pin_target(codec, 0x0f);
Takashi Iwai31150f22012-01-30 10:54:08 +01006350 if (!(val & (AC_PINCTL_IN_EN | AC_PINCTL_OUT_EN)))
6351 val |= AC_PINCTL_IN_EN;
6352 val |= AC_PINCTL_VREF_50;
Takashi Iwaicdd03ce2012-04-20 12:34:50 +02006353 snd_hda_set_pin_ctl(codec, 0x0f, val);
Takashi Iwai08c189f2012-12-19 15:22:24 +01006354 spec->gen.keep_vref_in_automute = 1;
Takashi Iwai31150f22012-01-30 10:54:08 +01006355}
6356
Takashi Iwaie652f4c2012-02-13 11:56:25 +01006357/* suppress the jack-detection */
6358static void alc_fixup_no_jack_detect(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01006359 const struct hda_fixup *fix, int action)
Takashi Iwaie652f4c2012-02-13 11:56:25 +01006360{
Takashi Iwai1727a772013-01-10 09:52:52 +01006361 if (action == HDA_FIXUP_ACT_PRE_PROBE)
Takashi Iwaie652f4c2012-02-13 11:56:25 +01006362 codec->no_jack_detect = 1;
Jesper Juhl7d7eb9e2012-04-12 22:11:25 +02006363}
Takashi Iwaie652f4c2012-02-13 11:56:25 +01006364
Takashi Iwai1727a772013-01-10 09:52:52 +01006365static const struct hda_fixup alc861_fixups[] = {
Takashi Iwaie652f4c2012-02-13 11:56:25 +01006366 [ALC861_FIXUP_FSC_AMILO_PI1505] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006367 .type = HDA_FIXUP_PINS,
6368 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02006369 { 0x0b, 0x0221101f }, /* HP */
6370 { 0x0f, 0x90170310 }, /* speaker */
6371 { }
6372 }
6373 },
Takashi Iwaie652f4c2012-02-13 11:56:25 +01006374 [ALC861_FIXUP_AMP_VREF_0F] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006375 .type = HDA_FIXUP_FUNC,
Takashi Iwai31150f22012-01-30 10:54:08 +01006376 .v.func = alc861_fixup_asus_amp_vref_0f,
Takashi Iwai3b25eb62012-01-25 09:55:46 +01006377 },
Takashi Iwaie652f4c2012-02-13 11:56:25 +01006378 [ALC861_FIXUP_NO_JACK_DETECT] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006379 .type = HDA_FIXUP_FUNC,
Takashi Iwaie652f4c2012-02-13 11:56:25 +01006380 .v.func = alc_fixup_no_jack_detect,
6381 },
6382 [ALC861_FIXUP_ASUS_A6RP] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006383 .type = HDA_FIXUP_FUNC,
Takashi Iwaie652f4c2012-02-13 11:56:25 +01006384 .v.func = alc861_fixup_asus_amp_vref_0f,
6385 .chained = true,
6386 .chain_id = ALC861_FIXUP_NO_JACK_DETECT,
Takashi Iwai6ddf0fd2013-11-29 12:47:34 +01006387 },
6388 [ALC660_FIXUP_ASUS_W7J] = {
6389 .type = HDA_FIXUP_VERBS,
6390 .v.verbs = (const struct hda_verb[]) {
6391 /* ASUS W7J needs a magic pin setup on unused NID 0x10
6392 * for enabling outputs
6393 */
6394 {0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
6395 { }
6396 },
Takashi Iwaie652f4c2012-02-13 11:56:25 +01006397 }
Takashi Iwai1d045db2011-07-07 18:23:21 +02006398};
6399
6400static const struct snd_pci_quirk alc861_fixup_tbl[] = {
Takashi Iwai6ddf0fd2013-11-29 12:47:34 +01006401 SND_PCI_QUIRK(0x1043, 0x1253, "ASUS W7J", ALC660_FIXUP_ASUS_W7J),
Takashi Iwaie7ca2372013-12-02 15:27:19 +01006402 SND_PCI_QUIRK(0x1043, 0x1263, "ASUS Z35HL", ALC660_FIXUP_ASUS_W7J),
Takashi Iwaie652f4c2012-02-13 11:56:25 +01006403 SND_PCI_QUIRK(0x1043, 0x1393, "ASUS A6Rp", ALC861_FIXUP_ASUS_A6RP),
6404 SND_PCI_QUIRK_VENDOR(0x1043, "ASUS laptop", ALC861_FIXUP_AMP_VREF_0F),
6405 SND_PCI_QUIRK(0x1462, 0x7254, "HP DX2200", ALC861_FIXUP_NO_JACK_DETECT),
6406 SND_PCI_QUIRK(0x1584, 0x2b01, "Haier W18", ALC861_FIXUP_AMP_VREF_0F),
6407 SND_PCI_QUIRK(0x1584, 0x0000, "Uniwill ECS M31EI", ALC861_FIXUP_AMP_VREF_0F),
6408 SND_PCI_QUIRK(0x1734, 0x10c7, "FSC Amilo Pi1505", ALC861_FIXUP_FSC_AMILO_PI1505),
Takashi Iwai1d045db2011-07-07 18:23:21 +02006409 {}
6410};
6411
6412/*
6413 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02006414static int patch_alc861(struct hda_codec *codec)
6415{
6416 struct alc_spec *spec;
Takashi Iwai1d045db2011-07-07 18:23:21 +02006417 int err;
6418
Takashi Iwai3de95172012-05-07 18:03:15 +02006419 err = alc_alloc_spec(codec, 0x15);
6420 if (err < 0)
6421 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02006422
Takashi Iwai3de95172012-05-07 18:03:15 +02006423 spec = codec->spec;
Takashi Iwai7504b6c2013-03-18 11:25:51 +01006424 spec->gen.beep_nid = 0x23;
Takashi Iwai1d045db2011-07-07 18:23:21 +02006425
Takashi Iwai225068a2015-05-29 10:42:14 +02006426#ifdef CONFIG_PM
6427 spec->power_hook = alc_power_eapd;
6428#endif
6429
Takashi Iwai1727a772013-01-10 09:52:52 +01006430 snd_hda_pick_fixup(codec, NULL, alc861_fixup_tbl, alc861_fixups);
6431 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
Takashi Iwai1d045db2011-07-07 18:23:21 +02006432
Takashi Iwaicb4e4822011-08-23 17:34:25 +02006433 /* automatic parse from the BIOS config */
6434 err = alc861_parse_auto_config(codec);
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02006435 if (err < 0)
6436 goto error;
Takashi Iwai1d045db2011-07-07 18:23:21 +02006437
Takashi Iwai7504b6c2013-03-18 11:25:51 +01006438 if (!spec->gen.no_analog)
Takashi Iwai3e6179b2011-07-08 16:55:13 +02006439 set_beep_amp(spec, 0x23, 0, HDA_OUTPUT);
Takashi Iwai1d045db2011-07-07 18:23:21 +02006440
Takashi Iwai1727a772013-01-10 09:52:52 +01006441 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
Takashi Iwai589876e2012-02-20 15:47:55 +01006442
Takashi Iwai1d045db2011-07-07 18:23:21 +02006443 return 0;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02006444
6445 error:
6446 alc_free(codec);
6447 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02006448}
6449
6450/*
6451 * ALC861-VD support
6452 *
6453 * Based on ALC882
6454 *
6455 * In addition, an independent DAC
6456 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02006457static int alc861vd_parse_auto_config(struct hda_codec *codec)
6458{
Takashi Iwai1d045db2011-07-07 18:23:21 +02006459 static const hda_nid_t alc861vd_ignore[] = { 0x1d, 0 };
Takashi Iwai3e6179b2011-07-08 16:55:13 +02006460 static const hda_nid_t alc861vd_ssids[] = { 0x15, 0x1b, 0x14, 0 };
6461 return alc_parse_auto_config(codec, alc861vd_ignore, alc861vd_ssids);
Takashi Iwai1d045db2011-07-07 18:23:21 +02006462}
6463
Takashi Iwai1d045db2011-07-07 18:23:21 +02006464enum {
Takashi Iwai8fdcb6f2011-08-23 17:28:55 +02006465 ALC660VD_FIX_ASUS_GPIO1,
6466 ALC861VD_FIX_DALLAS,
Takashi Iwai1d045db2011-07-07 18:23:21 +02006467};
6468
Takashi Iwai8fdcb6f2011-08-23 17:28:55 +02006469/* exclude VREF80 */
6470static void alc861vd_fixup_dallas(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01006471 const struct hda_fixup *fix, int action)
Takashi Iwai8fdcb6f2011-08-23 17:28:55 +02006472{
Takashi Iwai1727a772013-01-10 09:52:52 +01006473 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwaib78562b2012-12-17 20:06:49 +01006474 snd_hda_override_pin_caps(codec, 0x18, 0x00000734);
6475 snd_hda_override_pin_caps(codec, 0x19, 0x0000073c);
Takashi Iwai8fdcb6f2011-08-23 17:28:55 +02006476 }
6477}
6478
Takashi Iwai1727a772013-01-10 09:52:52 +01006479static const struct hda_fixup alc861vd_fixups[] = {
Takashi Iwai1d045db2011-07-07 18:23:21 +02006480 [ALC660VD_FIX_ASUS_GPIO1] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006481 .type = HDA_FIXUP_VERBS,
Takashi Iwai1d045db2011-07-07 18:23:21 +02006482 .v.verbs = (const struct hda_verb[]) {
Takashi Iwai8fdcb6f2011-08-23 17:28:55 +02006483 /* reset GPIO1 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02006484 {0x01, AC_VERB_SET_GPIO_MASK, 0x03},
6485 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x01},
6486 {0x01, AC_VERB_SET_GPIO_DATA, 0x01},
6487 { }
6488 }
6489 },
Takashi Iwai8fdcb6f2011-08-23 17:28:55 +02006490 [ALC861VD_FIX_DALLAS] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006491 .type = HDA_FIXUP_FUNC,
Takashi Iwai8fdcb6f2011-08-23 17:28:55 +02006492 .v.func = alc861vd_fixup_dallas,
6493 },
Takashi Iwai1d045db2011-07-07 18:23:21 +02006494};
6495
6496static const struct snd_pci_quirk alc861vd_fixup_tbl[] = {
Takashi Iwai8fdcb6f2011-08-23 17:28:55 +02006497 SND_PCI_QUIRK(0x103c, 0x30bf, "HP TX1000", ALC861VD_FIX_DALLAS),
Takashi Iwai1d045db2011-07-07 18:23:21 +02006498 SND_PCI_QUIRK(0x1043, 0x1339, "ASUS A7-K", ALC660VD_FIX_ASUS_GPIO1),
Takashi Iwai8fdcb6f2011-08-23 17:28:55 +02006499 SND_PCI_QUIRK(0x1179, 0xff31, "Toshiba L30-149", ALC861VD_FIX_DALLAS),
Takashi Iwai1d045db2011-07-07 18:23:21 +02006500 {}
6501};
6502
Takashi Iwai1d045db2011-07-07 18:23:21 +02006503/*
6504 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02006505static int patch_alc861vd(struct hda_codec *codec)
6506{
6507 struct alc_spec *spec;
Takashi Iwaicb4e4822011-08-23 17:34:25 +02006508 int err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02006509
Takashi Iwai3de95172012-05-07 18:03:15 +02006510 err = alc_alloc_spec(codec, 0x0b);
6511 if (err < 0)
6512 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02006513
Takashi Iwai3de95172012-05-07 18:03:15 +02006514 spec = codec->spec;
Takashi Iwai7504b6c2013-03-18 11:25:51 +01006515 spec->gen.beep_nid = 0x23;
Takashi Iwai1d045db2011-07-07 18:23:21 +02006516
Takashi Iwai225068a2015-05-29 10:42:14 +02006517 spec->shutup = alc_eapd_shutup;
6518
Takashi Iwai1727a772013-01-10 09:52:52 +01006519 snd_hda_pick_fixup(codec, NULL, alc861vd_fixup_tbl, alc861vd_fixups);
6520 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
Takashi Iwai1d045db2011-07-07 18:23:21 +02006521
Takashi Iwaicb4e4822011-08-23 17:34:25 +02006522 /* automatic parse from the BIOS config */
6523 err = alc861vd_parse_auto_config(codec);
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02006524 if (err < 0)
6525 goto error;
Takashi Iwai1d045db2011-07-07 18:23:21 +02006526
Takashi Iwai7504b6c2013-03-18 11:25:51 +01006527 if (!spec->gen.no_analog)
Takashi Iwai3e6179b2011-07-08 16:55:13 +02006528 set_beep_amp(spec, 0x0b, 0x05, HDA_INPUT);
Takashi Iwai1d045db2011-07-07 18:23:21 +02006529
Takashi Iwai1727a772013-01-10 09:52:52 +01006530 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
Takashi Iwai589876e2012-02-20 15:47:55 +01006531
Takashi Iwai1d045db2011-07-07 18:23:21 +02006532 return 0;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02006533
6534 error:
6535 alc_free(codec);
6536 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02006537}
6538
6539/*
6540 * ALC662 support
6541 *
6542 * ALC662 is almost identical with ALC880 but has cleaner and more flexible
6543 * configuration. Each pin widget can choose any input DACs and a mixer.
6544 * Each ADC is connected from a mixer of all inputs. This makes possible
6545 * 6-channel independent captures.
6546 *
6547 * In addition, an independent DAC for the multi-playback (not used in this
6548 * driver yet).
6549 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02006550
6551/*
6552 * BIOS auto configuration
6553 */
6554
Kailang Yangbc9f98a2007-04-12 13:06:07 +02006555static int alc662_parse_auto_config(struct hda_codec *codec)
6556{
Takashi Iwai4c6d72d2011-05-02 11:30:18 +02006557 static const hda_nid_t alc662_ignore[] = { 0x1d, 0 };
Takashi Iwai3e6179b2011-07-08 16:55:13 +02006558 static const hda_nid_t alc663_ssids[] = { 0x15, 0x1b, 0x14, 0x21 };
6559 static const hda_nid_t alc662_ssids[] = { 0x15, 0x1b, 0x14, 0 };
6560 const hda_nid_t *ssids;
Takashi Iwaiee979a142008-09-02 15:42:20 +02006561
Takashi Iwai7639a062015-03-03 10:07:24 +01006562 if (codec->core.vendor_id == 0x10ec0272 || codec->core.vendor_id == 0x10ec0663 ||
6563 codec->core.vendor_id == 0x10ec0665 || codec->core.vendor_id == 0x10ec0670 ||
6564 codec->core.vendor_id == 0x10ec0671)
Takashi Iwai3e6179b2011-07-08 16:55:13 +02006565 ssids = alc663_ssids;
Kailang Yang6227cdc2010-02-25 08:36:52 +01006566 else
Takashi Iwai3e6179b2011-07-08 16:55:13 +02006567 ssids = alc662_ssids;
6568 return alc_parse_auto_config(codec, alc662_ignore, ssids);
Kailang Yangbc9f98a2007-04-12 13:06:07 +02006569}
6570
Todd Broch6be79482010-12-07 16:51:05 -08006571static void alc272_fixup_mario(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01006572 const struct hda_fixup *fix, int action)
Takashi Iwai6fc398c2011-01-13 14:36:37 +01006573{
Takashi Iwai9bb1f062013-01-10 17:14:29 +01006574 if (action != HDA_FIXUP_ACT_PRE_PROBE)
Takashi Iwai6fc398c2011-01-13 14:36:37 +01006575 return;
Todd Broch6be79482010-12-07 16:51:05 -08006576 if (snd_hda_override_amp_caps(codec, 0x2, HDA_OUTPUT,
6577 (0x3b << AC_AMPCAP_OFFSET_SHIFT) |
6578 (0x3b << AC_AMPCAP_NUM_STEPS_SHIFT) |
6579 (0x03 << AC_AMPCAP_STEP_SIZE_SHIFT) |
6580 (0 << AC_AMPCAP_MUTE_SHIFT)))
Takashi Iwai4e76a882014-02-25 12:21:03 +01006581 codec_warn(codec, "failed to override amp caps for NID 0x2\n");
Todd Broch6be79482010-12-07 16:51:05 -08006582}
6583
Takashi Iwai8e383952013-10-30 17:41:12 +01006584static const struct snd_pcm_chmap_elem asus_pcm_2_1_chmaps[] = {
6585 { .channels = 2,
6586 .map = { SNDRV_CHMAP_FL, SNDRV_CHMAP_FR } },
6587 { .channels = 4,
6588 .map = { SNDRV_CHMAP_FL, SNDRV_CHMAP_FR,
6589 SNDRV_CHMAP_NA, SNDRV_CHMAP_LFE } }, /* LFE only on right */
6590 { }
6591};
6592
6593/* override the 2.1 chmap */
Takashi Iwaieb9ca3a2013-11-28 15:24:34 +01006594static void alc_fixup_bass_chmap(struct hda_codec *codec,
Takashi Iwai8e383952013-10-30 17:41:12 +01006595 const struct hda_fixup *fix, int action)
6596{
6597 if (action == HDA_FIXUP_ACT_BUILD) {
6598 struct alc_spec *spec = codec->spec;
Takashi Iwaibbbc7e82015-02-27 17:43:19 +01006599 spec->gen.pcm_rec[0]->stream[0].chmap = asus_pcm_2_1_chmaps;
Takashi Iwai8e383952013-10-30 17:41:12 +01006600 }
6601}
6602
Takashi Iwaibf686652014-01-13 16:18:25 +01006603/* avoid D3 for keeping GPIO up */
6604static unsigned int gpio_led_power_filter(struct hda_codec *codec,
6605 hda_nid_t nid,
6606 unsigned int power_state)
6607{
6608 struct alc_spec *spec = codec->spec;
Takashi Iwai7639a062015-03-03 10:07:24 +01006609 if (nid == codec->core.afg && power_state == AC_PWRST_D3 && spec->gpio_led)
Takashi Iwaibf686652014-01-13 16:18:25 +01006610 return AC_PWRST_D0;
6611 return power_state;
6612}
6613
Takashi Iwai3e887f32014-01-10 17:50:58 +01006614static void alc662_fixup_led_gpio1(struct hda_codec *codec,
6615 const struct hda_fixup *fix, int action)
6616{
6617 struct alc_spec *spec = codec->spec;
6618 static const struct hda_verb gpio_init[] = {
6619 { 0x01, AC_VERB_SET_GPIO_MASK, 0x01 },
6620 { 0x01, AC_VERB_SET_GPIO_DIRECTION, 0x01 },
6621 {}
6622 };
6623
6624 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwai0f32fd192014-11-19 12:16:14 +01006625 spec->gen.vmaster_mute.hook = alc_fixup_gpio_mute_hook;
Takashi Iwai3e887f32014-01-10 17:50:58 +01006626 spec->gpio_led = 0;
Takashi Iwai0f32fd192014-11-19 12:16:14 +01006627 spec->mute_led_polarity = 1;
6628 spec->gpio_mute_led_mask = 0x01;
Takashi Iwai3e887f32014-01-10 17:50:58 +01006629 snd_hda_add_verbs(codec, gpio_init);
Takashi Iwaibf686652014-01-13 16:18:25 +01006630 codec->power_filter = gpio_led_power_filter;
Takashi Iwai3e887f32014-01-10 17:50:58 +01006631 }
6632}
6633
Kailang Yangf3f91852014-10-24 15:43:46 +08006634static struct coef_fw alc668_coefs[] = {
6635 WRITE_COEF(0x01, 0xbebe), WRITE_COEF(0x02, 0xaaaa), WRITE_COEF(0x03, 0x0),
6636 WRITE_COEF(0x04, 0x0180), WRITE_COEF(0x06, 0x0), WRITE_COEF(0x07, 0x0f80),
6637 WRITE_COEF(0x08, 0x0031), WRITE_COEF(0x0a, 0x0060), WRITE_COEF(0x0b, 0x0),
6638 WRITE_COEF(0x0c, 0x7cf7), WRITE_COEF(0x0d, 0x1080), WRITE_COEF(0x0e, 0x7f7f),
6639 WRITE_COEF(0x0f, 0xcccc), WRITE_COEF(0x10, 0xddcc), WRITE_COEF(0x11, 0x0001),
6640 WRITE_COEF(0x13, 0x0), WRITE_COEF(0x14, 0x2aa0), WRITE_COEF(0x17, 0xa940),
6641 WRITE_COEF(0x19, 0x0), WRITE_COEF(0x1a, 0x0), WRITE_COEF(0x1b, 0x0),
6642 WRITE_COEF(0x1c, 0x0), WRITE_COEF(0x1d, 0x0), WRITE_COEF(0x1e, 0x7418),
6643 WRITE_COEF(0x1f, 0x0804), WRITE_COEF(0x20, 0x4200), WRITE_COEF(0x21, 0x0468),
6644 WRITE_COEF(0x22, 0x8ccc), WRITE_COEF(0x23, 0x0250), WRITE_COEF(0x24, 0x7418),
6645 WRITE_COEF(0x27, 0x0), WRITE_COEF(0x28, 0x8ccc), WRITE_COEF(0x2a, 0xff00),
6646 WRITE_COEF(0x2b, 0x8000), WRITE_COEF(0xa7, 0xff00), WRITE_COEF(0xa8, 0x8000),
6647 WRITE_COEF(0xaa, 0x2e17), WRITE_COEF(0xab, 0xa0c0), WRITE_COEF(0xac, 0x0),
6648 WRITE_COEF(0xad, 0x0), WRITE_COEF(0xae, 0x2ac6), WRITE_COEF(0xaf, 0xa480),
6649 WRITE_COEF(0xb0, 0x0), WRITE_COEF(0xb1, 0x0), WRITE_COEF(0xb2, 0x0),
6650 WRITE_COEF(0xb3, 0x0), WRITE_COEF(0xb4, 0x0), WRITE_COEF(0xb5, 0x1040),
6651 WRITE_COEF(0xb6, 0xd697), WRITE_COEF(0xb7, 0x902b), WRITE_COEF(0xb8, 0xd697),
6652 WRITE_COEF(0xb9, 0x902b), WRITE_COEF(0xba, 0xb8ba), WRITE_COEF(0xbb, 0xaaab),
6653 WRITE_COEF(0xbc, 0xaaaf), WRITE_COEF(0xbd, 0x6aaa), WRITE_COEF(0xbe, 0x1c02),
6654 WRITE_COEF(0xc0, 0x00ff), WRITE_COEF(0xc1, 0x0fa6),
6655 {}
6656};
6657
6658static void alc668_restore_default_value(struct hda_codec *codec)
6659{
6660 alc_process_coef_fw(codec, alc668_coefs);
6661}
6662
David Henningsson6cb3b702010-09-09 08:51:44 +02006663enum {
Daniel T Chen2df03512010-10-10 22:39:28 -04006664 ALC662_FIXUP_ASPIRE,
Takashi Iwai3e887f32014-01-10 17:50:58 +01006665 ALC662_FIXUP_LED_GPIO1,
David Henningsson6cb3b702010-09-09 08:51:44 +02006666 ALC662_FIXUP_IDEAPAD,
Todd Broch6be79482010-12-07 16:51:05 -08006667 ALC272_FIXUP_MARIO,
Anisse Astierd2ebd472011-01-20 12:36:21 +01006668 ALC662_FIXUP_CZC_P10T,
David Henningsson94024cd2011-04-29 14:10:55 +02006669 ALC662_FIXUP_SKU_IGNORE,
Takashi Iwaie59ea3e2011-06-29 17:21:00 +02006670 ALC662_FIXUP_HP_RP5800,
Takashi Iwai53c334a2011-08-23 18:27:14 +02006671 ALC662_FIXUP_ASUS_MODE1,
6672 ALC662_FIXUP_ASUS_MODE2,
6673 ALC662_FIXUP_ASUS_MODE3,
6674 ALC662_FIXUP_ASUS_MODE4,
6675 ALC662_FIXUP_ASUS_MODE5,
6676 ALC662_FIXUP_ASUS_MODE6,
6677 ALC662_FIXUP_ASUS_MODE7,
6678 ALC662_FIXUP_ASUS_MODE8,
Takashi Iwai1565cc32012-02-13 12:03:25 +01006679 ALC662_FIXUP_NO_JACK_DETECT,
David Henningssonedfe3bf2012-06-12 13:15:12 +02006680 ALC662_FIXUP_ZOTAC_Z68,
Takashi Iwai125821a2012-06-22 14:30:29 +02006681 ALC662_FIXUP_INV_DMIC,
David Henningsson1f8b46c2015-05-12 14:38:15 +02006682 ALC662_FIXUP_DELL_MIC_NO_PRESENCE,
David Henningsson73bdd592013-04-15 15:44:14 +02006683 ALC668_FIXUP_DELL_MIC_NO_PRESENCE,
David Henningsson1f8b46c2015-05-12 14:38:15 +02006684 ALC662_FIXUP_HEADSET_MODE,
David Henningsson73bdd592013-04-15 15:44:14 +02006685 ALC668_FIXUP_HEADSET_MODE,
David Henningsson8e54b4a2014-02-07 09:31:07 +01006686 ALC662_FIXUP_BASS_MODE4_CHMAP,
David Henningsson61a75f12014-02-07 09:31:08 +01006687 ALC662_FIXUP_BASS_16,
Takashi Iwaia30c9aa2013-11-21 08:00:20 +01006688 ALC662_FIXUP_BASS_1A,
David Henningsson8e54b4a2014-02-07 09:31:07 +01006689 ALC662_FIXUP_BASS_CHMAP,
Hui Wang493a52a2014-01-14 14:07:36 +08006690 ALC668_FIXUP_AUTO_MUTE,
Gabriele Mazzotta5e6db662014-05-20 18:58:26 +02006691 ALC668_FIXUP_DELL_DISABLE_AAMIX,
Gabriele Mazzotta033b0a72014-05-26 17:11:46 +02006692 ALC668_FIXUP_DELL_XPS13,
Bobi Mihalca9d4dc582016-03-23 13:26:11 +02006693 ALC662_FIXUP_ASUS_Nx50,
Yura Pakhuchiy3231e202016-05-07 23:53:36 +07006694 ALC668_FIXUP_ASUS_Nx51,
Kailang Yang78f4f7c2016-06-07 11:31:34 +08006695 ALC891_FIXUP_HEADSET_MODE,
6696 ALC891_FIXUP_DELL_MIC_NO_PRESENCE,
Shrirang Bagul9b51fe32016-08-01 13:16:17 +08006697 ALC662_FIXUP_ACER_VERITON,
Takashi Iwai1a3f0992016-10-20 12:14:51 +02006698 ALC892_FIXUP_ASROCK_MOBO,
David Henningsson6cb3b702010-09-09 08:51:44 +02006699};
6700
Takashi Iwai1727a772013-01-10 09:52:52 +01006701static const struct hda_fixup alc662_fixups[] = {
Daniel T Chen2df03512010-10-10 22:39:28 -04006702 [ALC662_FIXUP_ASPIRE] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006703 .type = HDA_FIXUP_PINS,
6704 .v.pins = (const struct hda_pintbl[]) {
Daniel T Chen2df03512010-10-10 22:39:28 -04006705 { 0x15, 0x99130112 }, /* subwoofer */
6706 { }
6707 }
6708 },
Takashi Iwai3e887f32014-01-10 17:50:58 +01006709 [ALC662_FIXUP_LED_GPIO1] = {
6710 .type = HDA_FIXUP_FUNC,
6711 .v.func = alc662_fixup_led_gpio1,
6712 },
David Henningsson6cb3b702010-09-09 08:51:44 +02006713 [ALC662_FIXUP_IDEAPAD] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006714 .type = HDA_FIXUP_PINS,
6715 .v.pins = (const struct hda_pintbl[]) {
David Henningsson6cb3b702010-09-09 08:51:44 +02006716 { 0x17, 0x99130112 }, /* subwoofer */
6717 { }
Takashi Iwai3e887f32014-01-10 17:50:58 +01006718 },
6719 .chained = true,
6720 .chain_id = ALC662_FIXUP_LED_GPIO1,
David Henningsson6cb3b702010-09-09 08:51:44 +02006721 },
Todd Broch6be79482010-12-07 16:51:05 -08006722 [ALC272_FIXUP_MARIO] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006723 .type = HDA_FIXUP_FUNC,
Takashi Iwaib5bfbc62011-01-13 14:22:32 +01006724 .v.func = alc272_fixup_mario,
Anisse Astierd2ebd472011-01-20 12:36:21 +01006725 },
6726 [ALC662_FIXUP_CZC_P10T] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006727 .type = HDA_FIXUP_VERBS,
Anisse Astierd2ebd472011-01-20 12:36:21 +01006728 .v.verbs = (const struct hda_verb[]) {
6729 {0x14, AC_VERB_SET_EAPD_BTLENABLE, 0},
6730 {}
6731 }
6732 },
David Henningsson94024cd2011-04-29 14:10:55 +02006733 [ALC662_FIXUP_SKU_IGNORE] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006734 .type = HDA_FIXUP_FUNC,
Takashi Iwai23d30f22012-05-07 17:17:32 +02006735 .v.func = alc_fixup_sku_ignore,
Takashi Iwaic6b35872011-03-28 12:05:31 +02006736 },
Takashi Iwaie59ea3e2011-06-29 17:21:00 +02006737 [ALC662_FIXUP_HP_RP5800] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006738 .type = HDA_FIXUP_PINS,
6739 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaie59ea3e2011-06-29 17:21:00 +02006740 { 0x14, 0x0221201f }, /* HP out */
6741 { }
6742 },
6743 .chained = true,
6744 .chain_id = ALC662_FIXUP_SKU_IGNORE
6745 },
Takashi Iwai53c334a2011-08-23 18:27:14 +02006746 [ALC662_FIXUP_ASUS_MODE1] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006747 .type = HDA_FIXUP_PINS,
6748 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai53c334a2011-08-23 18:27:14 +02006749 { 0x14, 0x99130110 }, /* speaker */
6750 { 0x18, 0x01a19c20 }, /* mic */
6751 { 0x19, 0x99a3092f }, /* int-mic */
6752 { 0x21, 0x0121401f }, /* HP out */
6753 { }
6754 },
6755 .chained = true,
6756 .chain_id = ALC662_FIXUP_SKU_IGNORE
6757 },
6758 [ALC662_FIXUP_ASUS_MODE2] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006759 .type = HDA_FIXUP_PINS,
6760 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai2996bdb2011-08-18 16:02:24 +02006761 { 0x14, 0x99130110 }, /* speaker */
6762 { 0x18, 0x01a19820 }, /* mic */
6763 { 0x19, 0x99a3092f }, /* int-mic */
6764 { 0x1b, 0x0121401f }, /* HP out */
6765 { }
6766 },
Takashi Iwai53c334a2011-08-23 18:27:14 +02006767 .chained = true,
6768 .chain_id = ALC662_FIXUP_SKU_IGNORE
6769 },
6770 [ALC662_FIXUP_ASUS_MODE3] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006771 .type = HDA_FIXUP_PINS,
6772 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai53c334a2011-08-23 18:27:14 +02006773 { 0x14, 0x99130110 }, /* speaker */
6774 { 0x15, 0x0121441f }, /* HP */
6775 { 0x18, 0x01a19840 }, /* mic */
6776 { 0x19, 0x99a3094f }, /* int-mic */
6777 { 0x21, 0x01211420 }, /* HP2 */
6778 { }
6779 },
6780 .chained = true,
6781 .chain_id = ALC662_FIXUP_SKU_IGNORE
6782 },
6783 [ALC662_FIXUP_ASUS_MODE4] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006784 .type = HDA_FIXUP_PINS,
6785 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai53c334a2011-08-23 18:27:14 +02006786 { 0x14, 0x99130110 }, /* speaker */
6787 { 0x16, 0x99130111 }, /* speaker */
6788 { 0x18, 0x01a19840 }, /* mic */
6789 { 0x19, 0x99a3094f }, /* int-mic */
6790 { 0x21, 0x0121441f }, /* HP */
6791 { }
6792 },
6793 .chained = true,
6794 .chain_id = ALC662_FIXUP_SKU_IGNORE
6795 },
6796 [ALC662_FIXUP_ASUS_MODE5] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006797 .type = HDA_FIXUP_PINS,
6798 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai53c334a2011-08-23 18:27:14 +02006799 { 0x14, 0x99130110 }, /* speaker */
6800 { 0x15, 0x0121441f }, /* HP */
6801 { 0x16, 0x99130111 }, /* speaker */
6802 { 0x18, 0x01a19840 }, /* mic */
6803 { 0x19, 0x99a3094f }, /* int-mic */
6804 { }
6805 },
6806 .chained = true,
6807 .chain_id = ALC662_FIXUP_SKU_IGNORE
6808 },
6809 [ALC662_FIXUP_ASUS_MODE6] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006810 .type = HDA_FIXUP_PINS,
6811 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai53c334a2011-08-23 18:27:14 +02006812 { 0x14, 0x99130110 }, /* speaker */
6813 { 0x15, 0x01211420 }, /* HP2 */
6814 { 0x18, 0x01a19840 }, /* mic */
6815 { 0x19, 0x99a3094f }, /* int-mic */
6816 { 0x1b, 0x0121441f }, /* HP */
6817 { }
6818 },
6819 .chained = true,
6820 .chain_id = ALC662_FIXUP_SKU_IGNORE
6821 },
6822 [ALC662_FIXUP_ASUS_MODE7] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006823 .type = HDA_FIXUP_PINS,
6824 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai53c334a2011-08-23 18:27:14 +02006825 { 0x14, 0x99130110 }, /* speaker */
6826 { 0x17, 0x99130111 }, /* speaker */
6827 { 0x18, 0x01a19840 }, /* mic */
6828 { 0x19, 0x99a3094f }, /* int-mic */
6829 { 0x1b, 0x01214020 }, /* HP */
6830 { 0x21, 0x0121401f }, /* HP */
6831 { }
6832 },
6833 .chained = true,
6834 .chain_id = ALC662_FIXUP_SKU_IGNORE
6835 },
6836 [ALC662_FIXUP_ASUS_MODE8] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006837 .type = HDA_FIXUP_PINS,
6838 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai53c334a2011-08-23 18:27:14 +02006839 { 0x14, 0x99130110 }, /* speaker */
6840 { 0x12, 0x99a30970 }, /* int-mic */
6841 { 0x15, 0x01214020 }, /* HP */
6842 { 0x17, 0x99130111 }, /* speaker */
6843 { 0x18, 0x01a19840 }, /* mic */
6844 { 0x21, 0x0121401f }, /* HP */
6845 { }
6846 },
6847 .chained = true,
6848 .chain_id = ALC662_FIXUP_SKU_IGNORE
Takashi Iwai2996bdb2011-08-18 16:02:24 +02006849 },
Takashi Iwai1565cc32012-02-13 12:03:25 +01006850 [ALC662_FIXUP_NO_JACK_DETECT] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006851 .type = HDA_FIXUP_FUNC,
Takashi Iwai1565cc32012-02-13 12:03:25 +01006852 .v.func = alc_fixup_no_jack_detect,
6853 },
David Henningssonedfe3bf2012-06-12 13:15:12 +02006854 [ALC662_FIXUP_ZOTAC_Z68] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006855 .type = HDA_FIXUP_PINS,
6856 .v.pins = (const struct hda_pintbl[]) {
David Henningssonedfe3bf2012-06-12 13:15:12 +02006857 { 0x1b, 0x02214020 }, /* Front HP */
6858 { }
6859 }
6860 },
Takashi Iwai125821a2012-06-22 14:30:29 +02006861 [ALC662_FIXUP_INV_DMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006862 .type = HDA_FIXUP_FUNC,
David Henningsson9d36a7d2014-10-07 10:18:42 +02006863 .v.func = alc_fixup_inv_dmic,
Takashi Iwai125821a2012-06-22 14:30:29 +02006864 },
Gabriele Mazzotta033b0a72014-05-26 17:11:46 +02006865 [ALC668_FIXUP_DELL_XPS13] = {
6866 .type = HDA_FIXUP_FUNC,
6867 .v.func = alc_fixup_dell_xps13,
6868 .chained = true,
6869 .chain_id = ALC668_FIXUP_DELL_DISABLE_AAMIX
6870 },
Gabriele Mazzotta5e6db662014-05-20 18:58:26 +02006871 [ALC668_FIXUP_DELL_DISABLE_AAMIX] = {
6872 .type = HDA_FIXUP_FUNC,
6873 .v.func = alc_fixup_disable_aamix,
6874 .chained = true,
6875 .chain_id = ALC668_FIXUP_DELL_MIC_NO_PRESENCE
6876 },
Hui Wang493a52a2014-01-14 14:07:36 +08006877 [ALC668_FIXUP_AUTO_MUTE] = {
6878 .type = HDA_FIXUP_FUNC,
6879 .v.func = alc_fixup_auto_mute_via_amp,
6880 .chained = true,
6881 .chain_id = ALC668_FIXUP_DELL_MIC_NO_PRESENCE
6882 },
David Henningsson1f8b46c2015-05-12 14:38:15 +02006883 [ALC662_FIXUP_DELL_MIC_NO_PRESENCE] = {
6884 .type = HDA_FIXUP_PINS,
6885 .v.pins = (const struct hda_pintbl[]) {
6886 { 0x19, 0x03a1113c }, /* use as headset mic, without its own jack detect */
6887 /* headphone mic by setting pin control of 0x1b (headphone out) to in + vref_50 */
6888 { }
6889 },
6890 .chained = true,
6891 .chain_id = ALC662_FIXUP_HEADSET_MODE
6892 },
6893 [ALC662_FIXUP_HEADSET_MODE] = {
6894 .type = HDA_FIXUP_FUNC,
6895 .v.func = alc_fixup_headset_mode_alc662,
6896 },
David Henningsson73bdd592013-04-15 15:44:14 +02006897 [ALC668_FIXUP_DELL_MIC_NO_PRESENCE] = {
6898 .type = HDA_FIXUP_PINS,
6899 .v.pins = (const struct hda_pintbl[]) {
6900 { 0x19, 0x03a1913d }, /* use as headphone mic, without its own jack detect */
6901 { 0x1b, 0x03a1113c }, /* use as headset mic, without its own jack detect */
6902 { }
6903 },
6904 .chained = true,
6905 .chain_id = ALC668_FIXUP_HEADSET_MODE
6906 },
6907 [ALC668_FIXUP_HEADSET_MODE] = {
6908 .type = HDA_FIXUP_FUNC,
6909 .v.func = alc_fixup_headset_mode_alc668,
6910 },
David Henningsson8e54b4a2014-02-07 09:31:07 +01006911 [ALC662_FIXUP_BASS_MODE4_CHMAP] = {
Takashi Iwai8e383952013-10-30 17:41:12 +01006912 .type = HDA_FIXUP_FUNC,
Takashi Iwaieb9ca3a2013-11-28 15:24:34 +01006913 .v.func = alc_fixup_bass_chmap,
Takashi Iwai8e383952013-10-30 17:41:12 +01006914 .chained = true,
6915 .chain_id = ALC662_FIXUP_ASUS_MODE4
6916 },
David Henningsson61a75f12014-02-07 09:31:08 +01006917 [ALC662_FIXUP_BASS_16] = {
6918 .type = HDA_FIXUP_PINS,
6919 .v.pins = (const struct hda_pintbl[]) {
6920 {0x16, 0x80106111}, /* bass speaker */
6921 {}
6922 },
6923 .chained = true,
6924 .chain_id = ALC662_FIXUP_BASS_CHMAP,
6925 },
Takashi Iwaia30c9aa2013-11-21 08:00:20 +01006926 [ALC662_FIXUP_BASS_1A] = {
6927 .type = HDA_FIXUP_PINS,
6928 .v.pins = (const struct hda_pintbl[]) {
6929 {0x1a, 0x80106111}, /* bass speaker */
6930 {}
6931 },
David Henningsson8e54b4a2014-02-07 09:31:07 +01006932 .chained = true,
6933 .chain_id = ALC662_FIXUP_BASS_CHMAP,
Takashi Iwaia30c9aa2013-11-21 08:00:20 +01006934 },
David Henningsson8e54b4a2014-02-07 09:31:07 +01006935 [ALC662_FIXUP_BASS_CHMAP] = {
Takashi Iwaia30c9aa2013-11-21 08:00:20 +01006936 .type = HDA_FIXUP_FUNC,
Takashi Iwaieb9ca3a2013-11-28 15:24:34 +01006937 .v.func = alc_fixup_bass_chmap,
Takashi Iwaia30c9aa2013-11-21 08:00:20 +01006938 },
Bobi Mihalca9d4dc582016-03-23 13:26:11 +02006939 [ALC662_FIXUP_ASUS_Nx50] = {
6940 .type = HDA_FIXUP_FUNC,
6941 .v.func = alc_fixup_auto_mute_via_amp,
6942 .chained = true,
6943 .chain_id = ALC662_FIXUP_BASS_1A
6944 },
Yura Pakhuchiy3231e202016-05-07 23:53:36 +07006945 [ALC668_FIXUP_ASUS_Nx51] = {
6946 .type = HDA_FIXUP_PINS,
6947 .v.pins = (const struct hda_pintbl[]) {
6948 {0x1a, 0x90170151}, /* bass speaker */
6949 {}
6950 },
6951 .chained = true,
6952 .chain_id = ALC662_FIXUP_BASS_CHMAP,
6953 },
Kailang Yang78f4f7c2016-06-07 11:31:34 +08006954 [ALC891_FIXUP_HEADSET_MODE] = {
6955 .type = HDA_FIXUP_FUNC,
6956 .v.func = alc_fixup_headset_mode,
6957 },
6958 [ALC891_FIXUP_DELL_MIC_NO_PRESENCE] = {
6959 .type = HDA_FIXUP_PINS,
6960 .v.pins = (const struct hda_pintbl[]) {
6961 { 0x19, 0x03a1913d }, /* use as headphone mic, without its own jack detect */
6962 { 0x1b, 0x03a1113c }, /* use as headset mic, without its own jack detect */
6963 { }
6964 },
6965 .chained = true,
6966 .chain_id = ALC891_FIXUP_HEADSET_MODE
6967 },
Shrirang Bagul9b51fe32016-08-01 13:16:17 +08006968 [ALC662_FIXUP_ACER_VERITON] = {
6969 .type = HDA_FIXUP_PINS,
6970 .v.pins = (const struct hda_pintbl[]) {
6971 { 0x15, 0x50170120 }, /* no internal speaker */
6972 { }
6973 }
6974 },
Takashi Iwai1a3f0992016-10-20 12:14:51 +02006975 [ALC892_FIXUP_ASROCK_MOBO] = {
6976 .type = HDA_FIXUP_PINS,
6977 .v.pins = (const struct hda_pintbl[]) {
6978 { 0x15, 0x40f000f0 }, /* disabled */
6979 { 0x16, 0x40f000f0 }, /* disabled */
Takashi Iwai1a3f0992016-10-20 12:14:51 +02006980 { }
6981 }
6982 },
David Henningsson6cb3b702010-09-09 08:51:44 +02006983};
6984
Takashi Iwaia9111322011-05-02 11:30:18 +02006985static const struct snd_pci_quirk alc662_fixup_tbl[] = {
Takashi Iwai53c334a2011-08-23 18:27:14 +02006986 SND_PCI_QUIRK(0x1019, 0x9087, "ECS", ALC662_FIXUP_ASUS_MODE2),
Takashi Iwaid3d38352013-08-19 20:05:50 +02006987 SND_PCI_QUIRK(0x1025, 0x022f, "Acer Aspire One", ALC662_FIXUP_INV_DMIC),
David Henningsson02f6ff92015-12-07 11:29:31 +01006988 SND_PCI_QUIRK(0x1025, 0x0241, "Packard Bell DOTS", ALC662_FIXUP_INV_DMIC),
David Henningssona6c47a82011-02-10 15:39:19 +01006989 SND_PCI_QUIRK(0x1025, 0x0308, "Acer Aspire 8942G", ALC662_FIXUP_ASPIRE),
David Henningsson94024cd2011-04-29 14:10:55 +02006990 SND_PCI_QUIRK(0x1025, 0x031c, "Gateway NV79", ALC662_FIXUP_SKU_IGNORE),
Takashi Iwai125821a2012-06-22 14:30:29 +02006991 SND_PCI_QUIRK(0x1025, 0x0349, "eMachines eM250", ALC662_FIXUP_INV_DMIC),
Takashi Iwai18019282013-08-16 08:17:05 +02006992 SND_PCI_QUIRK(0x1025, 0x034a, "Gateway LT27", ALC662_FIXUP_INV_DMIC),
Daniel T Chen2df03512010-10-10 22:39:28 -04006993 SND_PCI_QUIRK(0x1025, 0x038b, "Acer Aspire 8943G", ALC662_FIXUP_ASPIRE),
David Henningsson73bdd592013-04-15 15:44:14 +02006994 SND_PCI_QUIRK(0x1028, 0x05d8, "Dell", ALC668_FIXUP_DELL_MIC_NO_PRESENCE),
6995 SND_PCI_QUIRK(0x1028, 0x05db, "Dell", ALC668_FIXUP_DELL_MIC_NO_PRESENCE),
Takashi Iwaic5d019c2014-07-22 17:33:32 +02006996 SND_PCI_QUIRK(0x1028, 0x05fe, "Dell XPS 15", ALC668_FIXUP_DELL_XPS13),
Gabriele Mazzotta033b0a72014-05-26 17:11:46 +02006997 SND_PCI_QUIRK(0x1028, 0x060a, "Dell XPS 13", ALC668_FIXUP_DELL_XPS13),
Niranjan Sivakumar467e1432015-09-05 18:20:35 +02006998 SND_PCI_QUIRK(0x1028, 0x060d, "Dell M3800", ALC668_FIXUP_DELL_XPS13),
David Henningsson09d20142013-11-20 11:43:30 +01006999 SND_PCI_QUIRK(0x1028, 0x0625, "Dell", ALC668_FIXUP_DELL_MIC_NO_PRESENCE),
David Henningssonad8ff992013-11-07 09:28:59 +01007000 SND_PCI_QUIRK(0x1028, 0x0626, "Dell", ALC668_FIXUP_DELL_MIC_NO_PRESENCE),
Kailang Yang8dc9abb2014-04-16 15:53:12 +08007001 SND_PCI_QUIRK(0x1028, 0x0696, "Dell", ALC668_FIXUP_DELL_MIC_NO_PRESENCE),
7002 SND_PCI_QUIRK(0x1028, 0x0698, "Dell", ALC668_FIXUP_DELL_MIC_NO_PRESENCE),
Kailang Yang6a98e342014-10-27 16:20:30 +08007003 SND_PCI_QUIRK(0x1028, 0x069f, "Dell", ALC668_FIXUP_DELL_MIC_NO_PRESENCE),
Takashi Iwaie59ea3e2011-06-29 17:21:00 +02007004 SND_PCI_QUIRK(0x103c, 0x1632, "HP RP5800", ALC662_FIXUP_HP_RP5800),
Kaho Ng2da2dc92016-05-09 00:27:49 +08007005 SND_PCI_QUIRK(0x1043, 0x1080, "Asus UX501VW", ALC668_FIXUP_HEADSET_MODE),
Bobi Mihalca83a9efb2016-03-23 13:32:33 +02007006 SND_PCI_QUIRK(0x1043, 0x11cd, "Asus N550", ALC662_FIXUP_ASUS_Nx50),
Takashi Iwaidb8948e2016-01-18 09:17:30 +01007007 SND_PCI_QUIRK(0x1043, 0x13df, "Asus N550JX", ALC662_FIXUP_BASS_1A),
Bobi Mihalca9d4dc582016-03-23 13:26:11 +02007008 SND_PCI_QUIRK(0x1043, 0x129d, "Asus N750", ALC662_FIXUP_ASUS_Nx50),
David Henningsson8e54b4a2014-02-07 09:31:07 +01007009 SND_PCI_QUIRK(0x1043, 0x1477, "ASUS N56VZ", ALC662_FIXUP_BASS_MODE4_CHMAP),
David Henningsson61a75f12014-02-07 09:31:08 +01007010 SND_PCI_QUIRK(0x1043, 0x15a7, "ASUS UX51VZH", ALC662_FIXUP_BASS_16),
Yura Pakhuchiy3231e202016-05-07 23:53:36 +07007011 SND_PCI_QUIRK(0x1043, 0x177d, "ASUS N551", ALC668_FIXUP_ASUS_Nx51),
7012 SND_PCI_QUIRK(0x1043, 0x17bd, "ASUS N751", ALC668_FIXUP_ASUS_Nx51),
Takashi Iwai5b2c3ca2017-01-04 21:38:16 +01007013 SND_PCI_QUIRK(0x1043, 0x1963, "ASUS X71SL", ALC662_FIXUP_ASUS_MODE8),
David Henningsson61a75f12014-02-07 09:31:08 +01007014 SND_PCI_QUIRK(0x1043, 0x1b73, "ASUS N55SF", ALC662_FIXUP_BASS_16),
David Henningsson8e54b4a2014-02-07 09:31:07 +01007015 SND_PCI_QUIRK(0x1043, 0x1bf3, "ASUS N76VZ", ALC662_FIXUP_BASS_MODE4_CHMAP),
Takashi Iwai1565cc32012-02-13 12:03:25 +01007016 SND_PCI_QUIRK(0x1043, 0x8469, "ASUS mobo", ALC662_FIXUP_NO_JACK_DETECT),
Takashi Iwai53c334a2011-08-23 18:27:14 +02007017 SND_PCI_QUIRK(0x105b, 0x0cd6, "Foxconn", ALC662_FIXUP_ASUS_MODE2),
Daniel T Chena0e90ac2010-11-20 10:20:35 -05007018 SND_PCI_QUIRK(0x144d, 0xc051, "Samsung R720", ALC662_FIXUP_IDEAPAD),
Valentine Sinitsynd4118582010-10-01 22:24:08 +06007019 SND_PCI_QUIRK(0x17aa, 0x38af, "Lenovo Ideapad Y550P", ALC662_FIXUP_IDEAPAD),
David Henningsson6cb3b702010-09-09 08:51:44 +02007020 SND_PCI_QUIRK(0x17aa, 0x3a0d, "Lenovo Ideapad Y550", ALC662_FIXUP_IDEAPAD),
Takashi Iwai1a3f0992016-10-20 12:14:51 +02007021 SND_PCI_QUIRK(0x1849, 0x5892, "ASRock B150M", ALC892_FIXUP_ASROCK_MOBO),
David Henningssonedfe3bf2012-06-12 13:15:12 +02007022 SND_PCI_QUIRK(0x19da, 0xa130, "Zotac Z68", ALC662_FIXUP_ZOTAC_Z68),
Shrirang Bagul9b51fe32016-08-01 13:16:17 +08007023 SND_PCI_QUIRK(0x1b0a, 0x01b8, "ACER Veriton", ALC662_FIXUP_ACER_VERITON),
Anisse Astierd2ebd472011-01-20 12:36:21 +01007024 SND_PCI_QUIRK(0x1b35, 0x2206, "CZC P10T", ALC662_FIXUP_CZC_P10T),
Takashi Iwai53c334a2011-08-23 18:27:14 +02007025
7026#if 0
7027 /* Below is a quirk table taken from the old code.
7028 * Basically the device should work as is without the fixup table.
7029 * If BIOS doesn't give a proper info, enable the corresponding
7030 * fixup entry.
Jesper Juhl7d7eb9e2012-04-12 22:11:25 +02007031 */
Takashi Iwai53c334a2011-08-23 18:27:14 +02007032 SND_PCI_QUIRK(0x1043, 0x1000, "ASUS N50Vm", ALC662_FIXUP_ASUS_MODE1),
7033 SND_PCI_QUIRK(0x1043, 0x1092, "ASUS NB", ALC662_FIXUP_ASUS_MODE3),
7034 SND_PCI_QUIRK(0x1043, 0x1173, "ASUS K73Jn", ALC662_FIXUP_ASUS_MODE1),
7035 SND_PCI_QUIRK(0x1043, 0x11c3, "ASUS M70V", ALC662_FIXUP_ASUS_MODE3),
7036 SND_PCI_QUIRK(0x1043, 0x11d3, "ASUS NB", ALC662_FIXUP_ASUS_MODE1),
7037 SND_PCI_QUIRK(0x1043, 0x11f3, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
7038 SND_PCI_QUIRK(0x1043, 0x1203, "ASUS NB", ALC662_FIXUP_ASUS_MODE1),
7039 SND_PCI_QUIRK(0x1043, 0x1303, "ASUS G60J", ALC662_FIXUP_ASUS_MODE1),
7040 SND_PCI_QUIRK(0x1043, 0x1333, "ASUS G60Jx", ALC662_FIXUP_ASUS_MODE1),
7041 SND_PCI_QUIRK(0x1043, 0x1339, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
7042 SND_PCI_QUIRK(0x1043, 0x13e3, "ASUS N71JA", ALC662_FIXUP_ASUS_MODE7),
7043 SND_PCI_QUIRK(0x1043, 0x1463, "ASUS N71", ALC662_FIXUP_ASUS_MODE7),
7044 SND_PCI_QUIRK(0x1043, 0x14d3, "ASUS G72", ALC662_FIXUP_ASUS_MODE8),
7045 SND_PCI_QUIRK(0x1043, 0x1563, "ASUS N90", ALC662_FIXUP_ASUS_MODE3),
7046 SND_PCI_QUIRK(0x1043, 0x15d3, "ASUS N50SF F50SF", ALC662_FIXUP_ASUS_MODE1),
7047 SND_PCI_QUIRK(0x1043, 0x16c3, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
7048 SND_PCI_QUIRK(0x1043, 0x16f3, "ASUS K40C K50C", ALC662_FIXUP_ASUS_MODE2),
7049 SND_PCI_QUIRK(0x1043, 0x1733, "ASUS N81De", ALC662_FIXUP_ASUS_MODE1),
7050 SND_PCI_QUIRK(0x1043, 0x1753, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
7051 SND_PCI_QUIRK(0x1043, 0x1763, "ASUS NB", ALC662_FIXUP_ASUS_MODE6),
7052 SND_PCI_QUIRK(0x1043, 0x1765, "ASUS NB", ALC662_FIXUP_ASUS_MODE6),
7053 SND_PCI_QUIRK(0x1043, 0x1783, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
7054 SND_PCI_QUIRK(0x1043, 0x1793, "ASUS F50GX", ALC662_FIXUP_ASUS_MODE1),
7055 SND_PCI_QUIRK(0x1043, 0x17b3, "ASUS F70SL", ALC662_FIXUP_ASUS_MODE3),
7056 SND_PCI_QUIRK(0x1043, 0x17f3, "ASUS X58LE", ALC662_FIXUP_ASUS_MODE2),
7057 SND_PCI_QUIRK(0x1043, 0x1813, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
7058 SND_PCI_QUIRK(0x1043, 0x1823, "ASUS NB", ALC662_FIXUP_ASUS_MODE5),
7059 SND_PCI_QUIRK(0x1043, 0x1833, "ASUS NB", ALC662_FIXUP_ASUS_MODE6),
7060 SND_PCI_QUIRK(0x1043, 0x1843, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
7061 SND_PCI_QUIRK(0x1043, 0x1853, "ASUS F50Z", ALC662_FIXUP_ASUS_MODE1),
7062 SND_PCI_QUIRK(0x1043, 0x1864, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
7063 SND_PCI_QUIRK(0x1043, 0x1876, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
7064 SND_PCI_QUIRK(0x1043, 0x1893, "ASUS M50Vm", ALC662_FIXUP_ASUS_MODE3),
7065 SND_PCI_QUIRK(0x1043, 0x1894, "ASUS X55", ALC662_FIXUP_ASUS_MODE3),
7066 SND_PCI_QUIRK(0x1043, 0x18b3, "ASUS N80Vc", ALC662_FIXUP_ASUS_MODE1),
7067 SND_PCI_QUIRK(0x1043, 0x18c3, "ASUS VX5", ALC662_FIXUP_ASUS_MODE1),
7068 SND_PCI_QUIRK(0x1043, 0x18d3, "ASUS N81Te", ALC662_FIXUP_ASUS_MODE1),
7069 SND_PCI_QUIRK(0x1043, 0x18f3, "ASUS N505Tp", ALC662_FIXUP_ASUS_MODE1),
7070 SND_PCI_QUIRK(0x1043, 0x1903, "ASUS F5GL", ALC662_FIXUP_ASUS_MODE1),
7071 SND_PCI_QUIRK(0x1043, 0x1913, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
7072 SND_PCI_QUIRK(0x1043, 0x1933, "ASUS F80Q", ALC662_FIXUP_ASUS_MODE2),
7073 SND_PCI_QUIRK(0x1043, 0x1943, "ASUS Vx3V", ALC662_FIXUP_ASUS_MODE1),
7074 SND_PCI_QUIRK(0x1043, 0x1953, "ASUS NB", ALC662_FIXUP_ASUS_MODE1),
7075 SND_PCI_QUIRK(0x1043, 0x1963, "ASUS X71C", ALC662_FIXUP_ASUS_MODE3),
7076 SND_PCI_QUIRK(0x1043, 0x1983, "ASUS N5051A", ALC662_FIXUP_ASUS_MODE1),
7077 SND_PCI_QUIRK(0x1043, 0x1993, "ASUS N20", ALC662_FIXUP_ASUS_MODE1),
7078 SND_PCI_QUIRK(0x1043, 0x19b3, "ASUS F7Z", ALC662_FIXUP_ASUS_MODE1),
7079 SND_PCI_QUIRK(0x1043, 0x19c3, "ASUS F5Z/F6x", ALC662_FIXUP_ASUS_MODE2),
7080 SND_PCI_QUIRK(0x1043, 0x19e3, "ASUS NB", ALC662_FIXUP_ASUS_MODE1),
7081 SND_PCI_QUIRK(0x1043, 0x19f3, "ASUS NB", ALC662_FIXUP_ASUS_MODE4),
7082#endif
David Henningsson6cb3b702010-09-09 08:51:44 +02007083 {}
7084};
7085
Takashi Iwai1727a772013-01-10 09:52:52 +01007086static const struct hda_model_fixup alc662_fixup_models[] = {
Todd Broch6be79482010-12-07 16:51:05 -08007087 {.id = ALC272_FIXUP_MARIO, .name = "mario"},
Takashi Iwai53c334a2011-08-23 18:27:14 +02007088 {.id = ALC662_FIXUP_ASUS_MODE1, .name = "asus-mode1"},
7089 {.id = ALC662_FIXUP_ASUS_MODE2, .name = "asus-mode2"},
7090 {.id = ALC662_FIXUP_ASUS_MODE3, .name = "asus-mode3"},
7091 {.id = ALC662_FIXUP_ASUS_MODE4, .name = "asus-mode4"},
7092 {.id = ALC662_FIXUP_ASUS_MODE5, .name = "asus-mode5"},
7093 {.id = ALC662_FIXUP_ASUS_MODE6, .name = "asus-mode6"},
7094 {.id = ALC662_FIXUP_ASUS_MODE7, .name = "asus-mode7"},
7095 {.id = ALC662_FIXUP_ASUS_MODE8, .name = "asus-mode8"},
Takashi Iwai6e72aa52012-06-25 10:52:25 +02007096 {.id = ALC662_FIXUP_INV_DMIC, .name = "inv-dmic"},
David Henningssone32aa852013-06-17 11:04:02 +02007097 {.id = ALC668_FIXUP_DELL_MIC_NO_PRESENCE, .name = "dell-headset-multi"},
Todd Broch6be79482010-12-07 16:51:05 -08007098 {}
7099};
David Henningsson6cb3b702010-09-09 08:51:44 +02007100
Hui Wang532895c2014-05-29 15:59:19 +08007101static const struct snd_hda_pin_quirk alc662_pin_fixup_tbl[] = {
Kailang Yang78f4f7c2016-06-07 11:31:34 +08007102 SND_HDA_PIN_QUIRK(0x10ec0867, 0x1028, "Dell", ALC891_FIXUP_DELL_MIC_NO_PRESENCE,
7103 {0x17, 0x02211010},
7104 {0x18, 0x01a19030},
7105 {0x1a, 0x01813040},
7106 {0x21, 0x01014020}),
David Henningsson1f8b46c2015-05-12 14:38:15 +02007107 SND_HDA_PIN_QUIRK(0x10ec0662, 0x1028, "Dell", ALC662_FIXUP_DELL_MIC_NO_PRESENCE,
David Henningsson1f8b46c2015-05-12 14:38:15 +02007108 {0x14, 0x01014010},
David Henningsson1f8b46c2015-05-12 14:38:15 +02007109 {0x18, 0x01a19020},
David Henningsson1f8b46c2015-05-12 14:38:15 +02007110 {0x1a, 0x0181302f},
Hui Wang11580292015-08-03 11:03:49 +08007111 {0x1b, 0x0221401f}),
David Henningsson76c21322014-06-24 14:46:54 +02007112 SND_HDA_PIN_QUIRK(0x10ec0668, 0x1028, "Dell", ALC668_FIXUP_AUTO_MUTE,
7113 {0x12, 0x99a30130},
7114 {0x14, 0x90170110},
7115 {0x15, 0x0321101f},
Hui Wang11580292015-08-03 11:03:49 +08007116 {0x16, 0x03011020}),
David Henningsson76c21322014-06-24 14:46:54 +02007117 SND_HDA_PIN_QUIRK(0x10ec0668, 0x1028, "Dell", ALC668_FIXUP_AUTO_MUTE,
7118 {0x12, 0x99a30140},
7119 {0x14, 0x90170110},
7120 {0x15, 0x0321101f},
Hui Wang11580292015-08-03 11:03:49 +08007121 {0x16, 0x03011020}),
David Henningsson76c21322014-06-24 14:46:54 +02007122 SND_HDA_PIN_QUIRK(0x10ec0668, 0x1028, "Dell", ALC668_FIXUP_AUTO_MUTE,
7123 {0x12, 0x99a30150},
7124 {0x14, 0x90170110},
7125 {0x15, 0x0321101f},
Hui Wang11580292015-08-03 11:03:49 +08007126 {0x16, 0x03011020}),
David Henningsson76c21322014-06-24 14:46:54 +02007127 SND_HDA_PIN_QUIRK(0x10ec0668, 0x1028, "Dell", ALC668_FIXUP_AUTO_MUTE,
David Henningsson76c21322014-06-24 14:46:54 +02007128 {0x14, 0x90170110},
7129 {0x15, 0x0321101f},
Hui Wang11580292015-08-03 11:03:49 +08007130 {0x16, 0x03011020}),
David Henningsson76c21322014-06-24 14:46:54 +02007131 SND_HDA_PIN_QUIRK(0x10ec0668, 0x1028, "Dell XPS 15", ALC668_FIXUP_AUTO_MUTE,
7132 {0x12, 0x90a60130},
7133 {0x14, 0x90170110},
Hui Wang11580292015-08-03 11:03:49 +08007134 {0x15, 0x0321101f}),
Hui Wang532895c2014-05-29 15:59:19 +08007135 {}
7136};
7137
Takashi Iwai1d045db2011-07-07 18:23:21 +02007138/*
7139 */
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007140static int patch_alc662(struct hda_codec *codec)
7141{
7142 struct alc_spec *spec;
Takashi Iwai3de95172012-05-07 18:03:15 +02007143 int err;
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007144
Takashi Iwai3de95172012-05-07 18:03:15 +02007145 err = alc_alloc_spec(codec, 0x0b);
7146 if (err < 0)
7147 return err;
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007148
Takashi Iwai3de95172012-05-07 18:03:15 +02007149 spec = codec->spec;
Takashi Iwai1f0f4b82011-06-27 10:52:59 +02007150
Takashi Iwai225068a2015-05-29 10:42:14 +02007151 spec->shutup = alc_eapd_shutup;
7152
Takashi Iwai53c334a2011-08-23 18:27:14 +02007153 /* handle multiple HPs as is */
7154 spec->parse_flags = HDA_PINCFG_NO_HP_FIXUP;
7155
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +02007156 alc_fix_pll_init(codec, 0x20, 0x04, 15);
7157
Takashi Iwai7639a062015-03-03 10:07:24 +01007158 switch (codec->core.vendor_id) {
Kailang Yangf3f91852014-10-24 15:43:46 +08007159 case 0x10ec0668:
7160 spec->init_hook = alc668_restore_default_value;
7161 break;
Kailang Yangf3f91852014-10-24 15:43:46 +08007162 }
Kailang Yang8663ff72012-06-29 09:35:52 +02007163
Takashi Iwai1727a772013-01-10 09:52:52 +01007164 snd_hda_pick_fixup(codec, alc662_fixup_models,
Takashi Iwai8e5a0502012-06-21 15:49:33 +02007165 alc662_fixup_tbl, alc662_fixups);
Hui Wang532895c2014-05-29 15:59:19 +08007166 snd_hda_pick_pin_fixup(codec, alc662_pin_fixup_tbl, alc662_fixups);
Takashi Iwai1727a772013-01-10 09:52:52 +01007167 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
Takashi Iwai8e5a0502012-06-21 15:49:33 +02007168
7169 alc_auto_parse_customize_define(codec);
7170
Takashi Iwai7504b6c2013-03-18 11:25:51 +01007171 if (has_cdefine_beep(codec))
7172 spec->gen.beep_nid = 0x01;
7173
Takashi Iwai1bb7e432011-10-17 16:50:59 +02007174 if ((alc_get_coef0(codec) & (1 << 14)) &&
Takashi Iwai5100cd02014-02-15 10:03:19 +01007175 codec->bus->pci && codec->bus->pci->subsystem_vendor == 0x1025 &&
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02007176 spec->cdefine.platform_type == 1) {
Wei Yongjun6134b1a2013-04-18 11:12:59 +08007177 err = alc_codec_rename(codec, "ALC272X");
7178 if (err < 0)
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02007179 goto error;
Takashi Iwai20ca0c32011-10-17 16:00:35 +02007180 }
Kailang Yang274693f2009-12-03 10:07:50 +01007181
Takashi Iwaib9c51062011-08-24 18:08:07 +02007182 /* automatic parse from the BIOS config */
7183 err = alc662_parse_auto_config(codec);
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02007184 if (err < 0)
7185 goto error;
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007186
Takashi Iwai7504b6c2013-03-18 11:25:51 +01007187 if (!spec->gen.no_analog && spec->gen.beep_nid) {
Takashi Iwai7639a062015-03-03 10:07:24 +01007188 switch (codec->core.vendor_id) {
Kailang Yangda00c242010-03-19 11:23:45 +01007189 case 0x10ec0662:
7190 set_beep_amp(spec, 0x0b, 0x05, HDA_INPUT);
7191 break;
7192 case 0x10ec0272:
7193 case 0x10ec0663:
7194 case 0x10ec0665:
Kailang Yang9ad54542013-11-26 15:41:40 +08007195 case 0x10ec0668:
Kailang Yangda00c242010-03-19 11:23:45 +01007196 set_beep_amp(spec, 0x0b, 0x04, HDA_INPUT);
7197 break;
7198 case 0x10ec0273:
7199 set_beep_amp(spec, 0x0b, 0x03, HDA_INPUT);
7200 break;
7201 }
Kailang Yangcec27c82010-02-04 14:18:18 +01007202 }
Takashi Iwai2134ea42008-01-10 16:53:55 +01007203
Takashi Iwai1727a772013-01-10 09:52:52 +01007204 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
Takashi Iwai589876e2012-02-20 15:47:55 +01007205
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007206 return 0;
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007207
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02007208 error:
7209 alc_free(codec);
7210 return err;
Kailang Yangb478b992011-05-18 11:51:15 +02007211}
7212
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007213/*
Kailang Yangd1eb57f2010-06-23 16:25:26 +02007214 * ALC680 support
7215 */
Kailang Yangd1eb57f2010-06-23 16:25:26 +02007216
Kailang Yangd1eb57f2010-06-23 16:25:26 +02007217static int alc680_parse_auto_config(struct hda_codec *codec)
7218{
Takashi Iwai3e6179b2011-07-08 16:55:13 +02007219 return alc_parse_auto_config(codec, NULL, NULL);
Kailang Yangd1eb57f2010-06-23 16:25:26 +02007220}
7221
Kailang Yangd1eb57f2010-06-23 16:25:26 +02007222/*
Kailang Yangd1eb57f2010-06-23 16:25:26 +02007223 */
Kailang Yangd1eb57f2010-06-23 16:25:26 +02007224static int patch_alc680(struct hda_codec *codec)
7225{
Kailang Yangd1eb57f2010-06-23 16:25:26 +02007226 int err;
7227
Takashi Iwai1f0f4b82011-06-27 10:52:59 +02007228 /* ALC680 has no aa-loopback mixer */
Takashi Iwai3de95172012-05-07 18:03:15 +02007229 err = alc_alloc_spec(codec, 0);
7230 if (err < 0)
7231 return err;
Takashi Iwai1f0f4b82011-06-27 10:52:59 +02007232
Takashi Iwai1ebec5f2011-08-15 13:21:48 +02007233 /* automatic parse from the BIOS config */
7234 err = alc680_parse_auto_config(codec);
7235 if (err < 0) {
7236 alc_free(codec);
7237 return err;
Kailang Yangd1eb57f2010-06-23 16:25:26 +02007238 }
7239
Kailang Yangd1eb57f2010-06-23 16:25:26 +02007240 return 0;
7241}
7242
7243/*
Linus Torvalds1da177e2005-04-16 15:20:36 -07007244 * patch entries
7245 */
Takashi Iwaib9a94a92015-10-01 16:20:04 +02007246static const struct hda_device_id snd_hda_id_realtek[] = {
7247 HDA_CODEC_ENTRY(0x10ec0221, "ALC221", patch_alc269),
Kailang Yang42314302016-02-03 15:03:50 +08007248 HDA_CODEC_ENTRY(0x10ec0225, "ALC225", patch_alc269),
Takashi Iwaib9a94a92015-10-01 16:20:04 +02007249 HDA_CODEC_ENTRY(0x10ec0231, "ALC231", patch_alc269),
7250 HDA_CODEC_ENTRY(0x10ec0233, "ALC233", patch_alc269),
Kailang Yangdcd4f0d2016-05-04 15:50:18 +08007251 HDA_CODEC_ENTRY(0x10ec0234, "ALC234", patch_alc269),
Takashi Iwaib9a94a92015-10-01 16:20:04 +02007252 HDA_CODEC_ENTRY(0x10ec0235, "ALC233", patch_alc269),
Kailang Yang61ae3fb2017-10-20 15:06:34 +08007253 HDA_CODEC_ENTRY(0x10ec0236, "ALC236", patch_alc269),
Takashi Iwaib9a94a92015-10-01 16:20:04 +02007254 HDA_CODEC_ENTRY(0x10ec0255, "ALC255", patch_alc269),
7255 HDA_CODEC_ENTRY(0x10ec0256, "ALC256", patch_alc269),
7256 HDA_CODEC_ENTRY(0x10ec0260, "ALC260", patch_alc260),
7257 HDA_CODEC_ENTRY(0x10ec0262, "ALC262", patch_alc262),
7258 HDA_CODEC_ENTRY(0x10ec0267, "ALC267", patch_alc268),
7259 HDA_CODEC_ENTRY(0x10ec0268, "ALC268", patch_alc268),
7260 HDA_CODEC_ENTRY(0x10ec0269, "ALC269", patch_alc269),
7261 HDA_CODEC_ENTRY(0x10ec0270, "ALC270", patch_alc269),
7262 HDA_CODEC_ENTRY(0x10ec0272, "ALC272", patch_alc662),
Kailang Yangdcd4f0d2016-05-04 15:50:18 +08007263 HDA_CODEC_ENTRY(0x10ec0274, "ALC274", patch_alc269),
Takashi Iwaib9a94a92015-10-01 16:20:04 +02007264 HDA_CODEC_ENTRY(0x10ec0275, "ALC275", patch_alc269),
7265 HDA_CODEC_ENTRY(0x10ec0276, "ALC276", patch_alc269),
7266 HDA_CODEC_ENTRY(0x10ec0280, "ALC280", patch_alc269),
7267 HDA_CODEC_ENTRY(0x10ec0282, "ALC282", patch_alc269),
7268 HDA_CODEC_ENTRY(0x10ec0283, "ALC283", patch_alc269),
7269 HDA_CODEC_ENTRY(0x10ec0284, "ALC284", patch_alc269),
7270 HDA_CODEC_ENTRY(0x10ec0285, "ALC285", patch_alc269),
7271 HDA_CODEC_ENTRY(0x10ec0286, "ALC286", patch_alc269),
7272 HDA_CODEC_ENTRY(0x10ec0288, "ALC288", patch_alc269),
7273 HDA_CODEC_ENTRY(0x10ec0290, "ALC290", patch_alc269),
7274 HDA_CODEC_ENTRY(0x10ec0292, "ALC292", patch_alc269),
7275 HDA_CODEC_ENTRY(0x10ec0293, "ALC293", patch_alc269),
Kailang Yangdcd4f0d2016-05-04 15:50:18 +08007276 HDA_CODEC_ENTRY(0x10ec0294, "ALC294", patch_alc269),
Kailang Yang7d727862016-05-24 16:46:07 +08007277 HDA_CODEC_ENTRY(0x10ec0295, "ALC295", patch_alc269),
Takashi Iwaib9a94a92015-10-01 16:20:04 +02007278 HDA_CODEC_ENTRY(0x10ec0298, "ALC298", patch_alc269),
Kailang Yangf6e94c22017-01-04 14:49:07 +08007279 HDA_CODEC_ENTRY(0x10ec0299, "ALC299", patch_alc269),
Takashi Iwaib9a94a92015-10-01 16:20:04 +02007280 HDA_CODEC_REV_ENTRY(0x10ec0861, 0x100340, "ALC660", patch_alc861),
7281 HDA_CODEC_ENTRY(0x10ec0660, "ALC660-VD", patch_alc861vd),
7282 HDA_CODEC_ENTRY(0x10ec0861, "ALC861", patch_alc861),
7283 HDA_CODEC_ENTRY(0x10ec0862, "ALC861-VD", patch_alc861vd),
7284 HDA_CODEC_REV_ENTRY(0x10ec0662, 0x100002, "ALC662 rev2", patch_alc882),
7285 HDA_CODEC_REV_ENTRY(0x10ec0662, 0x100101, "ALC662 rev1", patch_alc662),
7286 HDA_CODEC_REV_ENTRY(0x10ec0662, 0x100300, "ALC662 rev3", patch_alc662),
7287 HDA_CODEC_ENTRY(0x10ec0663, "ALC663", patch_alc662),
7288 HDA_CODEC_ENTRY(0x10ec0665, "ALC665", patch_alc662),
7289 HDA_CODEC_ENTRY(0x10ec0667, "ALC667", patch_alc662),
7290 HDA_CODEC_ENTRY(0x10ec0668, "ALC668", patch_alc662),
7291 HDA_CODEC_ENTRY(0x10ec0670, "ALC670", patch_alc662),
7292 HDA_CODEC_ENTRY(0x10ec0671, "ALC671", patch_alc662),
7293 HDA_CODEC_ENTRY(0x10ec0680, "ALC680", patch_alc680),
Kailang Yang6fbae352016-05-30 16:44:20 +08007294 HDA_CODEC_ENTRY(0x10ec0700, "ALC700", patch_alc269),
7295 HDA_CODEC_ENTRY(0x10ec0701, "ALC701", patch_alc269),
7296 HDA_CODEC_ENTRY(0x10ec0703, "ALC703", patch_alc269),
Kailang Yang78f4f7c2016-06-07 11:31:34 +08007297 HDA_CODEC_ENTRY(0x10ec0867, "ALC891", patch_alc662),
Takashi Iwaib9a94a92015-10-01 16:20:04 +02007298 HDA_CODEC_ENTRY(0x10ec0880, "ALC880", patch_alc880),
7299 HDA_CODEC_ENTRY(0x10ec0882, "ALC882", patch_alc882),
7300 HDA_CODEC_ENTRY(0x10ec0883, "ALC883", patch_alc882),
7301 HDA_CODEC_REV_ENTRY(0x10ec0885, 0x100101, "ALC889A", patch_alc882),
7302 HDA_CODEC_REV_ENTRY(0x10ec0885, 0x100103, "ALC889A", patch_alc882),
7303 HDA_CODEC_ENTRY(0x10ec0885, "ALC885", patch_alc882),
7304 HDA_CODEC_ENTRY(0x10ec0887, "ALC887", patch_alc882),
7305 HDA_CODEC_REV_ENTRY(0x10ec0888, 0x100101, "ALC1200", patch_alc882),
7306 HDA_CODEC_ENTRY(0x10ec0888, "ALC888", patch_alc882),
7307 HDA_CODEC_ENTRY(0x10ec0889, "ALC889", patch_alc882),
7308 HDA_CODEC_ENTRY(0x10ec0892, "ALC892", patch_alc662),
7309 HDA_CODEC_ENTRY(0x10ec0899, "ALC898", patch_alc882),
7310 HDA_CODEC_ENTRY(0x10ec0900, "ALC1150", patch_alc882),
Linus Torvalds1da177e2005-04-16 15:20:36 -07007311 {} /* terminator */
7312};
Takashi Iwaib9a94a92015-10-01 16:20:04 +02007313MODULE_DEVICE_TABLE(hdaudio, snd_hda_id_realtek);
Takashi Iwai1289e9e2008-11-27 15:47:11 +01007314
7315MODULE_LICENSE("GPL");
7316MODULE_DESCRIPTION("Realtek HD-audio codec");
7317
Takashi Iwaid8a766a2015-02-17 15:25:37 +01007318static struct hda_codec_driver realtek_driver = {
Takashi Iwaib9a94a92015-10-01 16:20:04 +02007319 .id = snd_hda_id_realtek,
Takashi Iwai1289e9e2008-11-27 15:47:11 +01007320};
7321
Takashi Iwaid8a766a2015-02-17 15:25:37 +01007322module_hda_codec_driver(realtek_driver);