blob: 2a50e580aa561775bf4c4e797d402cbc3b6f77c8 [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>
Pierre-Louis Bossartbe57bff2018-08-22 15:24:57 -050035#include <sound/hda_codec.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070036#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
Takashi Iwai4a79ba32009-04-22 16:31:35 +020044/* extra amp-initialization sequence types */
45enum {
Takashi Iwai1c76aa52018-06-21 16:37:54 +020046 ALC_INIT_UNDEFINED,
Takashi Iwai4a79ba32009-04-22 16:31:35 +020047 ALC_INIT_NONE,
48 ALC_INIT_DEFAULT,
Takashi Iwai4a79ba32009-04-22 16:31:35 +020049};
50
David Henningsson73bdd592013-04-15 15:44:14 +020051enum {
52 ALC_HEADSET_MODE_UNKNOWN,
53 ALC_HEADSET_MODE_UNPLUGGED,
54 ALC_HEADSET_MODE_HEADSET,
55 ALC_HEADSET_MODE_MIC,
56 ALC_HEADSET_MODE_HEADPHONE,
57};
58
59enum {
60 ALC_HEADSET_TYPE_UNKNOWN,
61 ALC_HEADSET_TYPE_CTIA,
62 ALC_HEADSET_TYPE_OMTP,
63};
64
Hui Wangc7b60a82015-12-28 11:35:25 +080065enum {
66 ALC_KEY_MICMUTE_INDEX,
67};
68
Kailang Yangda00c242010-03-19 11:23:45 +010069struct alc_customize_define {
70 unsigned int sku_cfg;
71 unsigned char port_connectivity;
72 unsigned char check_sum;
73 unsigned char customization;
74 unsigned char external_amp;
75 unsigned int enable_pcbeep:1;
76 unsigned int platform_type:1;
77 unsigned int swap:1;
78 unsigned int override:1;
David Henningsson90622912010-10-14 14:50:18 +020079 unsigned int fixup:1; /* Means that this sku is set by driver, not read from hw */
Kailang Yangda00c242010-03-19 11:23:45 +010080};
81
Linus Torvalds1da177e2005-04-16 15:20:36 -070082struct alc_spec {
Takashi Iwai08c189f2012-12-19 15:22:24 +010083 struct hda_gen_spec gen; /* must be at head */
Takashi Iwai23d30f22012-05-07 17:17:32 +020084
Linus Torvalds1da177e2005-04-16 15:20:36 -070085 /* codec parameterization */
Kailang Yangda00c242010-03-19 11:23:45 +010086 struct alc_customize_define cdefine;
Takashi Iwai08c189f2012-12-19 15:22:24 +010087 unsigned int parse_flags; /* flag for snd_hda_parse_pin_defcfg() */
88
Takashi Iwai5579cd62018-06-19 22:22:41 +020089 /* GPIO bits */
90 unsigned int gpio_mask;
91 unsigned int gpio_dir;
92 unsigned int gpio_data;
Takashi Iwai215c8502018-06-19 22:34:26 +020093 bool gpio_write_delay; /* add a delay before writing gpio_data */
Takashi Iwai5579cd62018-06-19 22:22:41 +020094
Takashi Iwai08fb0d02013-01-10 17:33:58 +010095 /* mute LED for HP laptops, see alc269_fixup_mic_mute_hook() */
96 int mute_led_polarity;
97 hda_nid_t mute_led_nid;
Kailang Yang9c5dc3b2014-06-19 16:11:36 +080098 hda_nid_t cap_mute_led_nid;
Takashi Iwai08fb0d02013-01-10 17:33:58 +010099
Takashi Iwai0f32fd192014-11-19 12:16:14 +0100100 unsigned int gpio_mute_led_mask;
101 unsigned int gpio_mic_led_mask;
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +0100102
David Henningsson73bdd592013-04-15 15:44:14 +0200103 hda_nid_t headset_mic_pin;
104 hda_nid_t headphone_mic_pin;
105 int current_headset_mode;
106 int current_headset_type;
107
Takashi Iwaiae6b8132006-03-03 16:47:17 +0100108 /* hooks */
109 void (*init_hook)(struct hda_codec *codec);
Takashi Iwai83012a72012-08-24 18:38:08 +0200110#ifdef CONFIG_PM
Daniel T Chenc97259d2009-12-27 18:52:08 -0500111 void (*power_hook)(struct hda_codec *codec);
Hector Martinf5de24b2009-12-20 22:51:31 +0100112#endif
Takashi Iwai1c7161532011-04-07 10:37:16 +0200113 void (*shutup)(struct hda_codec *codec);
Takashi Iwai70a09762015-12-15 14:59:58 +0100114 void (*reboot_notify)(struct hda_codec *codec);
Takashi Iwaid922b512011-04-28 12:18:53 +0200115
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200116 int init_amp;
Takashi Iwaid433a672010-09-20 15:11:54 +0200117 int codec_variant; /* flag for other variants */
Kailang Yang97a26572013-11-29 00:35:26 -0500118 unsigned int has_alc5505_dsp:1;
119 unsigned int no_depop_delay:1;
Kailang Yang693abe12019-01-29 15:38:21 +0800120 unsigned int done_hp_init:1;
Takashi Iwaic0ca5ec2019-02-20 16:15:45 +0100121 unsigned int no_shutup_pins:1;
Kailang Yangd3ba58b2019-05-06 15:09:42 +0800122 unsigned int ultra_low_power:1;
Takashi Iwaie64f14f2009-01-20 18:32:55 +0100123
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +0200124 /* for PLL fix */
125 hda_nid_t pll_nid;
126 unsigned int pll_coef_idx, pll_coef_bit;
Takashi Iwai1bb7e432011-10-17 16:50:59 +0200127 unsigned int coef0;
David Henningsson33f4acd2015-01-07 15:50:13 +0100128 struct input_dev *kb_dev;
Hui Wangc7b60a82015-12-28 11:35:25 +0800129 u8 alc_mute_keycode_map[1];
Kailang Yangdf694da2005-12-05 19:42:22 +0100130};
131
Takashi Iwai23f0c042009-02-26 13:03:58 +0100132/*
Takashi Iwaif2a227c2014-08-18 13:35:22 +0200133 * COEF access helper functions
134 */
135
136static int alc_read_coefex_idx(struct hda_codec *codec, hda_nid_t nid,
137 unsigned int coef_idx)
138{
139 unsigned int val;
140
141 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_COEF_INDEX, coef_idx);
142 val = snd_hda_codec_read(codec, nid, 0, AC_VERB_GET_PROC_COEF, 0);
143 return val;
144}
145
146#define alc_read_coef_idx(codec, coef_idx) \
147 alc_read_coefex_idx(codec, 0x20, coef_idx)
148
149static void alc_write_coefex_idx(struct hda_codec *codec, hda_nid_t nid,
150 unsigned int coef_idx, unsigned int coef_val)
151{
152 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_COEF_INDEX, coef_idx);
153 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_PROC_COEF, coef_val);
154}
155
156#define alc_write_coef_idx(codec, coef_idx, coef_val) \
157 alc_write_coefex_idx(codec, 0x20, coef_idx, coef_val)
158
Takashi Iwai98b24882014-08-18 13:47:50 +0200159static void alc_update_coefex_idx(struct hda_codec *codec, hda_nid_t nid,
160 unsigned int coef_idx, unsigned int mask,
161 unsigned int bits_set)
162{
163 unsigned int val = alc_read_coefex_idx(codec, nid, coef_idx);
164
165 if (val != -1)
166 alc_write_coefex_idx(codec, nid, coef_idx,
167 (val & ~mask) | bits_set);
168}
169
170#define alc_update_coef_idx(codec, coef_idx, mask, bits_set) \
171 alc_update_coefex_idx(codec, 0x20, coef_idx, mask, bits_set)
172
Takashi Iwaif2a227c2014-08-18 13:35:22 +0200173/* a special bypass for COEF 0; read the cached value at the second time */
174static unsigned int alc_get_coef0(struct hda_codec *codec)
175{
176 struct alc_spec *spec = codec->spec;
177
178 if (!spec->coef0)
179 spec->coef0 = alc_read_coef_idx(codec, 0);
180 return spec->coef0;
181}
182
Takashi Iwai54db6c32014-08-18 15:11:19 +0200183/* coef writes/updates batch */
184struct coef_fw {
185 unsigned char nid;
186 unsigned char idx;
187 unsigned short mask;
188 unsigned short val;
189};
190
191#define UPDATE_COEFEX(_nid, _idx, _mask, _val) \
192 { .nid = (_nid), .idx = (_idx), .mask = (_mask), .val = (_val) }
193#define WRITE_COEFEX(_nid, _idx, _val) UPDATE_COEFEX(_nid, _idx, -1, _val)
194#define WRITE_COEF(_idx, _val) WRITE_COEFEX(0x20, _idx, _val)
195#define UPDATE_COEF(_idx, _mask, _val) UPDATE_COEFEX(0x20, _idx, _mask, _val)
196
197static void alc_process_coef_fw(struct hda_codec *codec,
198 const struct coef_fw *fw)
199{
200 for (; fw->nid; fw++) {
201 if (fw->mask == (unsigned short)-1)
202 alc_write_coefex_idx(codec, fw->nid, fw->idx, fw->val);
203 else
204 alc_update_coefex_idx(codec, fw->nid, fw->idx,
205 fw->mask, fw->val);
206 }
207}
208
Takashi Iwaif2a227c2014-08-18 13:35:22 +0200209/*
Takashi Iwai1d045db2011-07-07 18:23:21 +0200210 * GPIO setup tables, used in initialization
Kailang Yangdf694da2005-12-05 19:42:22 +0100211 */
Takashi Iwai5579cd62018-06-19 22:22:41 +0200212
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200213/* Enable GPIO mask and set output */
Takashi Iwai5579cd62018-06-19 22:22:41 +0200214static void alc_setup_gpio(struct hda_codec *codec, unsigned int mask)
215{
216 struct alc_spec *spec = codec->spec;
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200217
Takashi Iwai5579cd62018-06-19 22:22:41 +0200218 spec->gpio_mask |= mask;
219 spec->gpio_dir |= mask;
220 spec->gpio_data |= mask;
221}
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200222
Takashi Iwai5579cd62018-06-19 22:22:41 +0200223static void alc_write_gpio_data(struct hda_codec *codec)
224{
225 struct alc_spec *spec = codec->spec;
226
227 snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DATA,
228 spec->gpio_data);
229}
230
Takashi Iwaiaaf312d2018-06-19 22:28:22 +0200231static void alc_update_gpio_data(struct hda_codec *codec, unsigned int mask,
232 bool on)
233{
234 struct alc_spec *spec = codec->spec;
235 unsigned int oldval = spec->gpio_data;
236
237 if (on)
238 spec->gpio_data |= mask;
239 else
240 spec->gpio_data &= ~mask;
241 if (oldval != spec->gpio_data)
242 alc_write_gpio_data(codec);
243}
244
Takashi Iwai5579cd62018-06-19 22:22:41 +0200245static void alc_write_gpio(struct hda_codec *codec)
246{
247 struct alc_spec *spec = codec->spec;
248
249 if (!spec->gpio_mask)
250 return;
251
252 snd_hda_codec_write(codec, codec->core.afg, 0,
253 AC_VERB_SET_GPIO_MASK, spec->gpio_mask);
254 snd_hda_codec_write(codec, codec->core.afg, 0,
255 AC_VERB_SET_GPIO_DIRECTION, spec->gpio_dir);
Takashi Iwai215c8502018-06-19 22:34:26 +0200256 if (spec->gpio_write_delay)
257 msleep(1);
Takashi Iwai5579cd62018-06-19 22:22:41 +0200258 alc_write_gpio_data(codec);
259}
260
261static void alc_fixup_gpio(struct hda_codec *codec, int action,
262 unsigned int mask)
263{
264 if (action == HDA_FIXUP_ACT_PRE_PROBE)
265 alc_setup_gpio(codec, mask);
266}
267
268static void alc_fixup_gpio1(struct hda_codec *codec,
269 const struct hda_fixup *fix, int action)
270{
271 alc_fixup_gpio(codec, action, 0x01);
272}
273
274static void alc_fixup_gpio2(struct hda_codec *codec,
275 const struct hda_fixup *fix, int action)
276{
277 alc_fixup_gpio(codec, action, 0x02);
278}
279
280static void alc_fixup_gpio3(struct hda_codec *codec,
281 const struct hda_fixup *fix, int action)
282{
283 alc_fixup_gpio(codec, action, 0x03);
284}
Kailang Yangbdd148a2007-05-08 15:19:08 +0200285
Takashi Iwaiae065f12018-06-19 23:00:03 +0200286static void alc_fixup_gpio4(struct hda_codec *codec,
287 const struct hda_fixup *fix, int action)
288{
289 alc_fixup_gpio(codec, action, 0x04);
290}
291
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +0200292/*
293 * Fix hardware PLL issue
294 * On some codecs, the analog PLL gating control must be off while
295 * the default value is 1.
296 */
297static void alc_fix_pll(struct hda_codec *codec)
298{
299 struct alc_spec *spec = codec->spec;
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +0200300
Takashi Iwai98b24882014-08-18 13:47:50 +0200301 if (spec->pll_nid)
302 alc_update_coefex_idx(codec, spec->pll_nid, spec->pll_coef_idx,
303 1 << spec->pll_coef_bit, 0);
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +0200304}
305
306static void alc_fix_pll_init(struct hda_codec *codec, hda_nid_t nid,
307 unsigned int coef_idx, unsigned int coef_bit)
308{
309 struct alc_spec *spec = codec->spec;
310 spec->pll_nid = nid;
311 spec->pll_coef_idx = coef_idx;
312 spec->pll_coef_bit = coef_bit;
313 alc_fix_pll(codec);
314}
315
Takashi Iwaicf5a2272012-02-20 16:31:07 +0100316/* update the master volume per volume-knob's unsol event */
Takashi Iwai1a4f69d2014-09-11 15:22:46 +0200317static void alc_update_knob_master(struct hda_codec *codec,
318 struct hda_jack_callback *jack)
Takashi Iwaicf5a2272012-02-20 16:31:07 +0100319{
320 unsigned int val;
321 struct snd_kcontrol *kctl;
322 struct snd_ctl_elem_value *uctl;
323
324 kctl = snd_hda_find_mixer_ctl(codec, "Master Playback Volume");
325 if (!kctl)
326 return;
327 uctl = kzalloc(sizeof(*uctl), GFP_KERNEL);
328 if (!uctl)
329 return;
Takashi Iwai2ebab402016-02-09 10:23:52 +0100330 val = snd_hda_codec_read(codec, jack->nid, 0,
Takashi Iwaicf5a2272012-02-20 16:31:07 +0100331 AC_VERB_GET_VOLUME_KNOB_CONTROL, 0);
332 val &= HDA_AMP_VOLMASK;
333 uctl->value.integer.value[0] = val;
334 uctl->value.integer.value[1] = val;
335 kctl->put(kctl, uctl);
336 kfree(uctl);
337}
338
David Henningsson29adc4b2012-09-25 11:31:00 +0200339static void alc880_unsol_event(struct hda_codec *codec, unsigned int res)
Takashi Iwaif21d78e2012-01-19 12:10:29 +0100340{
David Henningsson29adc4b2012-09-25 11:31:00 +0200341 /* For some reason, the res given from ALC880 is broken.
342 Here we adjust it properly. */
343 snd_hda_jack_unsol_event(codec, res >> 2);
Takashi Iwaif21d78e2012-01-19 12:10:29 +0100344}
345
Kailang Yang394c97f2014-11-12 17:38:08 +0800346/* Change EAPD to verb control */
347static void alc_fill_eapd_coef(struct hda_codec *codec)
348{
349 int coef;
350
351 coef = alc_get_coef0(codec);
352
Takashi Iwai7639a062015-03-03 10:07:24 +0100353 switch (codec->core.vendor_id) {
Kailang Yang394c97f2014-11-12 17:38:08 +0800354 case 0x10ec0262:
355 alc_update_coef_idx(codec, 0x7, 0, 1<<5);
356 break;
357 case 0x10ec0267:
358 case 0x10ec0268:
359 alc_update_coef_idx(codec, 0x7, 0, 1<<13);
360 break;
361 case 0x10ec0269:
362 if ((coef & 0x00f0) == 0x0010)
363 alc_update_coef_idx(codec, 0xd, 0, 1<<14);
364 if ((coef & 0x00f0) == 0x0020)
365 alc_update_coef_idx(codec, 0x4, 1<<15, 0);
366 if ((coef & 0x00f0) == 0x0030)
367 alc_update_coef_idx(codec, 0x10, 1<<9, 0);
368 break;
369 case 0x10ec0280:
370 case 0x10ec0284:
371 case 0x10ec0290:
372 case 0x10ec0292:
373 alc_update_coef_idx(codec, 0x4, 1<<15, 0);
374 break;
Kailang Yang42314302016-02-03 15:03:50 +0800375 case 0x10ec0225:
Takashi Iwai44be77c2017-12-27 08:53:59 +0100376 case 0x10ec0295:
377 case 0x10ec0299:
378 alc_update_coef_idx(codec, 0x67, 0xf000, 0x3000);
379 /* fallthrough */
380 case 0x10ec0215:
Kailang Yang394c97f2014-11-12 17:38:08 +0800381 case 0x10ec0233:
Kailang Yangea04a1d2018-04-25 15:31:52 +0800382 case 0x10ec0235:
Kailang Yang736f20a2017-10-20 15:06:34 +0800383 case 0x10ec0236:
Kailang Yang394c97f2014-11-12 17:38:08 +0800384 case 0x10ec0255:
Kailang Yang4344aec2014-12-17 17:39:05 +0800385 case 0x10ec0256:
Kailang Yangf429e7e2017-12-05 15:38:24 +0800386 case 0x10ec0257:
Kailang Yang394c97f2014-11-12 17:38:08 +0800387 case 0x10ec0282:
388 case 0x10ec0283:
389 case 0x10ec0286:
390 case 0x10ec0288:
Kailang Yang0a6f0602017-06-30 16:00:48 +0800391 case 0x10ec0285:
Kailang Yang506b62c2014-12-18 17:07:44 +0800392 case 0x10ec0298:
Kailang Yang0a6f0602017-06-30 16:00:48 +0800393 case 0x10ec0289:
Kailang Yang1078bef2018-11-08 16:36:15 +0800394 case 0x10ec0300:
Kailang Yang394c97f2014-11-12 17:38:08 +0800395 alc_update_coef_idx(codec, 0x10, 1<<9, 0);
396 break;
Kailang Yang3aabf942017-11-08 15:28:33 +0800397 case 0x10ec0275:
398 alc_update_coef_idx(codec, 0xe, 0, 1<<0);
399 break;
Kailang Yang394c97f2014-11-12 17:38:08 +0800400 case 0x10ec0293:
401 alc_update_coef_idx(codec, 0xa, 1<<13, 0);
402 break;
Kailang Yangdcd4f0d2016-05-04 15:50:18 +0800403 case 0x10ec0234:
404 case 0x10ec0274:
405 case 0x10ec0294:
Kailang Yang6fbae352016-05-30 16:44:20 +0800406 case 0x10ec0700:
407 case 0x10ec0701:
408 case 0x10ec0703:
Kailang Yangdcd4f0d2016-05-04 15:50:18 +0800409 alc_update_coef_idx(codec, 0x10, 1<<15, 0);
410 break;
Kailang Yang394c97f2014-11-12 17:38:08 +0800411 case 0x10ec0662:
412 if ((coef & 0x00f0) == 0x0030)
413 alc_update_coef_idx(codec, 0x4, 1<<10, 0); /* EAPD Ctrl */
414 break;
415 case 0x10ec0272:
416 case 0x10ec0273:
417 case 0x10ec0663:
418 case 0x10ec0665:
419 case 0x10ec0670:
420 case 0x10ec0671:
421 case 0x10ec0672:
422 alc_update_coef_idx(codec, 0xd, 0, 1<<14); /* EAPD Ctrl */
423 break;
424 case 0x10ec0668:
425 alc_update_coef_idx(codec, 0x7, 3<<13, 0);
426 break;
427 case 0x10ec0867:
428 alc_update_coef_idx(codec, 0x4, 1<<10, 0);
429 break;
430 case 0x10ec0888:
431 if ((coef & 0x00f0) == 0x0020 || (coef & 0x00f0) == 0x0030)
432 alc_update_coef_idx(codec, 0x7, 1<<5, 0);
433 break;
434 case 0x10ec0892:
435 alc_update_coef_idx(codec, 0x7, 1<<5, 0);
436 break;
437 case 0x10ec0899:
438 case 0x10ec0900:
Kailang Yang65553b12017-07-11 15:15:47 +0800439 case 0x10ec1168:
Kailang Yanga535ad52017-01-16 16:59:26 +0800440 case 0x10ec1220:
Kailang Yang394c97f2014-11-12 17:38:08 +0800441 alc_update_coef_idx(codec, 0x7, 1<<1, 0);
442 break;
443 }
444}
445
Kailang Yangf9423e72008-05-27 12:32:25 +0200446/* additional initialization for ALC888 variants */
447static void alc888_coef_init(struct hda_codec *codec)
448{
Kailang Yang1df88742014-10-29 16:10:13 +0800449 switch (alc_get_coef0(codec) & 0x00f0) {
450 /* alc888-VA */
451 case 0x00:
452 /* alc888-VB */
453 case 0x10:
454 alc_update_coef_idx(codec, 7, 0, 0x2030); /* Turn EAPD to High */
455 break;
456 }
Jaroslav Kysela87a8c372009-07-23 10:58:29 +0200457}
458
Takashi Iwai3fb4a502010-01-19 15:46:37 +0100459/* turn on/off EAPD control (only if available) */
460static void set_eapd(struct hda_codec *codec, hda_nid_t nid, int on)
461{
462 if (get_wcaps_type(get_wcaps(codec, nid)) != AC_WID_PIN)
463 return;
464 if (snd_hda_query_pin_caps(codec, nid) & AC_PINCAP_EAPD)
465 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_EAPD_BTLENABLE,
466 on ? 2 : 0);
467}
468
Takashi Iwai691f1fc2011-04-07 10:31:43 +0200469/* turn on/off EAPD controls of the codec */
470static void alc_auto_setup_eapd(struct hda_codec *codec, bool on)
471{
472 /* We currently only handle front, HP */
Takashi Iwai39fa84e2011-06-27 15:28:57 +0200473 static hda_nid_t pins[] = {
Hui Wangaf95b412015-03-26 17:14:55 +0800474 0x0f, 0x10, 0x14, 0x15, 0x17, 0
Takashi Iwai39fa84e2011-06-27 15:28:57 +0200475 };
476 hda_nid_t *p;
477 for (p = pins; *p; p++)
478 set_eapd(codec, *p, on);
Takashi Iwai691f1fc2011-04-07 10:31:43 +0200479}
480
Kailang Yangdad31972019-05-10 16:28:57 +0800481static int find_ext_mic_pin(struct hda_codec *codec);
482
483static void alc_headset_mic_no_shutup(struct hda_codec *codec)
484{
485 const struct hda_pincfg *pin;
486 int mic_pin = find_ext_mic_pin(codec);
487 int i;
488
489 /* don't shut up pins when unloading the driver; otherwise it breaks
490 * the default pin setup at the next load of the driver
491 */
492 if (codec->bus->shutdown)
493 return;
494
495 snd_array_for_each(&codec->init_pins, i, pin) {
496 /* use read here for syncing after issuing each verb */
497 if (pin->nid != mic_pin)
498 snd_hda_codec_read(codec, pin->nid, 0,
499 AC_VERB_SET_PIN_WIDGET_CONTROL, 0);
500 }
501
502 codec->pins_shutup = 1;
503}
504
Takashi Iwaic0ca5ec2019-02-20 16:15:45 +0100505static void alc_shutup_pins(struct hda_codec *codec)
506{
507 struct alc_spec *spec = codec->spec;
508
Kailang Yangdad31972019-05-10 16:28:57 +0800509 switch (codec->core.vendor_id) {
510 case 0x10ec0286:
511 case 0x10ec0288:
512 case 0x10ec0298:
513 alc_headset_mic_no_shutup(codec);
514 break;
515 default:
516 if (!spec->no_shutup_pins)
517 snd_hda_shutup_pins(codec);
518 break;
519 }
Takashi Iwaic0ca5ec2019-02-20 16:15:45 +0100520}
521
Takashi Iwai1c7161532011-04-07 10:37:16 +0200522/* generic shutup callback;
Lars-Peter Clausen4ce8e6a2016-11-08 15:48:06 +0100523 * just turning off EAPD and a little pause for avoiding pop-noise
Takashi Iwai1c7161532011-04-07 10:37:16 +0200524 */
525static void alc_eapd_shutup(struct hda_codec *codec)
526{
Kailang Yang97a26572013-11-29 00:35:26 -0500527 struct alc_spec *spec = codec->spec;
528
Takashi Iwai1c7161532011-04-07 10:37:16 +0200529 alc_auto_setup_eapd(codec, false);
Kailang Yang97a26572013-11-29 00:35:26 -0500530 if (!spec->no_depop_delay)
531 msleep(200);
Takashi Iwaic0ca5ec2019-02-20 16:15:45 +0100532 alc_shutup_pins(codec);
Takashi Iwai1c7161532011-04-07 10:37:16 +0200533}
534
Takashi Iwai1d045db2011-07-07 18:23:21 +0200535/* generic EAPD initialization */
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200536static void alc_auto_init_amp(struct hda_codec *codec, int type)
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200537{
Takashi Iwai39fa84e2011-06-27 15:28:57 +0200538 alc_auto_setup_eapd(codec, true);
Takashi Iwai5579cd62018-06-19 22:22:41 +0200539 alc_write_gpio(codec);
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200540 switch (type) {
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200541 case ALC_INIT_DEFAULT:
Takashi Iwai7639a062015-03-03 10:07:24 +0100542 switch (codec->core.vendor_id) {
Kailang Yangc9b58002007-10-16 14:30:01 +0200543 case 0x10ec0260:
Takashi Iwai98b24882014-08-18 13:47:50 +0200544 alc_update_coefex_idx(codec, 0x1a, 7, 0, 0x2010);
Kailang Yangc9b58002007-10-16 14:30:01 +0200545 break;
Kailang Yangc9b58002007-10-16 14:30:01 +0200546 case 0x10ec0880:
547 case 0x10ec0882:
548 case 0x10ec0883:
549 case 0x10ec0885:
Kailang Yang1df88742014-10-29 16:10:13 +0800550 alc_update_coef_idx(codec, 7, 0, 0x2030);
Kailang Yangc9b58002007-10-16 14:30:01 +0200551 break;
Kailang Yangf9423e72008-05-27 12:32:25 +0200552 case 0x10ec0888:
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200553 alc888_coef_init(codec);
Kailang Yangf9423e72008-05-27 12:32:25 +0200554 break;
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200555 }
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200556 break;
557 }
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200558}
Kailang Yangea1fb292008-08-26 12:58:38 +0200559
Takashi Iwai35a39f92019-02-01 11:19:50 +0100560/* get a primary headphone pin if available */
561static hda_nid_t alc_get_hp_pin(struct alc_spec *spec)
562{
563 if (spec->gen.autocfg.hp_pins[0])
564 return spec->gen.autocfg.hp_pins[0];
565 if (spec->gen.autocfg.line_out_type == AC_JACK_HP_OUT)
566 return spec->gen.autocfg.line_out_pins[0];
567 return 0;
568}
Takashi Iwai1d045db2011-07-07 18:23:21 +0200569
570/*
571 * Realtek SSID verification
572 */
573
David Henningsson90622912010-10-14 14:50:18 +0200574/* Could be any non-zero and even value. When used as fixup, tells
575 * the driver to ignore any present sku defines.
576 */
577#define ALC_FIXUP_SKU_IGNORE (2)
578
Takashi Iwai23d30f22012-05-07 17:17:32 +0200579static void alc_fixup_sku_ignore(struct hda_codec *codec,
580 const struct hda_fixup *fix, int action)
581{
582 struct alc_spec *spec = codec->spec;
583 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
584 spec->cdefine.fixup = 1;
585 spec->cdefine.sku_cfg = ALC_FIXUP_SKU_IGNORE;
586 }
587}
588
Mengdong Linb5c66112013-11-29 00:35:35 -0500589static void alc_fixup_no_depop_delay(struct hda_codec *codec,
590 const struct hda_fixup *fix, int action)
591{
592 struct alc_spec *spec = codec->spec;
593
Mengdong Lin84d2dc32013-12-02 22:26:03 -0500594 if (action == HDA_FIXUP_ACT_PROBE) {
Mengdong Linb5c66112013-11-29 00:35:35 -0500595 spec->no_depop_delay = 1;
Mengdong Lin84d2dc32013-12-02 22:26:03 -0500596 codec->depop_delay = 0;
597 }
Mengdong Linb5c66112013-11-29 00:35:35 -0500598}
599
Kailang Yangda00c242010-03-19 11:23:45 +0100600static int alc_auto_parse_customize_define(struct hda_codec *codec)
601{
602 unsigned int ass, tmp, i;
Takashi Iwai7fb56222010-03-22 17:09:47 +0100603 unsigned nid = 0;
Kailang Yangda00c242010-03-19 11:23:45 +0100604 struct alc_spec *spec = codec->spec;
605
Takashi Iwaib6cbe512010-07-28 17:43:36 +0200606 spec->cdefine.enable_pcbeep = 1; /* assume always enabled */
607
David Henningsson90622912010-10-14 14:50:18 +0200608 if (spec->cdefine.fixup) {
609 ass = spec->cdefine.sku_cfg;
610 if (ass == ALC_FIXUP_SKU_IGNORE)
611 return -1;
612 goto do_sku;
613 }
614
Takashi Iwai5100cd02014-02-15 10:03:19 +0100615 if (!codec->bus->pci)
616 return -1;
Takashi Iwai7639a062015-03-03 10:07:24 +0100617 ass = codec->core.subsystem_id & 0xffff;
Takashi Iwaib6cbe512010-07-28 17:43:36 +0200618 if (ass != codec->bus->pci->subsystem_device && (ass & 1))
Kailang Yangda00c242010-03-19 11:23:45 +0100619 goto do_sku;
620
621 nid = 0x1d;
Takashi Iwai7639a062015-03-03 10:07:24 +0100622 if (codec->core.vendor_id == 0x10ec0260)
Kailang Yangda00c242010-03-19 11:23:45 +0100623 nid = 0x17;
624 ass = snd_hda_codec_get_pincfg(codec, nid);
625
626 if (!(ass & 1)) {
Takashi Iwai4e76a882014-02-25 12:21:03 +0100627 codec_info(codec, "%s: SKU not ready 0x%08x\n",
Takashi Iwai7639a062015-03-03 10:07:24 +0100628 codec->core.chip_name, ass);
Kailang Yangda00c242010-03-19 11:23:45 +0100629 return -1;
630 }
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 -1;
640
641 spec->cdefine.port_connectivity = ass >> 30;
642 spec->cdefine.enable_pcbeep = (ass & 0x100000) >> 20;
643 spec->cdefine.check_sum = (ass >> 16) & 0xf;
644 spec->cdefine.customization = ass >> 8;
645do_sku:
646 spec->cdefine.sku_cfg = ass;
647 spec->cdefine.external_amp = (ass & 0x38) >> 3;
648 spec->cdefine.platform_type = (ass & 0x4) >> 2;
649 spec->cdefine.swap = (ass & 0x2) >> 1;
650 spec->cdefine.override = ass & 0x1;
651
Takashi Iwai4e76a882014-02-25 12:21:03 +0100652 codec_dbg(codec, "SKU: Nid=0x%x sku_cfg=0x%08x\n",
Kailang Yangda00c242010-03-19 11:23:45 +0100653 nid, spec->cdefine.sku_cfg);
Takashi Iwai4e76a882014-02-25 12:21:03 +0100654 codec_dbg(codec, "SKU: port_connectivity=0x%x\n",
Kailang Yangda00c242010-03-19 11:23:45 +0100655 spec->cdefine.port_connectivity);
Takashi Iwai4e76a882014-02-25 12:21:03 +0100656 codec_dbg(codec, "SKU: enable_pcbeep=0x%x\n", spec->cdefine.enable_pcbeep);
657 codec_dbg(codec, "SKU: check_sum=0x%08x\n", spec->cdefine.check_sum);
658 codec_dbg(codec, "SKU: customization=0x%08x\n", spec->cdefine.customization);
659 codec_dbg(codec, "SKU: external_amp=0x%x\n", spec->cdefine.external_amp);
660 codec_dbg(codec, "SKU: platform_type=0x%x\n", spec->cdefine.platform_type);
661 codec_dbg(codec, "SKU: swap=0x%x\n", spec->cdefine.swap);
662 codec_dbg(codec, "SKU: override=0x%x\n", spec->cdefine.override);
Kailang Yangda00c242010-03-19 11:23:45 +0100663
664 return 0;
665}
666
Takashi Iwai08c189f2012-12-19 15:22:24 +0100667/* return the position of NID in the list, or -1 if not found */
668static int find_idx_in_nid_list(hda_nid_t nid, const hda_nid_t *list, int nums)
669{
670 int i;
671 for (i = 0; i < nums; i++)
672 if (list[i] == nid)
673 return i;
674 return -1;
675}
Takashi Iwai1d045db2011-07-07 18:23:21 +0200676/* return true if the given NID is found in the list */
Takashi Iwai3af9ee62011-06-27 12:34:01 +0200677static bool found_in_nid_list(hda_nid_t nid, const hda_nid_t *list, int nums)
678{
Takashi Iwai21268962011-07-07 15:01:13 +0200679 return find_idx_in_nid_list(nid, list, nums) >= 0;
Takashi Iwai3af9ee62011-06-27 12:34:01 +0200680}
681
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200682/* check subsystem ID and set up device-specific initialization;
683 * return 1 if initialized, 0 if invalid SSID
684 */
685/* 32-bit subsystem ID for BIOS loading in HD Audio codec.
686 * 31 ~ 16 : Manufacture ID
687 * 15 ~ 8 : SKU ID
688 * 7 ~ 0 : Assembly ID
689 * port-A --> pin 39/41, port-E --> pin 14/15, port-D --> pin 35/36
690 */
Takashi Iwai58c57cf2014-01-07 18:22:49 +0100691static int alc_subsystem_id(struct hda_codec *codec, const hda_nid_t *ports)
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200692{
693 unsigned int ass, tmp, i;
694 unsigned nid;
695 struct alc_spec *spec = codec->spec;
696
David Henningsson90622912010-10-14 14:50:18 +0200697 if (spec->cdefine.fixup) {
698 ass = spec->cdefine.sku_cfg;
699 if (ass == ALC_FIXUP_SKU_IGNORE)
700 return 0;
701 goto do_sku;
702 }
703
Takashi Iwai7639a062015-03-03 10:07:24 +0100704 ass = codec->core.subsystem_id & 0xffff;
Takashi Iwai5100cd02014-02-15 10:03:19 +0100705 if (codec->bus->pci &&
706 ass != codec->bus->pci->subsystem_device && (ass & 1))
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200707 goto do_sku;
708
709 /* invalid SSID, check the special NID pin defcfg instead */
710 /*
Sasha Alexandrdef319f2009-06-16 16:00:15 -0400711 * 31~30 : port connectivity
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200712 * 29~21 : reserve
713 * 20 : PCBEEP input
714 * 19~16 : Check sum (15:1)
715 * 15~1 : Custom
716 * 0 : override
717 */
718 nid = 0x1d;
Takashi Iwai7639a062015-03-03 10:07:24 +0100719 if (codec->core.vendor_id == 0x10ec0260)
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200720 nid = 0x17;
721 ass = snd_hda_codec_get_pincfg(codec, nid);
Takashi Iwai4e76a882014-02-25 12:21:03 +0100722 codec_dbg(codec,
723 "realtek: No valid SSID, checking pincfg 0x%08x for NID 0x%x\n",
Takashi Iwaicb6605c2009-04-28 13:03:19 +0200724 ass, nid);
Kailang Yang6227cdc2010-02-25 08:36:52 +0100725 if (!(ass & 1))
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200726 return 0;
727 if ((ass >> 30) != 1) /* no physical connection */
728 return 0;
729
730 /* check sum */
731 tmp = 0;
732 for (i = 1; i < 16; i++) {
733 if ((ass >> i) & 1)
734 tmp++;
735 }
736 if (((ass >> 16) & 0xf) != tmp)
737 return 0;
738do_sku:
Takashi Iwai4e76a882014-02-25 12:21:03 +0100739 codec_dbg(codec, "realtek: Enabling init ASM_ID=0x%04x CODEC_ID=%08x\n",
Takashi Iwai7639a062015-03-03 10:07:24 +0100740 ass & 0xffff, codec->core.vendor_id);
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200741 /*
742 * 0 : override
743 * 1 : Swap Jack
744 * 2 : 0 --> Desktop, 1 --> Laptop
745 * 3~5 : External Amplifier control
746 * 7~6 : Reserved
747 */
748 tmp = (ass & 0x38) >> 3; /* external Amp control */
Takashi Iwai1c76aa52018-06-21 16:37:54 +0200749 if (spec->init_amp == ALC_INIT_UNDEFINED) {
750 switch (tmp) {
751 case 1:
Takashi Iwai5579cd62018-06-19 22:22:41 +0200752 alc_setup_gpio(codec, 0x01);
Takashi Iwai1c76aa52018-06-21 16:37:54 +0200753 break;
754 case 3:
Takashi Iwai5579cd62018-06-19 22:22:41 +0200755 alc_setup_gpio(codec, 0x02);
Takashi Iwai1c76aa52018-06-21 16:37:54 +0200756 break;
757 case 7:
Takashi Iwai5579cd62018-06-19 22:22:41 +0200758 alc_setup_gpio(codec, 0x03);
Takashi Iwai1c76aa52018-06-21 16:37:54 +0200759 break;
760 case 5:
761 default:
762 spec->init_amp = ALC_INIT_DEFAULT;
763 break;
764 }
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200765 }
766
767 /* is laptop or Desktop and enable the function "Mute internal speaker
768 * when the external headphone out jack is plugged"
769 */
770 if (!(ass & 0x8000))
771 return 1;
772 /*
773 * 10~8 : Jack location
774 * 12~11: Headphone out -> 00: PortA, 01: PortE, 02: PortD, 03: Resvered
775 * 14~13: Resvered
776 * 15 : 1 --> enable the function "Mute internal speaker
777 * when the external headphone out jack is plugged"
778 */
Takashi Iwai35a39f92019-02-01 11:19:50 +0100779 if (!alc_get_hp_pin(spec)) {
Takashi Iwai01d48252009-10-06 13:21:54 +0200780 hda_nid_t nid;
Kailang Yangc9b58002007-10-16 14:30:01 +0200781 tmp = (ass >> 11) & 0x3; /* HP to chassis */
Takashi Iwai58c57cf2014-01-07 18:22:49 +0100782 nid = ports[tmp];
Takashi Iwai08c189f2012-12-19 15:22:24 +0100783 if (found_in_nid_list(nid, spec->gen.autocfg.line_out_pins,
784 spec->gen.autocfg.line_outs))
Takashi Iwai3af9ee62011-06-27 12:34:01 +0200785 return 1;
Takashi Iwai08c189f2012-12-19 15:22:24 +0100786 spec->gen.autocfg.hp_pins[0] = nid;
Kailang Yangc9b58002007-10-16 14:30:01 +0200787 }
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200788 return 1;
789}
Kailang Yangea1fb292008-08-26 12:58:38 +0200790
Takashi Iwai3e6179b2011-07-08 16:55:13 +0200791/* Check the validity of ALC subsystem-id
792 * ports contains an array of 4 pin NIDs for port-A, E, D and I */
793static void alc_ssid_check(struct hda_codec *codec, const hda_nid_t *ports)
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200794{
Takashi Iwai58c57cf2014-01-07 18:22:49 +0100795 if (!alc_subsystem_id(codec, ports)) {
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200796 struct alc_spec *spec = codec->spec;
Takashi Iwai4e76a882014-02-25 12:21:03 +0100797 codec_dbg(codec,
798 "realtek: Enable default setup for auto mode as fallback\n");
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200799 spec->init_amp = ALC_INIT_DEFAULT;
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200800 }
Takashi Iwai21268962011-07-07 15:01:13 +0200801}
Takashi Iwai1a1455d2011-04-28 17:36:18 +0200802
Takashi Iwai41e41f12005-06-08 14:48:49 +0200803/*
Takashi Iwai1d045db2011-07-07 18:23:21 +0200804 */
805
David Henningsson9d36a7d2014-10-07 10:18:42 +0200806static void alc_fixup_inv_dmic(struct hda_codec *codec,
807 const struct hda_fixup *fix, int action)
Takashi Iwai125821a2012-06-22 14:30:29 +0200808{
809 struct alc_spec *spec = codec->spec;
Takashi Iwai668d1e92012-11-29 14:10:17 +0100810
David Henningsson9d36a7d2014-10-07 10:18:42 +0200811 spec->gen.inv_dmic_split = 1;
Takashi Iwai6e72aa52012-06-25 10:52:25 +0200812}
813
Takashi Iwai603c4012008-07-30 15:01:44 +0200814
Takashi Iwai2eab6942012-12-18 15:30:41 +0100815static int alc_build_controls(struct hda_codec *codec)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700816{
Takashi Iwaia5cb4632018-06-20 12:50:11 +0200817 int err;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700818
Takashi Iwai08c189f2012-12-19 15:22:24 +0100819 err = snd_hda_gen_build_controls(codec);
820 if (err < 0)
821 return err;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700822
Takashi Iwai1727a772013-01-10 09:52:52 +0100823 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_BUILD);
Takashi Iwai420b0fe2012-03-12 12:35:27 +0100824 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700825}
826
Takashi Iwaie9edcee2005-06-13 14:16:38 +0200827
Linus Torvalds1da177e2005-04-16 15:20:36 -0700828/*
Takashi Iwaiae6b8132006-03-03 16:47:17 +0100829 * Common callbacks
Takashi Iwaie9edcee2005-06-13 14:16:38 +0200830 */
Takashi Iwai16ded522005-06-10 19:58:24 +0200831
Takashi Iwaic9af7532019-05-10 11:01:43 +0200832static void alc_pre_init(struct hda_codec *codec)
833{
834 alc_fill_eapd_coef(codec);
835}
836
837#define is_s4_resume(codec) \
838 ((codec)->core.dev.power.power_state.event == PM_EVENT_RESTORE)
839
Linus Torvalds1da177e2005-04-16 15:20:36 -0700840static int alc_init(struct hda_codec *codec)
841{
842 struct alc_spec *spec = codec->spec;
Takashi Iwaie9edcee2005-06-13 14:16:38 +0200843
Takashi Iwaic9af7532019-05-10 11:01:43 +0200844 /* hibernation resume needs the full chip initialization */
845 if (is_s4_resume(codec))
846 alc_pre_init(codec);
847
Takashi Iwai546bb672012-03-07 08:37:19 +0100848 if (spec->init_hook)
849 spec->init_hook(codec);
Kailang Yang526af6e2012-03-07 08:25:20 +0100850
Kailang Yang607ca3b2019-04-26 16:35:41 +0800851 snd_hda_gen_init(codec);
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +0200852 alc_fix_pll(codec);
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200853 alc_auto_init_amp(codec, spec->init_amp);
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +0200854
Takashi Iwai1727a772013-01-10 09:52:52 +0100855 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_INIT);
Takashi Iwaie08a0072006-09-07 17:52:14 +0200856
Linus Torvalds1da177e2005-04-16 15:20:36 -0700857 return 0;
858}
859
Takashi Iwaia4e09aa2009-12-27 11:22:24 +0100860static inline void alc_shutup(struct hda_codec *codec)
861{
Takashi Iwai1c7161532011-04-07 10:37:16 +0200862 struct alc_spec *spec = codec->spec;
863
Takashi Iwaic7273bd2018-06-12 16:09:57 +0200864 if (!snd_hda_get_bool_hint(codec, "shutup"))
865 return; /* disabled explicitly by hints */
866
Takashi Iwai1c7161532011-04-07 10:37:16 +0200867 if (spec && spec->shutup)
868 spec->shutup(codec);
Takashi Iwai9bfb2842013-07-24 14:31:50 +0200869 else
Takashi Iwaic0ca5ec2019-02-20 16:15:45 +0100870 alc_shutup_pins(codec);
Takashi Iwaia4e09aa2009-12-27 11:22:24 +0100871}
872
Takashi Iwai70a09762015-12-15 14:59:58 +0100873static void alc_reboot_notify(struct hda_codec *codec)
874{
875 struct alc_spec *spec = codec->spec;
876
877 if (spec && spec->reboot_notify)
878 spec->reboot_notify(codec);
879 else
880 alc_shutup(codec);
881}
882
883/* power down codec to D3 at reboot/shutdown; set as reboot_notify ops */
884static void alc_d3_at_reboot(struct hda_codec *codec)
885{
886 snd_hda_codec_set_power_to_all(codec, codec->core.afg, AC_PWRST_D3);
887 snd_hda_codec_write(codec, codec->core.afg, 0,
888 AC_VERB_SET_POWER_STATE, AC_PWRST_D3);
889 msleep(10);
890}
891
Takashi Iwai8a02c0c2014-02-10 18:09:45 +0100892#define alc_free snd_hda_gen_free
Linus Torvalds1da177e2005-04-16 15:20:36 -0700893
Takashi Iwai83012a72012-08-24 18:38:08 +0200894#ifdef CONFIG_PM
Daniel T Chenc97259d2009-12-27 18:52:08 -0500895static void alc_power_eapd(struct hda_codec *codec)
896{
Takashi Iwai691f1fc2011-04-07 10:31:43 +0200897 alc_auto_setup_eapd(codec, false);
Daniel T Chenc97259d2009-12-27 18:52:08 -0500898}
899
Takashi Iwai68cb2b52012-07-02 15:20:37 +0200900static int alc_suspend(struct hda_codec *codec)
Hector Martinf5de24b2009-12-20 22:51:31 +0100901{
902 struct alc_spec *spec = codec->spec;
Takashi Iwaia4e09aa2009-12-27 11:22:24 +0100903 alc_shutup(codec);
Hector Martinf5de24b2009-12-20 22:51:31 +0100904 if (spec && spec->power_hook)
Daniel T Chenc97259d2009-12-27 18:52:08 -0500905 spec->power_hook(codec);
Hector Martinf5de24b2009-12-20 22:51:31 +0100906 return 0;
907}
908#endif
909
Takashi Iwai2a439522011-07-26 09:52:50 +0200910#ifdef CONFIG_PM
Takashi Iwaie044c392008-10-27 16:56:24 +0100911static int alc_resume(struct hda_codec *codec)
912{
Kailang Yang97a26572013-11-29 00:35:26 -0500913 struct alc_spec *spec = codec->spec;
914
915 if (!spec->no_depop_delay)
916 msleep(150); /* to avoid pop noise */
Takashi Iwaie044c392008-10-27 16:56:24 +0100917 codec->patch_ops.init(codec);
Takashi Iwaieeecd9d2015-02-25 15:18:50 +0100918 regcache_sync(codec->core.regmap);
Takashi Iwai9e5341b2010-09-21 09:57:06 +0200919 hda_call_check_power_status(codec, 0x01);
Takashi Iwaie044c392008-10-27 16:56:24 +0100920 return 0;
921}
Takashi Iwaie044c392008-10-27 16:56:24 +0100922#endif
923
Linus Torvalds1da177e2005-04-16 15:20:36 -0700924/*
925 */
Takashi Iwaia9111322011-05-02 11:30:18 +0200926static const struct hda_codec_ops alc_patch_ops = {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700927 .build_controls = alc_build_controls,
Takashi Iwai08c189f2012-12-19 15:22:24 +0100928 .build_pcms = snd_hda_gen_build_pcms,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700929 .init = alc_init,
930 .free = alc_free,
David Henningsson29adc4b2012-09-25 11:31:00 +0200931 .unsol_event = snd_hda_jack_unsol_event,
Takashi Iwai2a439522011-07-26 09:52:50 +0200932#ifdef CONFIG_PM
Takashi Iwaie044c392008-10-27 16:56:24 +0100933 .resume = alc_resume,
Hector Martinf5de24b2009-12-20 22:51:31 +0100934 .suspend = alc_suspend,
Takashi Iwaifce52a32013-01-07 12:42:48 +0100935 .check_power_status = snd_hda_gen_check_power_status,
Takashi Iwaicb53c622007-08-10 17:21:45 +0200936#endif
Takashi Iwai70a09762015-12-15 14:59:58 +0100937 .reboot_notify = alc_reboot_notify,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700938};
939
David Henningsson29adc4b2012-09-25 11:31:00 +0200940
Takashi Iwaided255b2015-10-01 17:59:43 +0200941#define alc_codec_rename(codec, name) snd_hda_codec_set_name(codec, name)
Kailang Yangc027ddc2010-03-19 11:33:06 +0100942
Takashi Iwai2fa522b2005-05-12 14:51:12 +0200943/*
Kailang Yang4b016932013-11-28 11:55:09 +0100944 * Rename codecs appropriately from COEF value or subvendor id
Takashi Iwaie16fb6d2011-10-17 16:39:09 +0200945 */
946struct alc_codec_rename_table {
947 unsigned int vendor_id;
948 unsigned short coef_mask;
949 unsigned short coef_bits;
950 const char *name;
951};
952
Kailang Yang4b016932013-11-28 11:55:09 +0100953struct alc_codec_rename_pci_table {
954 unsigned int codec_vendor_id;
955 unsigned short pci_subvendor;
956 unsigned short pci_subdevice;
957 const char *name;
958};
959
Takashi Iwaie16fb6d2011-10-17 16:39:09 +0200960static struct alc_codec_rename_table rename_tbl[] = {
Kailang Yange6e5f7a2014-06-06 16:46:43 +0800961 { 0x10ec0221, 0xf00f, 0x1003, "ALC231" },
Takashi Iwaie16fb6d2011-10-17 16:39:09 +0200962 { 0x10ec0269, 0xfff0, 0x3010, "ALC277" },
963 { 0x10ec0269, 0xf0f0, 0x2010, "ALC259" },
964 { 0x10ec0269, 0xf0f0, 0x3010, "ALC258" },
965 { 0x10ec0269, 0x00f0, 0x0010, "ALC269VB" },
966 { 0x10ec0269, 0xffff, 0xa023, "ALC259" },
967 { 0x10ec0269, 0xffff, 0x6023, "ALC281X" },
968 { 0x10ec0269, 0x00f0, 0x0020, "ALC269VC" },
Kailang Yangadcc70b2012-05-25 08:08:38 +0200969 { 0x10ec0269, 0x00f0, 0x0030, "ALC269VD" },
Kailang Yange6e5f7a2014-06-06 16:46:43 +0800970 { 0x10ec0662, 0xffff, 0x4020, "ALC656" },
Takashi Iwaie16fb6d2011-10-17 16:39:09 +0200971 { 0x10ec0887, 0x00f0, 0x0030, "ALC887-VD" },
972 { 0x10ec0888, 0x00f0, 0x0030, "ALC888-VD" },
973 { 0x10ec0888, 0xf0f0, 0x3020, "ALC886" },
974 { 0x10ec0899, 0x2000, 0x2000, "ALC899" },
975 { 0x10ec0892, 0xffff, 0x8020, "ALC661" },
976 { 0x10ec0892, 0xffff, 0x8011, "ALC661" },
977 { 0x10ec0892, 0xffff, 0x4011, "ALC656" },
978 { } /* terminator */
979};
980
Kailang Yang4b016932013-11-28 11:55:09 +0100981static struct alc_codec_rename_pci_table rename_pci_tbl[] = {
982 { 0x10ec0280, 0x1028, 0, "ALC3220" },
983 { 0x10ec0282, 0x1028, 0, "ALC3221" },
984 { 0x10ec0283, 0x1028, 0, "ALC3223" },
Kailang Yang193177d2014-04-23 16:06:13 +0800985 { 0x10ec0288, 0x1028, 0, "ALC3263" },
Kailang Yang4b016932013-11-28 11:55:09 +0100986 { 0x10ec0292, 0x1028, 0, "ALC3226" },
Kailang Yang193177d2014-04-23 16:06:13 +0800987 { 0x10ec0293, 0x1028, 0, "ALC3235" },
Kailang Yang4b016932013-11-28 11:55:09 +0100988 { 0x10ec0255, 0x1028, 0, "ALC3234" },
989 { 0x10ec0668, 0x1028, 0, "ALC3661" },
Kailang Yange6e5f7a2014-06-06 16:46:43 +0800990 { 0x10ec0275, 0x1028, 0, "ALC3260" },
991 { 0x10ec0899, 0x1028, 0, "ALC3861" },
Kailang Yang2c674fa2015-05-05 15:02:42 +0800992 { 0x10ec0298, 0x1028, 0, "ALC3266" },
Kailang Yang736f20a2017-10-20 15:06:34 +0800993 { 0x10ec0236, 0x1028, 0, "ALC3204" },
Kailang Yang82324502015-05-25 17:16:49 +0800994 { 0x10ec0256, 0x1028, 0, "ALC3246" },
Kailang Yang42314302016-02-03 15:03:50 +0800995 { 0x10ec0225, 0x1028, 0, "ALC3253" },
Kailang Yang7d727862016-05-24 16:46:07 +0800996 { 0x10ec0295, 0x1028, 0, "ALC3254" },
Kailang Yang28f1f9b2017-01-04 14:49:07 +0800997 { 0x10ec0299, 0x1028, 0, "ALC3271" },
Kailang Yange6e5f7a2014-06-06 16:46:43 +0800998 { 0x10ec0670, 0x1025, 0, "ALC669X" },
999 { 0x10ec0676, 0x1025, 0, "ALC679X" },
1000 { 0x10ec0282, 0x1043, 0, "ALC3229" },
1001 { 0x10ec0233, 0x1043, 0, "ALC3236" },
1002 { 0x10ec0280, 0x103c, 0, "ALC3228" },
1003 { 0x10ec0282, 0x103c, 0, "ALC3227" },
1004 { 0x10ec0286, 0x103c, 0, "ALC3242" },
1005 { 0x10ec0290, 0x103c, 0, "ALC3241" },
1006 { 0x10ec0668, 0x103c, 0, "ALC3662" },
1007 { 0x10ec0283, 0x17aa, 0, "ALC3239" },
1008 { 0x10ec0292, 0x17aa, 0, "ALC3232" },
Kailang Yang4b016932013-11-28 11:55:09 +01001009 { } /* terminator */
1010};
1011
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02001012static int alc_codec_rename_from_preset(struct hda_codec *codec)
1013{
1014 const struct alc_codec_rename_table *p;
Kailang Yang4b016932013-11-28 11:55:09 +01001015 const struct alc_codec_rename_pci_table *q;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02001016
1017 for (p = rename_tbl; p->vendor_id; p++) {
Takashi Iwai7639a062015-03-03 10:07:24 +01001018 if (p->vendor_id != codec->core.vendor_id)
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02001019 continue;
Takashi Iwai1bb7e432011-10-17 16:50:59 +02001020 if ((alc_get_coef0(codec) & p->coef_mask) == p->coef_bits)
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02001021 return alc_codec_rename(codec, p->name);
1022 }
Kailang Yang4b016932013-11-28 11:55:09 +01001023
Takashi Iwai5100cd02014-02-15 10:03:19 +01001024 if (!codec->bus->pci)
1025 return 0;
Kailang Yang4b016932013-11-28 11:55:09 +01001026 for (q = rename_pci_tbl; q->codec_vendor_id; q++) {
Takashi Iwai7639a062015-03-03 10:07:24 +01001027 if (q->codec_vendor_id != codec->core.vendor_id)
Kailang Yang4b016932013-11-28 11:55:09 +01001028 continue;
1029 if (q->pci_subvendor != codec->bus->pci->subsystem_vendor)
1030 continue;
1031 if (!q->pci_subdevice ||
1032 q->pci_subdevice == codec->bus->pci->subsystem_device)
1033 return alc_codec_rename(codec, q->name);
1034 }
1035
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02001036 return 0;
1037}
1038
Takashi Iwaie4770622011-07-08 11:11:35 +02001039
1040/*
Takashi Iwai1d045db2011-07-07 18:23:21 +02001041 * Digital-beep handlers
1042 */
1043#ifdef CONFIG_SND_HDA_INPUT_BEEP
Takashi Iwaifea80fa2018-06-20 12:52:46 +02001044
1045/* additional beep mixers; private_value will be overwritten */
1046static const struct snd_kcontrol_new alc_beep_mixer[] = {
1047 HDA_CODEC_VOLUME("Beep Playback Volume", 0, 0, HDA_INPUT),
1048 HDA_CODEC_MUTE_BEEP("Beep Playback Switch", 0, 0, HDA_INPUT),
1049};
1050
1051/* set up and create beep controls */
1052static int set_beep_amp(struct alc_spec *spec, hda_nid_t nid,
1053 int idx, int dir)
1054{
1055 struct snd_kcontrol_new *knew;
1056 unsigned int beep_amp = HDA_COMPOSE_AMP_VAL(nid, 3, idx, dir);
1057 int i;
1058
1059 for (i = 0; i < ARRAY_SIZE(alc_beep_mixer); i++) {
1060 knew = snd_hda_gen_add_kctl(&spec->gen, NULL,
1061 &alc_beep_mixer[i]);
1062 if (!knew)
1063 return -ENOMEM;
1064 knew->private_value = beep_amp;
1065 }
1066 return 0;
1067}
Takashi Iwai1d045db2011-07-07 18:23:21 +02001068
1069static const struct snd_pci_quirk beep_white_list[] = {
Duncan Roe71100052012-10-10 14:19:50 +02001070 SND_PCI_QUIRK(0x1043, 0x103c, "ASUS", 1),
W. Trevor Kinga4b7f21d2014-03-29 17:47:24 -07001071 SND_PCI_QUIRK(0x1043, 0x115d, "ASUS", 1),
Takashi Iwai1d045db2011-07-07 18:23:21 +02001072 SND_PCI_QUIRK(0x1043, 0x829f, "ASUS", 1),
Takashi Iwai8554ee42013-02-25 09:54:43 +01001073 SND_PCI_QUIRK(0x1043, 0x8376, "EeePC", 1),
Takashi Iwai1d045db2011-07-07 18:23:21 +02001074 SND_PCI_QUIRK(0x1043, 0x83ce, "EeePC", 1),
1075 SND_PCI_QUIRK(0x1043, 0x831a, "EeePC", 1),
1076 SND_PCI_QUIRK(0x1043, 0x834a, "EeePC", 1),
Takashi Iwai78f8baf2012-03-06 14:02:32 +01001077 SND_PCI_QUIRK(0x1458, 0xa002, "GA-MA790X", 1),
Takashi Iwai1d045db2011-07-07 18:23:21 +02001078 SND_PCI_QUIRK(0x8086, 0xd613, "Intel", 1),
1079 {}
1080};
1081
1082static inline int has_cdefine_beep(struct hda_codec *codec)
1083{
1084 struct alc_spec *spec = codec->spec;
1085 const struct snd_pci_quirk *q;
1086 q = snd_pci_quirk_lookup(codec->bus->pci, beep_white_list);
1087 if (q)
1088 return q->value;
1089 return spec->cdefine.enable_pcbeep;
1090}
1091#else
Takashi Iwaifea80fa2018-06-20 12:52:46 +02001092#define set_beep_amp(spec, nid, idx, dir) 0
Takashi Iwai1d045db2011-07-07 18:23:21 +02001093#define has_cdefine_beep(codec) 0
1094#endif
1095
1096/* parse the BIOS configuration and set up the alc_spec */
1097/* return 1 if successful, 0 if the proper config is not found,
1098 * or a negative error code
1099 */
Takashi Iwai3e6179b2011-07-08 16:55:13 +02001100static int alc_parse_auto_config(struct hda_codec *codec,
1101 const hda_nid_t *ignore_nids,
1102 const hda_nid_t *ssid_nids)
Takashi Iwai1d045db2011-07-07 18:23:21 +02001103{
1104 struct alc_spec *spec = codec->spec;
Takashi Iwai08c189f2012-12-19 15:22:24 +01001105 struct auto_pin_cfg *cfg = &spec->gen.autocfg;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001106 int err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001107
Takashi Iwai53c334a2011-08-23 18:27:14 +02001108 err = snd_hda_parse_pin_defcfg(codec, cfg, ignore_nids,
1109 spec->parse_flags);
Takashi Iwai1d045db2011-07-07 18:23:21 +02001110 if (err < 0)
1111 return err;
Takashi Iwai3e6179b2011-07-08 16:55:13 +02001112
1113 if (ssid_nids)
1114 alc_ssid_check(codec, ssid_nids);
1115
Takashi Iwai08c189f2012-12-19 15:22:24 +01001116 err = snd_hda_gen_parse_auto_config(codec, cfg);
1117 if (err < 0)
1118 return err;
Takashi Iwai070cff42012-02-21 12:54:17 +01001119
Takashi Iwai1d045db2011-07-07 18:23:21 +02001120 return 1;
1121}
1122
Takashi Iwai3de95172012-05-07 18:03:15 +02001123/* common preparation job for alc_spec */
1124static int alc_alloc_spec(struct hda_codec *codec, hda_nid_t mixer_nid)
1125{
1126 struct alc_spec *spec = kzalloc(sizeof(*spec), GFP_KERNEL);
1127 int err;
1128
1129 if (!spec)
1130 return -ENOMEM;
1131 codec->spec = spec;
Takashi Iwai08c189f2012-12-19 15:22:24 +01001132 snd_hda_gen_spec_init(&spec->gen);
1133 spec->gen.mixer_nid = mixer_nid;
1134 spec->gen.own_eapd_ctl = 1;
Takashi Iwai1098b7c2012-12-17 20:03:15 +01001135 codec->single_adc_amp = 1;
Takashi Iwai08c189f2012-12-19 15:22:24 +01001136 /* FIXME: do we need this for all Realtek codec models? */
1137 codec->spdif_status_reset = 1;
Takashi Iwai225068a2015-05-29 10:42:14 +02001138 codec->patch_ops = alc_patch_ops;
Takashi Iwai3de95172012-05-07 18:03:15 +02001139
1140 err = alc_codec_rename_from_preset(codec);
1141 if (err < 0) {
1142 kfree(spec);
1143 return err;
1144 }
1145 return 0;
1146}
1147
Takashi Iwai3e6179b2011-07-08 16:55:13 +02001148static int alc880_parse_auto_config(struct hda_codec *codec)
1149{
1150 static const hda_nid_t alc880_ignore[] = { 0x1d, 0 };
Jesper Juhl7d7eb9e2012-04-12 22:11:25 +02001151 static const hda_nid_t alc880_ssids[] = { 0x15, 0x1b, 0x14, 0 };
Takashi Iwai3e6179b2011-07-08 16:55:13 +02001152 return alc_parse_auto_config(codec, alc880_ignore, alc880_ssids);
1153}
1154
Takashi Iwai1d045db2011-07-07 18:23:21 +02001155/*
Takashi Iwaiee3b2962011-11-15 14:26:54 +01001156 * ALC880 fix-ups
1157 */
1158enum {
Takashi Iwai411225a2012-02-20 17:48:19 +01001159 ALC880_FIXUP_GPIO1,
Takashi Iwaiee3b2962011-11-15 14:26:54 +01001160 ALC880_FIXUP_GPIO2,
1161 ALC880_FIXUP_MEDION_RIM,
Takashi Iwaidc6af522012-02-17 16:18:59 +01001162 ALC880_FIXUP_LG,
Takashi Iwaidb8a38e2013-08-09 12:34:42 +02001163 ALC880_FIXUP_LG_LW25,
Takashi Iwaif02aab52012-02-17 16:33:56 +01001164 ALC880_FIXUP_W810,
Takashi Iwai27e917f2012-02-17 17:49:54 +01001165 ALC880_FIXUP_EAPD_COEF,
Takashi Iwaib9368f52012-02-17 17:54:44 +01001166 ALC880_FIXUP_TCL_S700,
Takashi Iwaicf5a2272012-02-20 16:31:07 +01001167 ALC880_FIXUP_VOL_KNOB,
1168 ALC880_FIXUP_FUJITSU,
Takashi Iwaiba533812012-02-20 16:36:52 +01001169 ALC880_FIXUP_F1734,
Takashi Iwai817de922012-02-20 17:20:48 +01001170 ALC880_FIXUP_UNIWILL,
Takashi Iwai967b88c2012-02-20 17:31:02 +01001171 ALC880_FIXUP_UNIWILL_DIG,
Takashi Iwai96e225f2012-02-20 17:41:51 +01001172 ALC880_FIXUP_Z71V,
Takashi Iwai487a5882013-11-07 07:29:30 +01001173 ALC880_FIXUP_ASUS_W5A,
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001174 ALC880_FIXUP_3ST_BASE,
1175 ALC880_FIXUP_3ST,
1176 ALC880_FIXUP_3ST_DIG,
1177 ALC880_FIXUP_5ST_BASE,
1178 ALC880_FIXUP_5ST,
1179 ALC880_FIXUP_5ST_DIG,
1180 ALC880_FIXUP_6ST_BASE,
1181 ALC880_FIXUP_6ST,
1182 ALC880_FIXUP_6ST_DIG,
Takashi Iwai53971452013-01-23 18:21:37 +01001183 ALC880_FIXUP_6ST_AUTOMUTE,
Takashi Iwaiee3b2962011-11-15 14:26:54 +01001184};
1185
Takashi Iwaicf5a2272012-02-20 16:31:07 +01001186/* enable the volume-knob widget support on NID 0x21 */
1187static void alc880_fixup_vol_knob(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01001188 const struct hda_fixup *fix, int action)
Takashi Iwaicf5a2272012-02-20 16:31:07 +01001189{
Takashi Iwai1727a772013-01-10 09:52:52 +01001190 if (action == HDA_FIXUP_ACT_PROBE)
Takashi Iwai62f949b2014-09-11 14:06:53 +02001191 snd_hda_jack_detect_enable_callback(codec, 0x21,
1192 alc_update_knob_master);
Takashi Iwaicf5a2272012-02-20 16:31:07 +01001193}
1194
Takashi Iwai1727a772013-01-10 09:52:52 +01001195static const struct hda_fixup alc880_fixups[] = {
Takashi Iwai411225a2012-02-20 17:48:19 +01001196 [ALC880_FIXUP_GPIO1] = {
Takashi Iwai5579cd62018-06-19 22:22:41 +02001197 .type = HDA_FIXUP_FUNC,
1198 .v.func = alc_fixup_gpio1,
Takashi Iwai411225a2012-02-20 17:48:19 +01001199 },
Takashi Iwaiee3b2962011-11-15 14:26:54 +01001200 [ALC880_FIXUP_GPIO2] = {
Takashi Iwai5579cd62018-06-19 22:22:41 +02001201 .type = HDA_FIXUP_FUNC,
1202 .v.func = alc_fixup_gpio2,
Takashi Iwaiee3b2962011-11-15 14:26:54 +01001203 },
1204 [ALC880_FIXUP_MEDION_RIM] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001205 .type = HDA_FIXUP_VERBS,
Takashi Iwaiee3b2962011-11-15 14:26:54 +01001206 .v.verbs = (const struct hda_verb[]) {
1207 { 0x20, AC_VERB_SET_COEF_INDEX, 0x07 },
1208 { 0x20, AC_VERB_SET_PROC_COEF, 0x3060 },
1209 { }
1210 },
1211 .chained = true,
1212 .chain_id = ALC880_FIXUP_GPIO2,
1213 },
Takashi Iwaidc6af522012-02-17 16:18:59 +01001214 [ALC880_FIXUP_LG] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001215 .type = HDA_FIXUP_PINS,
1216 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaidc6af522012-02-17 16:18:59 +01001217 /* disable bogus unused pins */
1218 { 0x16, 0x411111f0 },
1219 { 0x18, 0x411111f0 },
1220 { 0x1a, 0x411111f0 },
1221 { }
1222 }
1223 },
Takashi Iwaidb8a38e2013-08-09 12:34:42 +02001224 [ALC880_FIXUP_LG_LW25] = {
1225 .type = HDA_FIXUP_PINS,
1226 .v.pins = (const struct hda_pintbl[]) {
1227 { 0x1a, 0x0181344f }, /* line-in */
1228 { 0x1b, 0x0321403f }, /* headphone */
1229 { }
1230 }
1231 },
Takashi Iwaif02aab52012-02-17 16:33:56 +01001232 [ALC880_FIXUP_W810] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001233 .type = HDA_FIXUP_PINS,
1234 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaif02aab52012-02-17 16:33:56 +01001235 /* disable bogus unused pins */
1236 { 0x17, 0x411111f0 },
1237 { }
1238 },
1239 .chained = true,
1240 .chain_id = ALC880_FIXUP_GPIO2,
1241 },
Takashi Iwai27e917f2012-02-17 17:49:54 +01001242 [ALC880_FIXUP_EAPD_COEF] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001243 .type = HDA_FIXUP_VERBS,
Takashi Iwai27e917f2012-02-17 17:49:54 +01001244 .v.verbs = (const struct hda_verb[]) {
1245 /* change to EAPD mode */
1246 { 0x20, AC_VERB_SET_COEF_INDEX, 0x07 },
1247 { 0x20, AC_VERB_SET_PROC_COEF, 0x3060 },
1248 {}
1249 },
1250 },
Takashi Iwaib9368f52012-02-17 17:54:44 +01001251 [ALC880_FIXUP_TCL_S700] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001252 .type = HDA_FIXUP_VERBS,
Takashi Iwaib9368f52012-02-17 17:54:44 +01001253 .v.verbs = (const struct hda_verb[]) {
1254 /* change to EAPD mode */
1255 { 0x20, AC_VERB_SET_COEF_INDEX, 0x07 },
1256 { 0x20, AC_VERB_SET_PROC_COEF, 0x3070 },
1257 {}
1258 },
1259 .chained = true,
1260 .chain_id = ALC880_FIXUP_GPIO2,
1261 },
Takashi Iwaicf5a2272012-02-20 16:31:07 +01001262 [ALC880_FIXUP_VOL_KNOB] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001263 .type = HDA_FIXUP_FUNC,
Takashi Iwaicf5a2272012-02-20 16:31:07 +01001264 .v.func = alc880_fixup_vol_knob,
1265 },
1266 [ALC880_FIXUP_FUJITSU] = {
1267 /* override all pins as BIOS on old Amilo is broken */
Takashi Iwai1727a772013-01-10 09:52:52 +01001268 .type = HDA_FIXUP_PINS,
1269 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaibb148bd2015-08-13 18:02:39 +02001270 { 0x14, 0x0121401f }, /* HP */
Takashi Iwaicf5a2272012-02-20 16:31:07 +01001271 { 0x15, 0x99030120 }, /* speaker */
1272 { 0x16, 0x99030130 }, /* bass speaker */
1273 { 0x17, 0x411111f0 }, /* N/A */
1274 { 0x18, 0x411111f0 }, /* N/A */
1275 { 0x19, 0x01a19950 }, /* mic-in */
1276 { 0x1a, 0x411111f0 }, /* N/A */
1277 { 0x1b, 0x411111f0 }, /* N/A */
1278 { 0x1c, 0x411111f0 }, /* N/A */
1279 { 0x1d, 0x411111f0 }, /* N/A */
1280 { 0x1e, 0x01454140 }, /* SPDIF out */
1281 { }
1282 },
1283 .chained = true,
1284 .chain_id = ALC880_FIXUP_VOL_KNOB,
1285 },
Takashi Iwaiba533812012-02-20 16:36:52 +01001286 [ALC880_FIXUP_F1734] = {
1287 /* almost compatible with FUJITSU, but no bass and SPDIF */
Takashi Iwai1727a772013-01-10 09:52:52 +01001288 .type = HDA_FIXUP_PINS,
1289 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaibb148bd2015-08-13 18:02:39 +02001290 { 0x14, 0x0121401f }, /* HP */
Takashi Iwaiba533812012-02-20 16:36:52 +01001291 { 0x15, 0x99030120 }, /* speaker */
1292 { 0x16, 0x411111f0 }, /* N/A */
1293 { 0x17, 0x411111f0 }, /* N/A */
1294 { 0x18, 0x411111f0 }, /* N/A */
1295 { 0x19, 0x01a19950 }, /* mic-in */
1296 { 0x1a, 0x411111f0 }, /* N/A */
1297 { 0x1b, 0x411111f0 }, /* N/A */
1298 { 0x1c, 0x411111f0 }, /* N/A */
1299 { 0x1d, 0x411111f0 }, /* N/A */
1300 { 0x1e, 0x411111f0 }, /* N/A */
1301 { }
1302 },
1303 .chained = true,
1304 .chain_id = ALC880_FIXUP_VOL_KNOB,
1305 },
Takashi Iwai817de922012-02-20 17:20:48 +01001306 [ALC880_FIXUP_UNIWILL] = {
1307 /* need to fix HP and speaker pins to be parsed correctly */
Takashi Iwai1727a772013-01-10 09:52:52 +01001308 .type = HDA_FIXUP_PINS,
1309 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai817de922012-02-20 17:20:48 +01001310 { 0x14, 0x0121411f }, /* HP */
1311 { 0x15, 0x99030120 }, /* speaker */
1312 { 0x16, 0x99030130 }, /* bass speaker */
1313 { }
1314 },
1315 },
Takashi Iwai967b88c2012-02-20 17:31:02 +01001316 [ALC880_FIXUP_UNIWILL_DIG] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001317 .type = HDA_FIXUP_PINS,
1318 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai967b88c2012-02-20 17:31:02 +01001319 /* disable bogus unused pins */
1320 { 0x17, 0x411111f0 },
1321 { 0x19, 0x411111f0 },
1322 { 0x1b, 0x411111f0 },
1323 { 0x1f, 0x411111f0 },
1324 { }
1325 }
1326 },
Takashi Iwai96e225f2012-02-20 17:41:51 +01001327 [ALC880_FIXUP_Z71V] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001328 .type = HDA_FIXUP_PINS,
1329 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai96e225f2012-02-20 17:41:51 +01001330 /* set up the whole pins as BIOS is utterly broken */
1331 { 0x14, 0x99030120 }, /* speaker */
1332 { 0x15, 0x0121411f }, /* HP */
1333 { 0x16, 0x411111f0 }, /* N/A */
1334 { 0x17, 0x411111f0 }, /* N/A */
1335 { 0x18, 0x01a19950 }, /* mic-in */
1336 { 0x19, 0x411111f0 }, /* N/A */
1337 { 0x1a, 0x01813031 }, /* line-in */
1338 { 0x1b, 0x411111f0 }, /* N/A */
1339 { 0x1c, 0x411111f0 }, /* N/A */
1340 { 0x1d, 0x411111f0 }, /* N/A */
1341 { 0x1e, 0x0144111e }, /* SPDIF */
1342 { }
1343 }
1344 },
Takashi Iwai487a5882013-11-07 07:29:30 +01001345 [ALC880_FIXUP_ASUS_W5A] = {
1346 .type = HDA_FIXUP_PINS,
1347 .v.pins = (const struct hda_pintbl[]) {
1348 /* set up the whole pins as BIOS is utterly broken */
1349 { 0x14, 0x0121411f }, /* HP */
1350 { 0x15, 0x411111f0 }, /* N/A */
1351 { 0x16, 0x411111f0 }, /* N/A */
1352 { 0x17, 0x411111f0 }, /* N/A */
1353 { 0x18, 0x90a60160 }, /* mic */
1354 { 0x19, 0x411111f0 }, /* N/A */
1355 { 0x1a, 0x411111f0 }, /* N/A */
1356 { 0x1b, 0x411111f0 }, /* N/A */
1357 { 0x1c, 0x411111f0 }, /* N/A */
1358 { 0x1d, 0x411111f0 }, /* N/A */
1359 { 0x1e, 0xb743111e }, /* SPDIF out */
1360 { }
1361 },
1362 .chained = true,
1363 .chain_id = ALC880_FIXUP_GPIO1,
1364 },
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001365 [ALC880_FIXUP_3ST_BASE] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001366 .type = HDA_FIXUP_PINS,
1367 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001368 { 0x14, 0x01014010 }, /* line-out */
1369 { 0x15, 0x411111f0 }, /* N/A */
1370 { 0x16, 0x411111f0 }, /* N/A */
1371 { 0x17, 0x411111f0 }, /* N/A */
1372 { 0x18, 0x01a19c30 }, /* mic-in */
1373 { 0x19, 0x0121411f }, /* HP */
1374 { 0x1a, 0x01813031 }, /* line-in */
1375 { 0x1b, 0x02a19c40 }, /* front-mic */
1376 { 0x1c, 0x411111f0 }, /* N/A */
1377 { 0x1d, 0x411111f0 }, /* N/A */
1378 /* 0x1e is filled in below */
1379 { 0x1f, 0x411111f0 }, /* N/A */
1380 { }
1381 }
1382 },
1383 [ALC880_FIXUP_3ST] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001384 .type = HDA_FIXUP_PINS,
1385 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001386 { 0x1e, 0x411111f0 }, /* N/A */
1387 { }
1388 },
1389 .chained = true,
1390 .chain_id = ALC880_FIXUP_3ST_BASE,
1391 },
1392 [ALC880_FIXUP_3ST_DIG] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001393 .type = HDA_FIXUP_PINS,
1394 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001395 { 0x1e, 0x0144111e }, /* SPDIF */
1396 { }
1397 },
1398 .chained = true,
1399 .chain_id = ALC880_FIXUP_3ST_BASE,
1400 },
1401 [ALC880_FIXUP_5ST_BASE] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001402 .type = HDA_FIXUP_PINS,
1403 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001404 { 0x14, 0x01014010 }, /* front */
1405 { 0x15, 0x411111f0 }, /* N/A */
1406 { 0x16, 0x01011411 }, /* CLFE */
1407 { 0x17, 0x01016412 }, /* surr */
1408 { 0x18, 0x01a19c30 }, /* mic-in */
1409 { 0x19, 0x0121411f }, /* HP */
1410 { 0x1a, 0x01813031 }, /* line-in */
1411 { 0x1b, 0x02a19c40 }, /* front-mic */
1412 { 0x1c, 0x411111f0 }, /* N/A */
1413 { 0x1d, 0x411111f0 }, /* N/A */
1414 /* 0x1e is filled in below */
1415 { 0x1f, 0x411111f0 }, /* N/A */
1416 { }
1417 }
1418 },
1419 [ALC880_FIXUP_5ST] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001420 .type = HDA_FIXUP_PINS,
1421 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001422 { 0x1e, 0x411111f0 }, /* N/A */
1423 { }
1424 },
1425 .chained = true,
1426 .chain_id = ALC880_FIXUP_5ST_BASE,
1427 },
1428 [ALC880_FIXUP_5ST_DIG] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001429 .type = HDA_FIXUP_PINS,
1430 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001431 { 0x1e, 0x0144111e }, /* SPDIF */
1432 { }
1433 },
1434 .chained = true,
1435 .chain_id = ALC880_FIXUP_5ST_BASE,
1436 },
1437 [ALC880_FIXUP_6ST_BASE] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001438 .type = HDA_FIXUP_PINS,
1439 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001440 { 0x14, 0x01014010 }, /* front */
1441 { 0x15, 0x01016412 }, /* surr */
1442 { 0x16, 0x01011411 }, /* CLFE */
1443 { 0x17, 0x01012414 }, /* side */
1444 { 0x18, 0x01a19c30 }, /* mic-in */
1445 { 0x19, 0x02a19c40 }, /* front-mic */
1446 { 0x1a, 0x01813031 }, /* line-in */
1447 { 0x1b, 0x0121411f }, /* HP */
1448 { 0x1c, 0x411111f0 }, /* N/A */
1449 { 0x1d, 0x411111f0 }, /* N/A */
1450 /* 0x1e is filled in below */
1451 { 0x1f, 0x411111f0 }, /* N/A */
1452 { }
1453 }
1454 },
1455 [ALC880_FIXUP_6ST] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001456 .type = HDA_FIXUP_PINS,
1457 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001458 { 0x1e, 0x411111f0 }, /* N/A */
1459 { }
1460 },
1461 .chained = true,
1462 .chain_id = ALC880_FIXUP_6ST_BASE,
1463 },
1464 [ALC880_FIXUP_6ST_DIG] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001465 .type = HDA_FIXUP_PINS,
1466 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001467 { 0x1e, 0x0144111e }, /* SPDIF */
1468 { }
1469 },
1470 .chained = true,
1471 .chain_id = ALC880_FIXUP_6ST_BASE,
1472 },
Takashi Iwai53971452013-01-23 18:21:37 +01001473 [ALC880_FIXUP_6ST_AUTOMUTE] = {
1474 .type = HDA_FIXUP_PINS,
1475 .v.pins = (const struct hda_pintbl[]) {
1476 { 0x1b, 0x0121401f }, /* HP with jack detect */
1477 { }
1478 },
1479 .chained_before = true,
1480 .chain_id = ALC880_FIXUP_6ST_BASE,
1481 },
Takashi Iwaiee3b2962011-11-15 14:26:54 +01001482};
1483
1484static const struct snd_pci_quirk alc880_fixup_tbl[] = {
Takashi Iwaif02aab52012-02-17 16:33:56 +01001485 SND_PCI_QUIRK(0x1019, 0x0f69, "Coeus G610P", ALC880_FIXUP_W810),
Takashi Iwai487a5882013-11-07 07:29:30 +01001486 SND_PCI_QUIRK(0x1043, 0x10c3, "ASUS W5A", ALC880_FIXUP_ASUS_W5A),
Takashi Iwai96e225f2012-02-20 17:41:51 +01001487 SND_PCI_QUIRK(0x1043, 0x1964, "ASUS Z71V", ALC880_FIXUP_Z71V),
Takashi Iwai29e3fdc2012-02-20 17:56:57 +01001488 SND_PCI_QUIRK_VENDOR(0x1043, "ASUS", ALC880_FIXUP_GPIO1),
David Henningsson6538de02014-06-10 10:52:50 +02001489 SND_PCI_QUIRK(0x147b, 0x1045, "ABit AA8XE", ALC880_FIXUP_6ST_AUTOMUTE),
Takashi Iwai29e3fdc2012-02-20 17:56:57 +01001490 SND_PCI_QUIRK(0x1558, 0x5401, "Clevo GPIO2", ALC880_FIXUP_GPIO2),
Takashi Iwai27e917f2012-02-17 17:49:54 +01001491 SND_PCI_QUIRK_VENDOR(0x1558, "Clevo", ALC880_FIXUP_EAPD_COEF),
Takashi Iwai967b88c2012-02-20 17:31:02 +01001492 SND_PCI_QUIRK(0x1584, 0x9050, "Uniwill", ALC880_FIXUP_UNIWILL_DIG),
Takashi Iwaiba533812012-02-20 16:36:52 +01001493 SND_PCI_QUIRK(0x1584, 0x9054, "Uniwill", ALC880_FIXUP_F1734),
Takashi Iwai817de922012-02-20 17:20:48 +01001494 SND_PCI_QUIRK(0x1584, 0x9070, "Uniwill", ALC880_FIXUP_UNIWILL),
Takashi Iwai7833c7e2012-02-20 17:11:38 +01001495 SND_PCI_QUIRK(0x1584, 0x9077, "Uniwill P53", ALC880_FIXUP_VOL_KNOB),
Takashi Iwaif02aab52012-02-17 16:33:56 +01001496 SND_PCI_QUIRK(0x161f, 0x203d, "W810", ALC880_FIXUP_W810),
Takashi Iwaiee3b2962011-11-15 14:26:54 +01001497 SND_PCI_QUIRK(0x161f, 0x205d, "Medion Rim 2150", ALC880_FIXUP_MEDION_RIM),
Takashi Iwai53971452013-01-23 18:21:37 +01001498 SND_PCI_QUIRK(0x1631, 0xe011, "PB 13201056", ALC880_FIXUP_6ST_AUTOMUTE),
Takashi Iwaia1615742015-08-13 18:05:06 +02001499 SND_PCI_QUIRK(0x1734, 0x107c, "FSC Amilo M1437", ALC880_FIXUP_FUJITSU),
Takashi Iwaicf5a2272012-02-20 16:31:07 +01001500 SND_PCI_QUIRK(0x1734, 0x1094, "FSC Amilo M1451G", ALC880_FIXUP_FUJITSU),
Takashi Iwaiba533812012-02-20 16:36:52 +01001501 SND_PCI_QUIRK(0x1734, 0x10ac, "FSC AMILO Xi 1526", ALC880_FIXUP_F1734),
Takashi Iwaicf5a2272012-02-20 16:31:07 +01001502 SND_PCI_QUIRK(0x1734, 0x10b0, "FSC Amilo Pi1556", ALC880_FIXUP_FUJITSU),
Takashi Iwaidc6af522012-02-17 16:18:59 +01001503 SND_PCI_QUIRK(0x1854, 0x003b, "LG", ALC880_FIXUP_LG),
1504 SND_PCI_QUIRK(0x1854, 0x005f, "LG P1 Express", ALC880_FIXUP_LG),
1505 SND_PCI_QUIRK(0x1854, 0x0068, "LG w1", ALC880_FIXUP_LG),
Takashi Iwaidb8a38e2013-08-09 12:34:42 +02001506 SND_PCI_QUIRK(0x1854, 0x0077, "LG LW25", ALC880_FIXUP_LG_LW25),
Takashi Iwaib9368f52012-02-17 17:54:44 +01001507 SND_PCI_QUIRK(0x19db, 0x4188, "TCL S700", ALC880_FIXUP_TCL_S700),
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001508
1509 /* Below is the copied entries from alc880_quirks.c.
1510 * It's not quite sure whether BIOS sets the correct pin-config table
1511 * on these machines, thus they are kept to be compatible with
1512 * the old static quirks. Once when it's confirmed to work without
1513 * these overrides, it'd be better to remove.
1514 */
1515 SND_PCI_QUIRK(0x1019, 0xa880, "ECS", ALC880_FIXUP_5ST_DIG),
1516 SND_PCI_QUIRK(0x1019, 0xa884, "Acer APFV", ALC880_FIXUP_6ST),
1517 SND_PCI_QUIRK(0x1025, 0x0070, "ULI", ALC880_FIXUP_3ST_DIG),
1518 SND_PCI_QUIRK(0x1025, 0x0077, "ULI", ALC880_FIXUP_6ST_DIG),
1519 SND_PCI_QUIRK(0x1025, 0x0078, "ULI", ALC880_FIXUP_6ST_DIG),
1520 SND_PCI_QUIRK(0x1025, 0x0087, "ULI", ALC880_FIXUP_6ST_DIG),
1521 SND_PCI_QUIRK(0x1025, 0xe309, "ULI", ALC880_FIXUP_3ST_DIG),
1522 SND_PCI_QUIRK(0x1025, 0xe310, "ULI", ALC880_FIXUP_3ST),
1523 SND_PCI_QUIRK(0x1039, 0x1234, NULL, ALC880_FIXUP_6ST_DIG),
1524 SND_PCI_QUIRK(0x104d, 0x81a0, "Sony", ALC880_FIXUP_3ST),
1525 SND_PCI_QUIRK(0x104d, 0x81d6, "Sony", ALC880_FIXUP_3ST),
1526 SND_PCI_QUIRK(0x107b, 0x3032, "Gateway", ALC880_FIXUP_5ST),
1527 SND_PCI_QUIRK(0x107b, 0x3033, "Gateway", ALC880_FIXUP_5ST),
1528 SND_PCI_QUIRK(0x107b, 0x4039, "Gateway", ALC880_FIXUP_5ST),
1529 SND_PCI_QUIRK(0x1297, 0xc790, "Shuttle ST20G5", ALC880_FIXUP_6ST_DIG),
1530 SND_PCI_QUIRK(0x1458, 0xa102, "Gigabyte K8", ALC880_FIXUP_6ST_DIG),
1531 SND_PCI_QUIRK(0x1462, 0x1150, "MSI", ALC880_FIXUP_6ST_DIG),
1532 SND_PCI_QUIRK(0x1509, 0x925d, "FIC P4M", ALC880_FIXUP_6ST_DIG),
1533 SND_PCI_QUIRK(0x1565, 0x8202, "Biostar", ALC880_FIXUP_5ST_DIG),
1534 SND_PCI_QUIRK(0x1695, 0x400d, "EPoX", ALC880_FIXUP_5ST_DIG),
1535 SND_PCI_QUIRK(0x1695, 0x4012, "EPox EP-5LDA", ALC880_FIXUP_5ST_DIG),
1536 SND_PCI_QUIRK(0x2668, 0x8086, NULL, ALC880_FIXUP_6ST_DIG), /* broken BIOS */
1537 SND_PCI_QUIRK(0x8086, 0x2668, NULL, ALC880_FIXUP_6ST_DIG),
1538 SND_PCI_QUIRK(0x8086, 0xa100, "Intel mobo", ALC880_FIXUP_5ST_DIG),
1539 SND_PCI_QUIRK(0x8086, 0xd400, "Intel mobo", ALC880_FIXUP_5ST_DIG),
1540 SND_PCI_QUIRK(0x8086, 0xd401, "Intel mobo", ALC880_FIXUP_5ST_DIG),
1541 SND_PCI_QUIRK(0x8086, 0xd402, "Intel mobo", ALC880_FIXUP_3ST_DIG),
1542 SND_PCI_QUIRK(0x8086, 0xe224, "Intel mobo", ALC880_FIXUP_5ST_DIG),
1543 SND_PCI_QUIRK(0x8086, 0xe305, "Intel mobo", ALC880_FIXUP_3ST_DIG),
1544 SND_PCI_QUIRK(0x8086, 0xe308, "Intel mobo", ALC880_FIXUP_3ST_DIG),
1545 SND_PCI_QUIRK(0x8086, 0xe400, "Intel mobo", ALC880_FIXUP_5ST_DIG),
1546 SND_PCI_QUIRK(0x8086, 0xe401, "Intel mobo", ALC880_FIXUP_5ST_DIG),
1547 SND_PCI_QUIRK(0x8086, 0xe402, "Intel mobo", ALC880_FIXUP_5ST_DIG),
1548 /* default Intel */
1549 SND_PCI_QUIRK_VENDOR(0x8086, "Intel mobo", ALC880_FIXUP_3ST),
1550 SND_PCI_QUIRK(0xa0a0, 0x0560, "AOpen i915GMm-HFS", ALC880_FIXUP_5ST_DIG),
1551 SND_PCI_QUIRK(0xe803, 0x1019, NULL, ALC880_FIXUP_6ST_DIG),
1552 {}
1553};
1554
Takashi Iwai1727a772013-01-10 09:52:52 +01001555static const struct hda_model_fixup alc880_fixup_models[] = {
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001556 {.id = ALC880_FIXUP_3ST, .name = "3stack"},
1557 {.id = ALC880_FIXUP_3ST_DIG, .name = "3stack-digout"},
1558 {.id = ALC880_FIXUP_5ST, .name = "5stack"},
1559 {.id = ALC880_FIXUP_5ST_DIG, .name = "5stack-digout"},
1560 {.id = ALC880_FIXUP_6ST, .name = "6stack"},
1561 {.id = ALC880_FIXUP_6ST_DIG, .name = "6stack-digout"},
Takashi Iwai53971452013-01-23 18:21:37 +01001562 {.id = ALC880_FIXUP_6ST_AUTOMUTE, .name = "6stack-automute"},
Takashi Iwaiee3b2962011-11-15 14:26:54 +01001563 {}
1564};
1565
1566
1567/*
Takashi Iwai1d045db2011-07-07 18:23:21 +02001568 * OK, here we have finally the patch for ALC880
1569 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02001570static int patch_alc880(struct hda_codec *codec)
1571{
1572 struct alc_spec *spec;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001573 int err;
1574
Takashi Iwai3de95172012-05-07 18:03:15 +02001575 err = alc_alloc_spec(codec, 0x0b);
1576 if (err < 0)
1577 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001578
Takashi Iwai3de95172012-05-07 18:03:15 +02001579 spec = codec->spec;
Takashi Iwai08c189f2012-12-19 15:22:24 +01001580 spec->gen.need_dac_fix = 1;
Takashi Iwai7504b6c2013-03-18 11:25:51 +01001581 spec->gen.beep_nid = 0x01;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001582
Takashi Iwai225068a2015-05-29 10:42:14 +02001583 codec->patch_ops.unsol_event = alc880_unsol_event;
1584
Takashi Iwaic9af7532019-05-10 11:01:43 +02001585 alc_pre_init(codec);
1586
Takashi Iwai1727a772013-01-10 09:52:52 +01001587 snd_hda_pick_fixup(codec, alc880_fixup_models, alc880_fixup_tbl,
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001588 alc880_fixups);
Takashi Iwai1727a772013-01-10 09:52:52 +01001589 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
Takashi Iwai1d045db2011-07-07 18:23:21 +02001590
Takashi Iwai67b6ec32012-02-20 18:20:42 +01001591 /* automatic parse from the BIOS config */
1592 err = alc880_parse_auto_config(codec);
1593 if (err < 0)
1594 goto error;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001595
Takashi Iwaifea80fa2018-06-20 12:52:46 +02001596 if (!spec->gen.no_analog) {
1597 err = set_beep_amp(spec, 0x0b, 0x05, HDA_INPUT);
1598 if (err < 0)
1599 goto error;
1600 }
Takashi Iwai1d045db2011-07-07 18:23:21 +02001601
Takashi Iwai1727a772013-01-10 09:52:52 +01001602 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
Takashi Iwai589876e2012-02-20 15:47:55 +01001603
Takashi Iwai1d045db2011-07-07 18:23:21 +02001604 return 0;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02001605
1606 error:
1607 alc_free(codec);
1608 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001609}
1610
1611
1612/*
1613 * ALC260 support
1614 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02001615static int alc260_parse_auto_config(struct hda_codec *codec)
1616{
Takashi Iwai1d045db2011-07-07 18:23:21 +02001617 static const hda_nid_t alc260_ignore[] = { 0x17, 0 };
Takashi Iwai3e6179b2011-07-08 16:55:13 +02001618 static const hda_nid_t alc260_ssids[] = { 0x10, 0x15, 0x0f, 0 };
1619 return alc_parse_auto_config(codec, alc260_ignore, alc260_ssids);
Takashi Iwai1d045db2011-07-07 18:23:21 +02001620}
1621
Takashi Iwai1d045db2011-07-07 18:23:21 +02001622/*
1623 * Pin config fixes
1624 */
1625enum {
Takashi Iwaica8f0422012-02-16 11:51:19 +01001626 ALC260_FIXUP_HP_DC5750,
1627 ALC260_FIXUP_HP_PIN_0F,
1628 ALC260_FIXUP_COEF,
Takashi Iwai15317ab2012-02-16 12:02:53 +01001629 ALC260_FIXUP_GPIO1,
Takashi Iwai20f7d922012-02-16 12:35:16 +01001630 ALC260_FIXUP_GPIO1_TOGGLE,
1631 ALC260_FIXUP_REPLACER,
Takashi Iwai0a1c4fa2012-02-16 12:42:30 +01001632 ALC260_FIXUP_HP_B1900,
Takashi Iwai118cb4a2012-04-19 07:33:27 +02001633 ALC260_FIXUP_KN1,
Takashi Iwai39aedee2013-01-10 17:10:40 +01001634 ALC260_FIXUP_FSC_S7020,
Takashi Iwai5ebd3bb2013-02-19 18:23:31 +01001635 ALC260_FIXUP_FSC_S7020_JWSE,
Takashi Iwaid08c5ef2013-11-22 08:06:36 +01001636 ALC260_FIXUP_VAIO_PINS,
Takashi Iwai1d045db2011-07-07 18:23:21 +02001637};
1638
Takashi Iwai20f7d922012-02-16 12:35:16 +01001639static void alc260_gpio1_automute(struct hda_codec *codec)
1640{
1641 struct alc_spec *spec = codec->spec;
Takashi Iwaiaaf312d2018-06-19 22:28:22 +02001642
1643 alc_update_gpio_data(codec, 0x01, spec->gen.hp_jack_present);
Takashi Iwai20f7d922012-02-16 12:35:16 +01001644}
1645
1646static void alc260_fixup_gpio1_toggle(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01001647 const struct hda_fixup *fix, int action)
Takashi Iwai20f7d922012-02-16 12:35:16 +01001648{
1649 struct alc_spec *spec = codec->spec;
Takashi Iwai1727a772013-01-10 09:52:52 +01001650 if (action == HDA_FIXUP_ACT_PROBE) {
Takashi Iwai20f7d922012-02-16 12:35:16 +01001651 /* although the machine has only one output pin, we need to
1652 * toggle GPIO1 according to the jack state
1653 */
Takashi Iwai08c189f2012-12-19 15:22:24 +01001654 spec->gen.automute_hook = alc260_gpio1_automute;
1655 spec->gen.detect_hp = 1;
1656 spec->gen.automute_speaker = 1;
1657 spec->gen.autocfg.hp_pins[0] = 0x0f; /* copy it for automute */
Takashi Iwai62f949b2014-09-11 14:06:53 +02001658 snd_hda_jack_detect_enable_callback(codec, 0x0f,
Takashi Iwai08c189f2012-12-19 15:22:24 +01001659 snd_hda_gen_hp_automute);
Takashi Iwai5579cd62018-06-19 22:22:41 +02001660 alc_setup_gpio(codec, 0x01);
Takashi Iwai20f7d922012-02-16 12:35:16 +01001661 }
1662}
1663
Takashi Iwai118cb4a2012-04-19 07:33:27 +02001664static void alc260_fixup_kn1(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01001665 const struct hda_fixup *fix, int action)
Takashi Iwai118cb4a2012-04-19 07:33:27 +02001666{
1667 struct alc_spec *spec = codec->spec;
Takashi Iwai1727a772013-01-10 09:52:52 +01001668 static const struct hda_pintbl pincfgs[] = {
Takashi Iwai118cb4a2012-04-19 07:33:27 +02001669 { 0x0f, 0x02214000 }, /* HP/speaker */
1670 { 0x12, 0x90a60160 }, /* int mic */
1671 { 0x13, 0x02a19000 }, /* ext mic */
1672 { 0x18, 0x01446000 }, /* SPDIF out */
1673 /* disable bogus I/O pins */
1674 { 0x10, 0x411111f0 },
1675 { 0x11, 0x411111f0 },
1676 { 0x14, 0x411111f0 },
1677 { 0x15, 0x411111f0 },
1678 { 0x16, 0x411111f0 },
1679 { 0x17, 0x411111f0 },
1680 { 0x19, 0x411111f0 },
1681 { }
1682 };
1683
1684 switch (action) {
Takashi Iwai1727a772013-01-10 09:52:52 +01001685 case HDA_FIXUP_ACT_PRE_PROBE:
1686 snd_hda_apply_pincfgs(codec, pincfgs);
Takashi Iwai118cb4a2012-04-19 07:33:27 +02001687 spec->init_amp = ALC_INIT_NONE;
1688 break;
1689 }
1690}
1691
Takashi Iwai39aedee2013-01-10 17:10:40 +01001692static void alc260_fixup_fsc_s7020(struct hda_codec *codec,
1693 const struct hda_fixup *fix, int action)
1694{
1695 struct alc_spec *spec = codec->spec;
Takashi Iwai1c76aa52018-06-21 16:37:54 +02001696 if (action == HDA_FIXUP_ACT_PRE_PROBE)
Takashi Iwaie6e0ee52013-02-18 17:04:20 +01001697 spec->init_amp = ALC_INIT_NONE;
Takashi Iwai5ebd3bb2013-02-19 18:23:31 +01001698}
1699
1700static void alc260_fixup_fsc_s7020_jwse(struct hda_codec *codec,
1701 const struct hda_fixup *fix, int action)
1702{
1703 struct alc_spec *spec = codec->spec;
1704 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwaif811c3c2013-03-07 18:32:59 +01001705 spec->gen.add_jack_modes = 1;
Takashi Iwai5ebd3bb2013-02-19 18:23:31 +01001706 spec->gen.hp_mic = 1;
Takashi Iwaie6e0ee52013-02-18 17:04:20 +01001707 }
Takashi Iwai39aedee2013-01-10 17:10:40 +01001708}
1709
Takashi Iwai1727a772013-01-10 09:52:52 +01001710static const struct hda_fixup alc260_fixups[] = {
Takashi Iwaica8f0422012-02-16 11:51:19 +01001711 [ALC260_FIXUP_HP_DC5750] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001712 .type = HDA_FIXUP_PINS,
1713 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02001714 { 0x11, 0x90130110 }, /* speaker */
1715 { }
1716 }
1717 },
Takashi Iwaica8f0422012-02-16 11:51:19 +01001718 [ALC260_FIXUP_HP_PIN_0F] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001719 .type = HDA_FIXUP_PINS,
1720 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaica8f0422012-02-16 11:51:19 +01001721 { 0x0f, 0x01214000 }, /* HP */
1722 { }
1723 }
1724 },
1725 [ALC260_FIXUP_COEF] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001726 .type = HDA_FIXUP_VERBS,
Takashi Iwaica8f0422012-02-16 11:51:19 +01001727 .v.verbs = (const struct hda_verb[]) {
Ronan Marquete30cf2d2014-06-01 18:38:53 +02001728 { 0x1a, AC_VERB_SET_COEF_INDEX, 0x07 },
1729 { 0x1a, AC_VERB_SET_PROC_COEF, 0x3040 },
Takashi Iwaica8f0422012-02-16 11:51:19 +01001730 { }
1731 },
Takashi Iwaica8f0422012-02-16 11:51:19 +01001732 },
Takashi Iwai15317ab2012-02-16 12:02:53 +01001733 [ALC260_FIXUP_GPIO1] = {
Takashi Iwai5579cd62018-06-19 22:22:41 +02001734 .type = HDA_FIXUP_FUNC,
1735 .v.func = alc_fixup_gpio1,
Takashi Iwai15317ab2012-02-16 12:02:53 +01001736 },
Takashi Iwai20f7d922012-02-16 12:35:16 +01001737 [ALC260_FIXUP_GPIO1_TOGGLE] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001738 .type = HDA_FIXUP_FUNC,
Takashi Iwai20f7d922012-02-16 12:35:16 +01001739 .v.func = alc260_fixup_gpio1_toggle,
1740 .chained = true,
1741 .chain_id = ALC260_FIXUP_HP_PIN_0F,
1742 },
1743 [ALC260_FIXUP_REPLACER] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001744 .type = HDA_FIXUP_VERBS,
Takashi Iwai20f7d922012-02-16 12:35:16 +01001745 .v.verbs = (const struct hda_verb[]) {
Takashi Iwai192a98e2014-06-02 15:16:07 +02001746 { 0x1a, AC_VERB_SET_COEF_INDEX, 0x07 },
1747 { 0x1a, AC_VERB_SET_PROC_COEF, 0x3050 },
Takashi Iwai20f7d922012-02-16 12:35:16 +01001748 { }
1749 },
1750 .chained = true,
1751 .chain_id = ALC260_FIXUP_GPIO1_TOGGLE,
1752 },
Takashi Iwai0a1c4fa2012-02-16 12:42:30 +01001753 [ALC260_FIXUP_HP_B1900] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001754 .type = HDA_FIXUP_FUNC,
Takashi Iwai0a1c4fa2012-02-16 12:42:30 +01001755 .v.func = alc260_fixup_gpio1_toggle,
1756 .chained = true,
1757 .chain_id = ALC260_FIXUP_COEF,
Takashi Iwai118cb4a2012-04-19 07:33:27 +02001758 },
1759 [ALC260_FIXUP_KN1] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01001760 .type = HDA_FIXUP_FUNC,
Takashi Iwai118cb4a2012-04-19 07:33:27 +02001761 .v.func = alc260_fixup_kn1,
1762 },
Takashi Iwai39aedee2013-01-10 17:10:40 +01001763 [ALC260_FIXUP_FSC_S7020] = {
1764 .type = HDA_FIXUP_FUNC,
1765 .v.func = alc260_fixup_fsc_s7020,
1766 },
Takashi Iwai5ebd3bb2013-02-19 18:23:31 +01001767 [ALC260_FIXUP_FSC_S7020_JWSE] = {
1768 .type = HDA_FIXUP_FUNC,
1769 .v.func = alc260_fixup_fsc_s7020_jwse,
1770 .chained = true,
1771 .chain_id = ALC260_FIXUP_FSC_S7020,
1772 },
Takashi Iwaid08c5ef2013-11-22 08:06:36 +01001773 [ALC260_FIXUP_VAIO_PINS] = {
1774 .type = HDA_FIXUP_PINS,
1775 .v.pins = (const struct hda_pintbl[]) {
1776 /* Pin configs are missing completely on some VAIOs */
1777 { 0x0f, 0x01211020 },
1778 { 0x10, 0x0001003f },
1779 { 0x11, 0x411111f0 },
1780 { 0x12, 0x01a15930 },
1781 { 0x13, 0x411111f0 },
1782 { 0x14, 0x411111f0 },
1783 { 0x15, 0x411111f0 },
1784 { 0x16, 0x411111f0 },
1785 { 0x17, 0x411111f0 },
1786 { 0x18, 0x411111f0 },
1787 { 0x19, 0x411111f0 },
1788 { }
1789 }
1790 },
Takashi Iwai1d045db2011-07-07 18:23:21 +02001791};
1792
1793static const struct snd_pci_quirk alc260_fixup_tbl[] = {
Takashi Iwai15317ab2012-02-16 12:02:53 +01001794 SND_PCI_QUIRK(0x1025, 0x007b, "Acer C20x", ALC260_FIXUP_GPIO1),
Takashi Iwaica8f0422012-02-16 11:51:19 +01001795 SND_PCI_QUIRK(0x1025, 0x007f, "Acer Aspire 9500", ALC260_FIXUP_COEF),
Takashi Iwai15317ab2012-02-16 12:02:53 +01001796 SND_PCI_QUIRK(0x1025, 0x008f, "Acer", ALC260_FIXUP_GPIO1),
Takashi Iwaica8f0422012-02-16 11:51:19 +01001797 SND_PCI_QUIRK(0x103c, 0x280a, "HP dc5750", ALC260_FIXUP_HP_DC5750),
Takashi Iwai0a1c4fa2012-02-16 12:42:30 +01001798 SND_PCI_QUIRK(0x103c, 0x30ba, "HP Presario B1900", ALC260_FIXUP_HP_B1900),
Takashi Iwaid08c5ef2013-11-22 08:06:36 +01001799 SND_PCI_QUIRK(0x104d, 0x81bb, "Sony VAIO", ALC260_FIXUP_VAIO_PINS),
Takashi Iwai0f5a5b82013-11-21 09:12:52 +01001800 SND_PCI_QUIRK(0x104d, 0x81e2, "Sony VAIO TX", ALC260_FIXUP_HP_PIN_0F),
Takashi Iwai39aedee2013-01-10 17:10:40 +01001801 SND_PCI_QUIRK(0x10cf, 0x1326, "FSC LifeBook S7020", ALC260_FIXUP_FSC_S7020),
Takashi Iwaib1f58082012-02-16 12:45:03 +01001802 SND_PCI_QUIRK(0x1509, 0x4540, "Favorit 100XS", ALC260_FIXUP_GPIO1),
Takashi Iwai118cb4a2012-04-19 07:33:27 +02001803 SND_PCI_QUIRK(0x152d, 0x0729, "Quanta KN1", ALC260_FIXUP_KN1),
Takashi Iwai20f7d922012-02-16 12:35:16 +01001804 SND_PCI_QUIRK(0x161f, 0x2057, "Replacer 672V", ALC260_FIXUP_REPLACER),
Takashi Iwaica8f0422012-02-16 11:51:19 +01001805 SND_PCI_QUIRK(0x1631, 0xc017, "PB V7900", ALC260_FIXUP_COEF),
Takashi Iwai1d045db2011-07-07 18:23:21 +02001806 {}
1807};
1808
Takashi Iwai5ebd3bb2013-02-19 18:23:31 +01001809static const struct hda_model_fixup alc260_fixup_models[] = {
1810 {.id = ALC260_FIXUP_GPIO1, .name = "gpio1"},
1811 {.id = ALC260_FIXUP_COEF, .name = "coef"},
1812 {.id = ALC260_FIXUP_FSC_S7020, .name = "fujitsu"},
1813 {.id = ALC260_FIXUP_FSC_S7020_JWSE, .name = "fujitsu-jwse"},
1814 {}
1815};
1816
Takashi Iwai1d045db2011-07-07 18:23:21 +02001817/*
1818 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02001819static int patch_alc260(struct hda_codec *codec)
1820{
1821 struct alc_spec *spec;
Takashi Iwaic3c2c9e2012-02-16 12:59:55 +01001822 int err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001823
Takashi Iwai3de95172012-05-07 18:03:15 +02001824 err = alc_alloc_spec(codec, 0x07);
1825 if (err < 0)
1826 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001827
Takashi Iwai3de95172012-05-07 18:03:15 +02001828 spec = codec->spec;
Takashi Iwaiea46c3c2013-01-15 18:45:53 +01001829 /* as quite a few machines require HP amp for speaker outputs,
1830 * it's easier to enable it unconditionally; even if it's unneeded,
1831 * it's almost harmless.
1832 */
1833 spec->gen.prefer_hp_amp = 1;
Takashi Iwai7504b6c2013-03-18 11:25:51 +01001834 spec->gen.beep_nid = 0x01;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001835
Takashi Iwai225068a2015-05-29 10:42:14 +02001836 spec->shutup = alc_eapd_shutup;
1837
Takashi Iwaic9af7532019-05-10 11:01:43 +02001838 alc_pre_init(codec);
1839
Takashi Iwai5ebd3bb2013-02-19 18:23:31 +01001840 snd_hda_pick_fixup(codec, alc260_fixup_models, alc260_fixup_tbl,
1841 alc260_fixups);
Takashi Iwai1727a772013-01-10 09:52:52 +01001842 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
Takashi Iwai1d045db2011-07-07 18:23:21 +02001843
Takashi Iwaic3c2c9e2012-02-16 12:59:55 +01001844 /* automatic parse from the BIOS config */
1845 err = alc260_parse_auto_config(codec);
1846 if (err < 0)
1847 goto error;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001848
Takashi Iwaifea80fa2018-06-20 12:52:46 +02001849 if (!spec->gen.no_analog) {
1850 err = set_beep_amp(spec, 0x07, 0x05, HDA_INPUT);
1851 if (err < 0)
1852 goto error;
1853 }
Takashi Iwai1d045db2011-07-07 18:23:21 +02001854
Takashi Iwai1727a772013-01-10 09:52:52 +01001855 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
Takashi Iwai589876e2012-02-20 15:47:55 +01001856
Takashi Iwai1d045db2011-07-07 18:23:21 +02001857 return 0;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02001858
1859 error:
1860 alc_free(codec);
1861 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02001862}
1863
1864
1865/*
1866 * ALC882/883/885/888/889 support
1867 *
1868 * ALC882 is almost identical with ALC880 but has cleaner and more flexible
1869 * configuration. Each pin widget can choose any input DACs and a mixer.
1870 * Each ADC is connected from a mixer of all inputs. This makes possible
1871 * 6-channel independent captures.
1872 *
1873 * In addition, an independent DAC for the multi-playback (not used in this
1874 * driver yet).
1875 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02001876
1877/*
1878 * Pin config fixes
1879 */
1880enum {
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01001881 ALC882_FIXUP_ABIT_AW9D_MAX,
1882 ALC882_FIXUP_LENOVO_Y530,
1883 ALC882_FIXUP_PB_M5210,
1884 ALC882_FIXUP_ACER_ASPIRE_7736,
1885 ALC882_FIXUP_ASUS_W90V,
Marton Balint8f239212012-03-05 21:33:23 +01001886 ALC889_FIXUP_CD,
David Henningssonb2c53e22014-01-01 14:01:34 +01001887 ALC889_FIXUP_FRONT_HP_NO_PRESENCE,
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01001888 ALC889_FIXUP_VAIO_TT,
Takashi Iwai0e7cc2e2011-11-09 12:42:48 +01001889 ALC888_FIXUP_EEE1601,
Takashi Iwai177943a32011-11-09 12:55:18 +01001890 ALC882_FIXUP_EAPD,
Takashi Iwai7a6069b2011-11-09 15:22:01 +01001891 ALC883_FIXUP_EAPD,
Takashi Iwai8812c4f2011-11-09 17:39:15 +01001892 ALC883_FIXUP_ACER_EAPD,
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001893 ALC882_FIXUP_GPIO1,
1894 ALC882_FIXUP_GPIO2,
Takashi Iwaieb844d52011-11-09 18:03:07 +01001895 ALC882_FIXUP_GPIO3,
Takashi Iwai68ef0562011-11-09 18:24:44 +01001896 ALC889_FIXUP_COEF,
1897 ALC882_FIXUP_ASUS_W2JC,
Takashi Iwaic3e837b2011-11-10 16:01:47 +01001898 ALC882_FIXUP_ACER_ASPIRE_4930G,
1899 ALC882_FIXUP_ACER_ASPIRE_8930G,
1900 ALC882_FIXUP_ASPIRE_8930G_VERBS,
Takashi Iwai56710872011-11-14 17:42:11 +01001901 ALC885_FIXUP_MACPRO_GPIO,
Takashi Iwai02a237b2012-02-13 15:25:07 +01001902 ALC889_FIXUP_DAC_ROUTE,
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001903 ALC889_FIXUP_MBP_VREF,
1904 ALC889_FIXUP_IMAC91_VREF,
Adrien Vergée7729a42014-01-24 14:56:14 -05001905 ALC889_FIXUP_MBA11_VREF,
Takashi Iwai0756f092013-12-04 13:59:45 +01001906 ALC889_FIXUP_MBA21_VREF,
Takashi Iwaic20f31e2014-02-03 11:02:10 +01001907 ALC889_FIXUP_MP11_VREF,
Mario Kleiner9f660a12015-12-22 00:45:43 +01001908 ALC889_FIXUP_MP41_VREF,
Takashi Iwai6e72aa52012-06-25 10:52:25 +02001909 ALC882_FIXUP_INV_DMIC,
Takashi Iwaie427c232012-07-29 10:04:08 +02001910 ALC882_FIXUP_NO_PRIMARY_HP,
Takashi Iwai1f0bbf02013-11-28 15:21:21 +01001911 ALC887_FIXUP_ASUS_BASS,
Takashi Iwaieb9ca3a2013-11-28 15:24:34 +01001912 ALC887_FIXUP_BASS_CHMAP,
Takashi Iwai7beb3a62017-04-10 18:05:52 +02001913 ALC1220_FIXUP_GB_DUAL_CODECS,
Peisen0202f5c2017-10-26 10:35:36 +08001914 ALC1220_FIXUP_CLEVO_P950,
Richard Sailer80690a22019-04-02 15:52:04 +02001915 ALC1220_FIXUP_CLEVO_PB51ED,
1916 ALC1220_FIXUP_CLEVO_PB51ED_PINS,
Takashi Iwai1d045db2011-07-07 18:23:21 +02001917};
1918
Takashi Iwai68ef0562011-11-09 18:24:44 +01001919static void alc889_fixup_coef(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01001920 const struct hda_fixup *fix, int action)
Takashi Iwai68ef0562011-11-09 18:24:44 +01001921{
Takashi Iwai1727a772013-01-10 09:52:52 +01001922 if (action != HDA_FIXUP_ACT_INIT)
Takashi Iwai68ef0562011-11-09 18:24:44 +01001923 return;
Kailang Yang1df88742014-10-29 16:10:13 +08001924 alc_update_coef_idx(codec, 7, 0, 0x2030);
Takashi Iwai68ef0562011-11-09 18:24:44 +01001925}
1926
Takashi Iwai56710872011-11-14 17:42:11 +01001927/* set up GPIO at initialization */
1928static void alc885_fixup_macpro_gpio(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01001929 const struct hda_fixup *fix, int action)
Takashi Iwai56710872011-11-14 17:42:11 +01001930{
Takashi Iwai215c8502018-06-19 22:34:26 +02001931 struct alc_spec *spec = codec->spec;
1932
1933 spec->gpio_write_delay = true;
1934 alc_fixup_gpio3(codec, fix, action);
Takashi Iwai56710872011-11-14 17:42:11 +01001935}
1936
Takashi Iwai02a237b2012-02-13 15:25:07 +01001937/* Fix the connection of some pins for ALC889:
1938 * At least, Acer Aspire 5935 shows the connections to DAC3/4 don't
1939 * work correctly (bko#42740)
1940 */
1941static void alc889_fixup_dac_route(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01001942 const struct hda_fixup *fix, int action)
Takashi Iwai02a237b2012-02-13 15:25:07 +01001943{
Takashi Iwai1727a772013-01-10 09:52:52 +01001944 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwaief8d60f2012-02-17 10:12:38 +01001945 /* fake the connections during parsing the tree */
Takashi Iwai02a237b2012-02-13 15:25:07 +01001946 hda_nid_t conn1[2] = { 0x0c, 0x0d };
1947 hda_nid_t conn2[2] = { 0x0e, 0x0f };
1948 snd_hda_override_conn_list(codec, 0x14, 2, conn1);
1949 snd_hda_override_conn_list(codec, 0x15, 2, conn1);
1950 snd_hda_override_conn_list(codec, 0x18, 2, conn2);
1951 snd_hda_override_conn_list(codec, 0x1a, 2, conn2);
Takashi Iwai1727a772013-01-10 09:52:52 +01001952 } else if (action == HDA_FIXUP_ACT_PROBE) {
Takashi Iwaief8d60f2012-02-17 10:12:38 +01001953 /* restore the connections */
1954 hda_nid_t conn[5] = { 0x0c, 0x0d, 0x0e, 0x0f, 0x26 };
1955 snd_hda_override_conn_list(codec, 0x14, 5, conn);
1956 snd_hda_override_conn_list(codec, 0x15, 5, conn);
1957 snd_hda_override_conn_list(codec, 0x18, 5, conn);
1958 snd_hda_override_conn_list(codec, 0x1a, 5, conn);
Takashi Iwai02a237b2012-02-13 15:25:07 +01001959 }
1960}
1961
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001962/* Set VREF on HP pin */
1963static void alc889_fixup_mbp_vref(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01001964 const struct hda_fixup *fix, int action)
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001965{
1966 struct alc_spec *spec = codec->spec;
Mario Kleiner9f660a12015-12-22 00:45:43 +01001967 static hda_nid_t nids[3] = { 0x14, 0x15, 0x19 };
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001968 int i;
1969
Takashi Iwai1727a772013-01-10 09:52:52 +01001970 if (action != HDA_FIXUP_ACT_INIT)
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001971 return;
1972 for (i = 0; i < ARRAY_SIZE(nids); i++) {
1973 unsigned int val = snd_hda_codec_get_pincfg(codec, nids[i]);
1974 if (get_defcfg_device(val) != AC_JACK_HP_OUT)
1975 continue;
Takashi Iwaid3f02d62013-01-10 10:12:22 +01001976 val = snd_hda_codec_get_pin_target(codec, nids[i]);
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001977 val |= AC_PINCTL_VREF_80;
Takashi Iwaicdd03ce2012-04-20 12:34:50 +02001978 snd_hda_set_pin_ctl(codec, nids[i], val);
Takashi Iwai08c189f2012-12-19 15:22:24 +01001979 spec->gen.keep_vref_in_automute = 1;
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001980 break;
1981 }
1982}
1983
Takashi Iwai0756f092013-12-04 13:59:45 +01001984static void alc889_fixup_mac_pins(struct hda_codec *codec,
1985 const hda_nid_t *nids, int num_nids)
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001986{
1987 struct alc_spec *spec = codec->spec;
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001988 int i;
1989
Takashi Iwai0756f092013-12-04 13:59:45 +01001990 for (i = 0; i < num_nids; i++) {
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001991 unsigned int val;
Takashi Iwaid3f02d62013-01-10 10:12:22 +01001992 val = snd_hda_codec_get_pin_target(codec, nids[i]);
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001993 val |= AC_PINCTL_VREF_50;
Takashi Iwaicdd03ce2012-04-20 12:34:50 +02001994 snd_hda_set_pin_ctl(codec, nids[i], val);
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001995 }
Takashi Iwai08c189f2012-12-19 15:22:24 +01001996 spec->gen.keep_vref_in_automute = 1;
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01001997}
1998
Takashi Iwai0756f092013-12-04 13:59:45 +01001999/* Set VREF on speaker pins on imac91 */
2000static void alc889_fixup_imac91_vref(struct hda_codec *codec,
2001 const struct hda_fixup *fix, int action)
2002{
2003 static hda_nid_t nids[2] = { 0x18, 0x1a };
2004
2005 if (action == HDA_FIXUP_ACT_INIT)
2006 alc889_fixup_mac_pins(codec, nids, ARRAY_SIZE(nids));
2007}
2008
Adrien Vergée7729a42014-01-24 14:56:14 -05002009/* Set VREF on speaker pins on mba11 */
2010static void alc889_fixup_mba11_vref(struct hda_codec *codec,
2011 const struct hda_fixup *fix, int action)
2012{
2013 static hda_nid_t nids[1] = { 0x18 };
2014
2015 if (action == HDA_FIXUP_ACT_INIT)
2016 alc889_fixup_mac_pins(codec, nids, ARRAY_SIZE(nids));
2017}
2018
Takashi Iwai0756f092013-12-04 13:59:45 +01002019/* Set VREF on speaker pins on mba21 */
2020static void alc889_fixup_mba21_vref(struct hda_codec *codec,
2021 const struct hda_fixup *fix, int action)
2022{
2023 static hda_nid_t nids[2] = { 0x18, 0x19 };
2024
2025 if (action == HDA_FIXUP_ACT_INIT)
2026 alc889_fixup_mac_pins(codec, nids, ARRAY_SIZE(nids));
2027}
2028
Takashi Iwaie427c232012-07-29 10:04:08 +02002029/* Don't take HP output as primary
Fernando Luis Vázquez Caod9111492013-02-12 16:49:46 +09002030 * Strangely, the speaker output doesn't work on Vaio Z and some Vaio
2031 * all-in-one desktop PCs (for example VGC-LN51JGB) through DAC 0x05
Takashi Iwaie427c232012-07-29 10:04:08 +02002032 */
2033static void alc882_fixup_no_primary_hp(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01002034 const struct hda_fixup *fix, int action)
Takashi Iwaie427c232012-07-29 10:04:08 +02002035{
2036 struct alc_spec *spec = codec->spec;
Takashi Iwaida96fb52013-07-29 16:54:36 +02002037 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwai08c189f2012-12-19 15:22:24 +01002038 spec->gen.no_primary_hp = 1;
Takashi Iwaida96fb52013-07-29 16:54:36 +02002039 spec->gen.no_multi_io = 1;
2040 }
Takashi Iwaie427c232012-07-29 10:04:08 +02002041}
2042
Takashi Iwaieb9ca3a2013-11-28 15:24:34 +01002043static void alc_fixup_bass_chmap(struct hda_codec *codec,
2044 const struct hda_fixup *fix, int action);
2045
Takashi Iwai7beb3a62017-04-10 18:05:52 +02002046/* For dual-codec configuration, we need to disable some features to avoid
2047 * conflicts of kctls and PCM streams
2048 */
2049static void alc_fixup_dual_codecs(struct hda_codec *codec,
2050 const struct hda_fixup *fix, int action)
2051{
2052 struct alc_spec *spec = codec->spec;
2053
2054 if (action != HDA_FIXUP_ACT_PRE_PROBE)
2055 return;
2056 /* disable vmaster */
2057 spec->gen.suppress_vmaster = 1;
2058 /* auto-mute and auto-mic switch don't work with multiple codecs */
2059 spec->gen.suppress_auto_mute = 1;
2060 spec->gen.suppress_auto_mic = 1;
2061 /* disable aamix as well */
2062 spec->gen.mixer_nid = 0;
2063 /* add location prefix to avoid conflicts */
2064 codec->force_pin_prefix = 1;
2065}
2066
2067static void rename_ctl(struct hda_codec *codec, const char *oldname,
2068 const char *newname)
2069{
2070 struct snd_kcontrol *kctl;
2071
2072 kctl = snd_hda_find_mixer_ctl(codec, oldname);
2073 if (kctl)
2074 strcpy(kctl->id.name, newname);
2075}
2076
2077static void alc1220_fixup_gb_dual_codecs(struct hda_codec *codec,
2078 const struct hda_fixup *fix,
2079 int action)
2080{
2081 alc_fixup_dual_codecs(codec, fix, action);
2082 switch (action) {
2083 case HDA_FIXUP_ACT_PRE_PROBE:
2084 /* override card longname to provide a unique UCM profile */
2085 strcpy(codec->card->longname, "HDAudio-Gigabyte-ALC1220DualCodecs");
2086 break;
2087 case HDA_FIXUP_ACT_BUILD:
2088 /* rename Capture controls depending on the codec */
2089 rename_ctl(codec, "Capture Volume",
2090 codec->addr == 0 ?
2091 "Rear-Panel Capture Volume" :
2092 "Front-Panel Capture Volume");
2093 rename_ctl(codec, "Capture Switch",
2094 codec->addr == 0 ?
2095 "Rear-Panel Capture Switch" :
2096 "Front-Panel Capture Switch");
2097 break;
2098 }
2099}
2100
Peisen0202f5c2017-10-26 10:35:36 +08002101static void alc1220_fixup_clevo_p950(struct hda_codec *codec,
2102 const struct hda_fixup *fix,
2103 int action)
2104{
2105 hda_nid_t conn1[1] = { 0x0c };
2106
2107 if (action != HDA_FIXUP_ACT_PRE_PROBE)
2108 return;
2109
2110 alc_update_coef_idx(codec, 0x7, 0, 0x3c3);
2111 /* We therefore want to make sure 0x14 (front headphone) and
2112 * 0x1b (speakers) use the stereo DAC 0x02
2113 */
2114 snd_hda_override_conn_list(codec, 0x14, 1, conn1);
2115 snd_hda_override_conn_list(codec, 0x1b, 1, conn1);
2116}
2117
Jeremy Soller7f665b12019-02-13 10:56:19 -07002118static void alc_fixup_headset_mode_no_hp_mic(struct hda_codec *codec,
2119 const struct hda_fixup *fix, int action);
2120
Richard Sailer80690a22019-04-02 15:52:04 +02002121static void alc1220_fixup_clevo_pb51ed(struct hda_codec *codec,
Jeremy Soller7f665b12019-02-13 10:56:19 -07002122 const struct hda_fixup *fix,
2123 int action)
2124{
2125 alc1220_fixup_clevo_p950(codec, fix, action);
2126 alc_fixup_headset_mode_no_hp_mic(codec, fix, action);
2127}
2128
Takashi Iwai1727a772013-01-10 09:52:52 +01002129static const struct hda_fixup alc882_fixups[] = {
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01002130 [ALC882_FIXUP_ABIT_AW9D_MAX] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002131 .type = HDA_FIXUP_PINS,
2132 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02002133 { 0x15, 0x01080104 }, /* side */
2134 { 0x16, 0x01011012 }, /* rear */
2135 { 0x17, 0x01016011 }, /* clfe */
2136 { }
2137 }
2138 },
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01002139 [ALC882_FIXUP_LENOVO_Y530] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002140 .type = HDA_FIXUP_PINS,
2141 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02002142 { 0x15, 0x99130112 }, /* rear int speakers */
2143 { 0x16, 0x99130111 }, /* subwoofer */
2144 { }
2145 }
2146 },
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01002147 [ALC882_FIXUP_PB_M5210] = {
Takashi Iwaifd108212013-01-10 10:18:14 +01002148 .type = HDA_FIXUP_PINCTLS,
2149 .v.pins = (const struct hda_pintbl[]) {
2150 { 0x19, PIN_VREF50 },
Takashi Iwai1d045db2011-07-07 18:23:21 +02002151 {}
2152 }
2153 },
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01002154 [ALC882_FIXUP_ACER_ASPIRE_7736] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002155 .type = HDA_FIXUP_FUNC,
Takashi Iwai23d30f22012-05-07 17:17:32 +02002156 .v.func = alc_fixup_sku_ignore,
Takashi Iwai1d045db2011-07-07 18:23:21 +02002157 },
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01002158 [ALC882_FIXUP_ASUS_W90V] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002159 .type = HDA_FIXUP_PINS,
2160 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai5cdf7452011-10-26 23:04:08 +02002161 { 0x16, 0x99130110 }, /* fix sequence for CLFE */
2162 { }
2163 }
2164 },
Marton Balint8f239212012-03-05 21:33:23 +01002165 [ALC889_FIXUP_CD] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002166 .type = HDA_FIXUP_PINS,
2167 .v.pins = (const struct hda_pintbl[]) {
Marton Balint8f239212012-03-05 21:33:23 +01002168 { 0x1c, 0x993301f0 }, /* CD */
2169 { }
2170 }
2171 },
David Henningssonb2c53e22014-01-01 14:01:34 +01002172 [ALC889_FIXUP_FRONT_HP_NO_PRESENCE] = {
2173 .type = HDA_FIXUP_PINS,
2174 .v.pins = (const struct hda_pintbl[]) {
2175 { 0x1b, 0x02214120 }, /* Front HP jack is flaky, disable jack detect */
2176 { }
2177 },
2178 .chained = true,
2179 .chain_id = ALC889_FIXUP_CD,
2180 },
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01002181 [ALC889_FIXUP_VAIO_TT] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002182 .type = HDA_FIXUP_PINS,
2183 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01002184 { 0x17, 0x90170111 }, /* hidden surround speaker */
2185 { }
2186 }
2187 },
Takashi Iwai0e7cc2e2011-11-09 12:42:48 +01002188 [ALC888_FIXUP_EEE1601] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002189 .type = HDA_FIXUP_VERBS,
Takashi Iwai0e7cc2e2011-11-09 12:42:48 +01002190 .v.verbs = (const struct hda_verb[]) {
2191 { 0x20, AC_VERB_SET_COEF_INDEX, 0x0b },
2192 { 0x20, AC_VERB_SET_PROC_COEF, 0x0838 },
2193 { }
2194 }
Takashi Iwai177943a32011-11-09 12:55:18 +01002195 },
2196 [ALC882_FIXUP_EAPD] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002197 .type = HDA_FIXUP_VERBS,
Takashi Iwai177943a32011-11-09 12:55:18 +01002198 .v.verbs = (const struct hda_verb[]) {
2199 /* change to EAPD mode */
2200 { 0x20, AC_VERB_SET_COEF_INDEX, 0x07 },
2201 { 0x20, AC_VERB_SET_PROC_COEF, 0x3060 },
2202 { }
2203 }
2204 },
Takashi Iwai7a6069b2011-11-09 15:22:01 +01002205 [ALC883_FIXUP_EAPD] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002206 .type = HDA_FIXUP_VERBS,
Takashi Iwai7a6069b2011-11-09 15:22:01 +01002207 .v.verbs = (const struct hda_verb[]) {
2208 /* change to EAPD mode */
2209 { 0x20, AC_VERB_SET_COEF_INDEX, 0x07 },
2210 { 0x20, AC_VERB_SET_PROC_COEF, 0x3070 },
2211 { }
2212 }
2213 },
Takashi Iwai8812c4f2011-11-09 17:39:15 +01002214 [ALC883_FIXUP_ACER_EAPD] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002215 .type = HDA_FIXUP_VERBS,
Takashi Iwai8812c4f2011-11-09 17:39:15 +01002216 .v.verbs = (const struct hda_verb[]) {
2217 /* eanable EAPD on Acer laptops */
2218 { 0x20, AC_VERB_SET_COEF_INDEX, 0x07 },
2219 { 0x20, AC_VERB_SET_PROC_COEF, 0x3050 },
2220 { }
2221 }
2222 },
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002223 [ALC882_FIXUP_GPIO1] = {
Takashi Iwai5579cd62018-06-19 22:22:41 +02002224 .type = HDA_FIXUP_FUNC,
2225 .v.func = alc_fixup_gpio1,
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002226 },
2227 [ALC882_FIXUP_GPIO2] = {
Takashi Iwai5579cd62018-06-19 22:22:41 +02002228 .type = HDA_FIXUP_FUNC,
2229 .v.func = alc_fixup_gpio2,
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002230 },
Takashi Iwaieb844d52011-11-09 18:03:07 +01002231 [ALC882_FIXUP_GPIO3] = {
Takashi Iwai5579cd62018-06-19 22:22:41 +02002232 .type = HDA_FIXUP_FUNC,
2233 .v.func = alc_fixup_gpio3,
Takashi Iwaieb844d52011-11-09 18:03:07 +01002234 },
Takashi Iwai68ef0562011-11-09 18:24:44 +01002235 [ALC882_FIXUP_ASUS_W2JC] = {
Takashi Iwai5579cd62018-06-19 22:22:41 +02002236 .type = HDA_FIXUP_FUNC,
2237 .v.func = alc_fixup_gpio1,
Takashi Iwai68ef0562011-11-09 18:24:44 +01002238 .chained = true,
2239 .chain_id = ALC882_FIXUP_EAPD,
2240 },
2241 [ALC889_FIXUP_COEF] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002242 .type = HDA_FIXUP_FUNC,
Takashi Iwai68ef0562011-11-09 18:24:44 +01002243 .v.func = alc889_fixup_coef,
2244 },
Takashi Iwaic3e837b2011-11-10 16:01:47 +01002245 [ALC882_FIXUP_ACER_ASPIRE_4930G] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002246 .type = HDA_FIXUP_PINS,
2247 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaic3e837b2011-11-10 16:01:47 +01002248 { 0x16, 0x99130111 }, /* CLFE speaker */
2249 { 0x17, 0x99130112 }, /* surround speaker */
2250 { }
Takashi Iwai038d4fe2012-04-11 17:18:12 +02002251 },
2252 .chained = true,
2253 .chain_id = ALC882_FIXUP_GPIO1,
Takashi Iwaic3e837b2011-11-10 16:01:47 +01002254 },
2255 [ALC882_FIXUP_ACER_ASPIRE_8930G] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002256 .type = HDA_FIXUP_PINS,
2257 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaic3e837b2011-11-10 16:01:47 +01002258 { 0x16, 0x99130111 }, /* CLFE speaker */
2259 { 0x1b, 0x99130112 }, /* surround speaker */
2260 { }
2261 },
2262 .chained = true,
2263 .chain_id = ALC882_FIXUP_ASPIRE_8930G_VERBS,
2264 },
2265 [ALC882_FIXUP_ASPIRE_8930G_VERBS] = {
2266 /* additional init verbs for Acer Aspire 8930G */
Takashi Iwai1727a772013-01-10 09:52:52 +01002267 .type = HDA_FIXUP_VERBS,
Takashi Iwaic3e837b2011-11-10 16:01:47 +01002268 .v.verbs = (const struct hda_verb[]) {
2269 /* Enable all DACs */
2270 /* DAC DISABLE/MUTE 1? */
2271 /* setting bits 1-5 disables DAC nids 0x02-0x06
2272 * apparently. Init=0x38 */
2273 { 0x20, AC_VERB_SET_COEF_INDEX, 0x03 },
2274 { 0x20, AC_VERB_SET_PROC_COEF, 0x0000 },
2275 /* DAC DISABLE/MUTE 2? */
2276 /* some bit here disables the other DACs.
2277 * Init=0x4900 */
2278 { 0x20, AC_VERB_SET_COEF_INDEX, 0x08 },
2279 { 0x20, AC_VERB_SET_PROC_COEF, 0x0000 },
2280 /* DMIC fix
2281 * This laptop has a stereo digital microphone.
2282 * The mics are only 1cm apart which makes the stereo
2283 * useless. However, either the mic or the ALC889
2284 * makes the signal become a difference/sum signal
2285 * instead of standard stereo, which is annoying.
2286 * So instead we flip this bit which makes the
2287 * codec replicate the sum signal to both channels,
2288 * turning it into a normal mono mic.
2289 */
2290 /* DMIC_CONTROL? Init value = 0x0001 */
2291 { 0x20, AC_VERB_SET_COEF_INDEX, 0x0b },
2292 { 0x20, AC_VERB_SET_PROC_COEF, 0x0003 },
2293 { 0x20, AC_VERB_SET_COEF_INDEX, 0x07 },
2294 { 0x20, AC_VERB_SET_PROC_COEF, 0x3050 },
2295 { }
Takashi Iwai038d4fe2012-04-11 17:18:12 +02002296 },
2297 .chained = true,
2298 .chain_id = ALC882_FIXUP_GPIO1,
Takashi Iwaic3e837b2011-11-10 16:01:47 +01002299 },
Takashi Iwai56710872011-11-14 17:42:11 +01002300 [ALC885_FIXUP_MACPRO_GPIO] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002301 .type = HDA_FIXUP_FUNC,
Takashi Iwai56710872011-11-14 17:42:11 +01002302 .v.func = alc885_fixup_macpro_gpio,
2303 },
Takashi Iwai02a237b2012-02-13 15:25:07 +01002304 [ALC889_FIXUP_DAC_ROUTE] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002305 .type = HDA_FIXUP_FUNC,
Takashi Iwai02a237b2012-02-13 15:25:07 +01002306 .v.func = alc889_fixup_dac_route,
2307 },
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002308 [ALC889_FIXUP_MBP_VREF] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002309 .type = HDA_FIXUP_FUNC,
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002310 .v.func = alc889_fixup_mbp_vref,
2311 .chained = true,
2312 .chain_id = ALC882_FIXUP_GPIO1,
2313 },
2314 [ALC889_FIXUP_IMAC91_VREF] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002315 .type = HDA_FIXUP_FUNC,
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002316 .v.func = alc889_fixup_imac91_vref,
2317 .chained = true,
2318 .chain_id = ALC882_FIXUP_GPIO1,
2319 },
Adrien Vergée7729a42014-01-24 14:56:14 -05002320 [ALC889_FIXUP_MBA11_VREF] = {
2321 .type = HDA_FIXUP_FUNC,
2322 .v.func = alc889_fixup_mba11_vref,
2323 .chained = true,
2324 .chain_id = ALC889_FIXUP_MBP_VREF,
2325 },
Takashi Iwai0756f092013-12-04 13:59:45 +01002326 [ALC889_FIXUP_MBA21_VREF] = {
2327 .type = HDA_FIXUP_FUNC,
2328 .v.func = alc889_fixup_mba21_vref,
2329 .chained = true,
2330 .chain_id = ALC889_FIXUP_MBP_VREF,
2331 },
Takashi Iwaic20f31e2014-02-03 11:02:10 +01002332 [ALC889_FIXUP_MP11_VREF] = {
2333 .type = HDA_FIXUP_FUNC,
2334 .v.func = alc889_fixup_mba11_vref,
2335 .chained = true,
2336 .chain_id = ALC885_FIXUP_MACPRO_GPIO,
2337 },
Mario Kleiner9f660a12015-12-22 00:45:43 +01002338 [ALC889_FIXUP_MP41_VREF] = {
2339 .type = HDA_FIXUP_FUNC,
2340 .v.func = alc889_fixup_mbp_vref,
2341 .chained = true,
2342 .chain_id = ALC885_FIXUP_MACPRO_GPIO,
2343 },
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002344 [ALC882_FIXUP_INV_DMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002345 .type = HDA_FIXUP_FUNC,
David Henningsson9d36a7d2014-10-07 10:18:42 +02002346 .v.func = alc_fixup_inv_dmic,
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002347 },
Takashi Iwaie427c232012-07-29 10:04:08 +02002348 [ALC882_FIXUP_NO_PRIMARY_HP] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002349 .type = HDA_FIXUP_FUNC,
Takashi Iwaie427c232012-07-29 10:04:08 +02002350 .v.func = alc882_fixup_no_primary_hp,
2351 },
Takashi Iwai1f0bbf02013-11-28 15:21:21 +01002352 [ALC887_FIXUP_ASUS_BASS] = {
2353 .type = HDA_FIXUP_PINS,
2354 .v.pins = (const struct hda_pintbl[]) {
2355 {0x16, 0x99130130}, /* bass speaker */
2356 {}
2357 },
Takashi Iwaieb9ca3a2013-11-28 15:24:34 +01002358 .chained = true,
2359 .chain_id = ALC887_FIXUP_BASS_CHMAP,
2360 },
2361 [ALC887_FIXUP_BASS_CHMAP] = {
2362 .type = HDA_FIXUP_FUNC,
2363 .v.func = alc_fixup_bass_chmap,
Takashi Iwai1f0bbf02013-11-28 15:21:21 +01002364 },
Takashi Iwai7beb3a62017-04-10 18:05:52 +02002365 [ALC1220_FIXUP_GB_DUAL_CODECS] = {
2366 .type = HDA_FIXUP_FUNC,
2367 .v.func = alc1220_fixup_gb_dual_codecs,
2368 },
Peisen0202f5c2017-10-26 10:35:36 +08002369 [ALC1220_FIXUP_CLEVO_P950] = {
2370 .type = HDA_FIXUP_FUNC,
2371 .v.func = alc1220_fixup_clevo_p950,
2372 },
Richard Sailer80690a22019-04-02 15:52:04 +02002373 [ALC1220_FIXUP_CLEVO_PB51ED] = {
Jeremy Soller7f665b12019-02-13 10:56:19 -07002374 .type = HDA_FIXUP_FUNC,
Richard Sailer80690a22019-04-02 15:52:04 +02002375 .v.func = alc1220_fixup_clevo_pb51ed,
Jeremy Soller7f665b12019-02-13 10:56:19 -07002376 },
Richard Sailer80690a22019-04-02 15:52:04 +02002377 [ALC1220_FIXUP_CLEVO_PB51ED_PINS] = {
Jeremy Soller7f665b12019-02-13 10:56:19 -07002378 .type = HDA_FIXUP_PINS,
2379 .v.pins = (const struct hda_pintbl[]) {
2380 { 0x19, 0x01a1913c }, /* use as headset mic, without its own jack detect */
2381 {}
2382 },
2383 .chained = true,
Richard Sailer80690a22019-04-02 15:52:04 +02002384 .chain_id = ALC1220_FIXUP_CLEVO_PB51ED,
Jeremy Soller7f665b12019-02-13 10:56:19 -07002385 },
Takashi Iwai1d045db2011-07-07 18:23:21 +02002386};
2387
2388static const struct snd_pci_quirk alc882_fixup_tbl[] = {
Takashi Iwai8812c4f2011-11-09 17:39:15 +01002389 SND_PCI_QUIRK(0x1025, 0x006c, "Acer Aspire 9810", ALC883_FIXUP_ACER_EAPD),
2390 SND_PCI_QUIRK(0x1025, 0x0090, "Acer Aspire", ALC883_FIXUP_ACER_EAPD),
Takashi Iwaib5d724b2015-06-02 19:57:08 +02002391 SND_PCI_QUIRK(0x1025, 0x0107, "Acer Aspire", ALC883_FIXUP_ACER_EAPD),
Takashi Iwai8812c4f2011-11-09 17:39:15 +01002392 SND_PCI_QUIRK(0x1025, 0x010a, "Acer Ferrari 5000", ALC883_FIXUP_ACER_EAPD),
2393 SND_PCI_QUIRK(0x1025, 0x0110, "Acer Aspire", ALC883_FIXUP_ACER_EAPD),
2394 SND_PCI_QUIRK(0x1025, 0x0112, "Acer Aspire 9303", ALC883_FIXUP_ACER_EAPD),
2395 SND_PCI_QUIRK(0x1025, 0x0121, "Acer Aspire 5920G", ALC883_FIXUP_ACER_EAPD),
Takashi Iwaic3e837b2011-11-10 16:01:47 +01002396 SND_PCI_QUIRK(0x1025, 0x013e, "Acer Aspire 4930G",
2397 ALC882_FIXUP_ACER_ASPIRE_4930G),
2398 SND_PCI_QUIRK(0x1025, 0x013f, "Acer Aspire 5930G",
2399 ALC882_FIXUP_ACER_ASPIRE_4930G),
2400 SND_PCI_QUIRK(0x1025, 0x0145, "Acer Aspire 8930G",
2401 ALC882_FIXUP_ACER_ASPIRE_8930G),
2402 SND_PCI_QUIRK(0x1025, 0x0146, "Acer Aspire 6935G",
2403 ALC882_FIXUP_ACER_ASPIRE_8930G),
2404 SND_PCI_QUIRK(0x1025, 0x015e, "Acer Aspire 6930G",
2405 ALC882_FIXUP_ACER_ASPIRE_4930G),
2406 SND_PCI_QUIRK(0x1025, 0x0166, "Acer Aspire 6530G",
2407 ALC882_FIXUP_ACER_ASPIRE_4930G),
2408 SND_PCI_QUIRK(0x1025, 0x0142, "Acer Aspire 7730G",
2409 ALC882_FIXUP_ACER_ASPIRE_4930G),
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01002410 SND_PCI_QUIRK(0x1025, 0x0155, "Packard-Bell M5120", ALC882_FIXUP_PB_M5210),
Takashi Iwaif5c53d82012-05-07 10:07:33 +02002411 SND_PCI_QUIRK(0x1025, 0x021e, "Acer Aspire 5739G",
2412 ALC882_FIXUP_ACER_ASPIRE_4930G),
Takashi Iwai02a237b2012-02-13 15:25:07 +01002413 SND_PCI_QUIRK(0x1025, 0x0259, "Acer Aspire 5935", ALC889_FIXUP_DAC_ROUTE),
Takashi Iwaife97da12012-04-12 08:00:19 +02002414 SND_PCI_QUIRK(0x1025, 0x026b, "Acer Aspire 8940G", ALC882_FIXUP_ACER_ASPIRE_8930G),
Takashi Iwaiac9b1cd2011-11-09 17:45:55 +01002415 SND_PCI_QUIRK(0x1025, 0x0296, "Acer Aspire 7736z", ALC882_FIXUP_ACER_ASPIRE_7736),
Takashi Iwai177943a32011-11-09 12:55:18 +01002416 SND_PCI_QUIRK(0x1043, 0x13c2, "Asus A7M", ALC882_FIXUP_EAPD),
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01002417 SND_PCI_QUIRK(0x1043, 0x1873, "ASUS W90V", ALC882_FIXUP_ASUS_W90V),
Takashi Iwai68ef0562011-11-09 18:24:44 +01002418 SND_PCI_QUIRK(0x1043, 0x1971, "Asus W2JC", ALC882_FIXUP_ASUS_W2JC),
Takashi Iwai0e7cc2e2011-11-09 12:42:48 +01002419 SND_PCI_QUIRK(0x1043, 0x835f, "Asus Eee 1601", ALC888_FIXUP_EEE1601),
Takashi Iwai1f0bbf02013-11-28 15:21:21 +01002420 SND_PCI_QUIRK(0x1043, 0x84bc, "ASUS ET2700", ALC887_FIXUP_ASUS_BASS),
Takashi Iwai85bcf962016-12-06 16:20:36 +01002421 SND_PCI_QUIRK(0x1043, 0x8691, "ASUS ROG Ranger VIII", ALC882_FIXUP_GPIO3),
Takashi Iwaiac9b1cd2011-11-09 17:45:55 +01002422 SND_PCI_QUIRK(0x104d, 0x9047, "Sony Vaio TT", ALC889_FIXUP_VAIO_TT),
Takashi Iwaie427c232012-07-29 10:04:08 +02002423 SND_PCI_QUIRK(0x104d, 0x905a, "Sony Vaio Z", ALC882_FIXUP_NO_PRIMARY_HP),
Sergei A. Trusov3f3c3712017-08-02 20:23:48 +10002424 SND_PCI_QUIRK(0x104d, 0x9060, "Sony Vaio VPCL14M1R", ALC882_FIXUP_NO_PRIMARY_HP),
Fernando Luis Vázquez Cao12e31a72013-02-12 16:47:44 +09002425 SND_PCI_QUIRK(0x104d, 0x9043, "Sony Vaio VGC-LN51JGB", ALC882_FIXUP_NO_PRIMARY_HP),
Takashi Iwaic44d9b12016-02-07 09:38:26 +01002426 SND_PCI_QUIRK(0x104d, 0x9044, "Sony VAIO AiO", ALC882_FIXUP_NO_PRIMARY_HP),
Takashi Iwai56710872011-11-14 17:42:11 +01002427
2428 /* All Apple entries are in codec SSIDs */
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002429 SND_PCI_QUIRK(0x106b, 0x00a0, "MacBookPro 3,1", ALC889_FIXUP_MBP_VREF),
2430 SND_PCI_QUIRK(0x106b, 0x00a1, "Macbook", ALC889_FIXUP_MBP_VREF),
2431 SND_PCI_QUIRK(0x106b, 0x00a4, "MacbookPro 4,1", ALC889_FIXUP_MBP_VREF),
Takashi Iwaic20f31e2014-02-03 11:02:10 +01002432 SND_PCI_QUIRK(0x106b, 0x0c00, "Mac Pro", ALC889_FIXUP_MP11_VREF),
Takashi Iwai56710872011-11-14 17:42:11 +01002433 SND_PCI_QUIRK(0x106b, 0x1000, "iMac 24", ALC885_FIXUP_MACPRO_GPIO),
2434 SND_PCI_QUIRK(0x106b, 0x2800, "AppleTV", ALC885_FIXUP_MACPRO_GPIO),
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002435 SND_PCI_QUIRK(0x106b, 0x2c00, "MacbookPro rev3", ALC889_FIXUP_MBP_VREF),
2436 SND_PCI_QUIRK(0x106b, 0x3000, "iMac", ALC889_FIXUP_MBP_VREF),
Takashi Iwai56710872011-11-14 17:42:11 +01002437 SND_PCI_QUIRK(0x106b, 0x3200, "iMac 7,1 Aluminum", ALC882_FIXUP_EAPD),
Adrien Vergée7729a42014-01-24 14:56:14 -05002438 SND_PCI_QUIRK(0x106b, 0x3400, "MacBookAir 1,1", ALC889_FIXUP_MBA11_VREF),
Takashi Iwai0756f092013-12-04 13:59:45 +01002439 SND_PCI_QUIRK(0x106b, 0x3500, "MacBookAir 2,1", ALC889_FIXUP_MBA21_VREF),
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002440 SND_PCI_QUIRK(0x106b, 0x3600, "Macbook 3,1", ALC889_FIXUP_MBP_VREF),
2441 SND_PCI_QUIRK(0x106b, 0x3800, "MacbookPro 4,1", ALC889_FIXUP_MBP_VREF),
Takashi Iwai56710872011-11-14 17:42:11 +01002442 SND_PCI_QUIRK(0x106b, 0x3e00, "iMac 24 Aluminum", ALC885_FIXUP_MACPRO_GPIO),
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002443 SND_PCI_QUIRK(0x106b, 0x3f00, "Macbook 5,1", ALC889_FIXUP_IMAC91_VREF),
2444 SND_PCI_QUIRK(0x106b, 0x4000, "MacbookPro 5,1", ALC889_FIXUP_IMAC91_VREF),
2445 SND_PCI_QUIRK(0x106b, 0x4100, "Macmini 3,1", ALC889_FIXUP_IMAC91_VREF),
Mario Kleiner9f660a12015-12-22 00:45:43 +01002446 SND_PCI_QUIRK(0x106b, 0x4200, "Mac Pro 4,1/5,1", ALC889_FIXUP_MP41_VREF),
Takashi Iwai05193632012-11-12 10:07:36 +01002447 SND_PCI_QUIRK(0x106b, 0x4300, "iMac 9,1", ALC889_FIXUP_IMAC91_VREF),
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002448 SND_PCI_QUIRK(0x106b, 0x4600, "MacbookPro 5,2", ALC889_FIXUP_IMAC91_VREF),
2449 SND_PCI_QUIRK(0x106b, 0x4900, "iMac 9,1 Aluminum", ALC889_FIXUP_IMAC91_VREF),
Takashi Iwai649ccd02015-07-30 22:30:29 +02002450 SND_PCI_QUIRK(0x106b, 0x4a00, "Macbook 5,2", ALC889_FIXUP_MBA11_VREF),
Takashi Iwai56710872011-11-14 17:42:11 +01002451
Takashi Iwai7a6069b2011-11-09 15:22:01 +01002452 SND_PCI_QUIRK(0x1071, 0x8258, "Evesham Voyaeger", ALC882_FIXUP_EAPD),
David Henningssonb2c53e22014-01-01 14:01:34 +01002453 SND_PCI_QUIRK(0x1458, 0xa002, "Gigabyte EP45-DS3/Z87X-UD3H", ALC889_FIXUP_FRONT_HP_NO_PRESENCE),
Takashi Iwai7beb3a62017-04-10 18:05:52 +02002454 SND_PCI_QUIRK(0x1458, 0xa0b8, "Gigabyte AZ370-Gaming", ALC1220_FIXUP_GB_DUAL_CODECS),
Takashi Iwaid2c3b142017-06-01 09:35:30 +02002455 SND_PCI_QUIRK(0x1462, 0x7350, "MSI-7350", ALC889_FIXUP_CD),
Takashi Iwai63691582017-05-22 16:32:46 +02002456 SND_PCI_QUIRK(0x1462, 0xda57, "MSI Z270-Gaming", ALC1220_FIXUP_GB_DUAL_CODECS),
Takashi Iwaid2c3b142017-06-01 09:35:30 +02002457 SND_PCI_QUIRK_VENDOR(0x1462, "MSI", ALC882_FIXUP_GPIO3),
Takashi Iwai5c0ebfb2011-11-07 17:59:13 +01002458 SND_PCI_QUIRK(0x147b, 0x107a, "Abit AW9D-MAX", ALC882_FIXUP_ABIT_AW9D_MAX),
Peisen0202f5c2017-10-26 10:35:36 +08002459 SND_PCI_QUIRK(0x1558, 0x9501, "Clevo P950HR", ALC1220_FIXUP_CLEVO_P950),
Takashi Iwaif3d737b2018-07-17 17:08:32 +02002460 SND_PCI_QUIRK(0x1558, 0x95e1, "Clevo P95xER", ALC1220_FIXUP_CLEVO_P950),
Jeremy Soller2f0d5202018-05-07 09:28:45 -06002461 SND_PCI_QUIRK(0x1558, 0x95e2, "Clevo P950ER", ALC1220_FIXUP_CLEVO_P950),
Richard Sailer80690a22019-04-02 15:52:04 +02002462 SND_PCI_QUIRK(0x1558, 0x96e1, "System76 Oryx Pro (oryp5)", ALC1220_FIXUP_CLEVO_PB51ED_PINS),
2463 SND_PCI_QUIRK(0x1558, 0x97e1, "System76 Oryx Pro (oryp5)", ALC1220_FIXUP_CLEVO_PB51ED_PINS),
2464 SND_PCI_QUIRK(0x1558, 0x65d1, "Tuxedo Book XC1509", ALC1220_FIXUP_CLEVO_PB51ED_PINS),
Takashi Iwai7a6069b2011-11-09 15:22:01 +01002465 SND_PCI_QUIRK_VENDOR(0x1558, "Clevo laptop", ALC882_FIXUP_EAPD),
2466 SND_PCI_QUIRK(0x161f, 0x2054, "Medion laptop", ALC883_FIXUP_EAPD),
Takashi Iwaiac9b1cd2011-11-09 17:45:55 +01002467 SND_PCI_QUIRK(0x17aa, 0x3a0d, "Lenovo Y530", ALC882_FIXUP_LENOVO_Y530),
Takashi Iwai68ef0562011-11-09 18:24:44 +01002468 SND_PCI_QUIRK(0x8086, 0x0022, "DX58SO", ALC889_FIXUP_COEF),
Takashi Iwai1d045db2011-07-07 18:23:21 +02002469 {}
2470};
2471
Takashi Iwai1727a772013-01-10 09:52:52 +01002472static const struct hda_model_fixup alc882_fixup_models[] = {
Takashi Iwai772c2912018-06-26 17:17:53 +02002473 {.id = ALC882_FIXUP_ABIT_AW9D_MAX, .name = "abit-aw9d"},
2474 {.id = ALC882_FIXUP_LENOVO_Y530, .name = "lenovo-y530"},
2475 {.id = ALC882_FIXUP_ACER_ASPIRE_7736, .name = "acer-aspire-7736"},
2476 {.id = ALC882_FIXUP_ASUS_W90V, .name = "asus-w90v"},
2477 {.id = ALC889_FIXUP_CD, .name = "cd"},
2478 {.id = ALC889_FIXUP_FRONT_HP_NO_PRESENCE, .name = "no-front-hp"},
2479 {.id = ALC889_FIXUP_VAIO_TT, .name = "vaio-tt"},
2480 {.id = ALC888_FIXUP_EEE1601, .name = "eee1601"},
2481 {.id = ALC882_FIXUP_EAPD, .name = "alc882-eapd"},
2482 {.id = ALC883_FIXUP_EAPD, .name = "alc883-eapd"},
2483 {.id = ALC882_FIXUP_GPIO1, .name = "gpio1"},
2484 {.id = ALC882_FIXUP_GPIO2, .name = "gpio2"},
2485 {.id = ALC882_FIXUP_GPIO3, .name = "gpio3"},
2486 {.id = ALC889_FIXUP_COEF, .name = "alc889-coef"},
2487 {.id = ALC882_FIXUP_ASUS_W2JC, .name = "asus-w2jc"},
Takashi Iwai912093b2012-04-11 14:03:41 +02002488 {.id = ALC882_FIXUP_ACER_ASPIRE_4930G, .name = "acer-aspire-4930g"},
2489 {.id = ALC882_FIXUP_ACER_ASPIRE_8930G, .name = "acer-aspire-8930g"},
2490 {.id = ALC883_FIXUP_ACER_EAPD, .name = "acer-aspire"},
Takashi Iwai772c2912018-06-26 17:17:53 +02002491 {.id = ALC885_FIXUP_MACPRO_GPIO, .name = "macpro-gpio"},
2492 {.id = ALC889_FIXUP_DAC_ROUTE, .name = "dac-route"},
2493 {.id = ALC889_FIXUP_MBP_VREF, .name = "mbp-vref"},
2494 {.id = ALC889_FIXUP_IMAC91_VREF, .name = "imac91-vref"},
2495 {.id = ALC889_FIXUP_MBA11_VREF, .name = "mba11-vref"},
2496 {.id = ALC889_FIXUP_MBA21_VREF, .name = "mba21-vref"},
2497 {.id = ALC889_FIXUP_MP11_VREF, .name = "mp11-vref"},
2498 {.id = ALC889_FIXUP_MP41_VREF, .name = "mp41-vref"},
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002499 {.id = ALC882_FIXUP_INV_DMIC, .name = "inv-dmic"},
Takashi Iwaie427c232012-07-29 10:04:08 +02002500 {.id = ALC882_FIXUP_NO_PRIMARY_HP, .name = "no-primary-hp"},
Takashi Iwai772c2912018-06-26 17:17:53 +02002501 {.id = ALC887_FIXUP_ASUS_BASS, .name = "asus-bass"},
Takashi Iwaiba90d6a2017-05-22 16:38:47 +02002502 {.id = ALC1220_FIXUP_GB_DUAL_CODECS, .name = "dual-codecs"},
Takashi Iwai772c2912018-06-26 17:17:53 +02002503 {.id = ALC1220_FIXUP_CLEVO_P950, .name = "clevo-p950"},
Takashi Iwai912093b2012-04-11 14:03:41 +02002504 {}
2505};
2506
Takashi Iwai1d045db2011-07-07 18:23:21 +02002507/*
2508 * BIOS auto configuration
2509 */
2510/* almost identical with ALC880 parser... */
2511static int alc882_parse_auto_config(struct hda_codec *codec)
2512{
Takashi Iwai1d045db2011-07-07 18:23:21 +02002513 static const hda_nid_t alc882_ignore[] = { 0x1d, 0 };
Takashi Iwai3e6179b2011-07-08 16:55:13 +02002514 static const hda_nid_t alc882_ssids[] = { 0x15, 0x1b, 0x14, 0 };
2515 return alc_parse_auto_config(codec, alc882_ignore, alc882_ssids);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002516}
2517
Takashi Iwai1d045db2011-07-07 18:23:21 +02002518/*
2519 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02002520static int patch_alc882(struct hda_codec *codec)
2521{
2522 struct alc_spec *spec;
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002523 int err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002524
Takashi Iwai3de95172012-05-07 18:03:15 +02002525 err = alc_alloc_spec(codec, 0x0b);
2526 if (err < 0)
2527 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002528
Takashi Iwai3de95172012-05-07 18:03:15 +02002529 spec = codec->spec;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002530
Takashi Iwai7639a062015-03-03 10:07:24 +01002531 switch (codec->core.vendor_id) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02002532 case 0x10ec0882:
2533 case 0x10ec0885:
Takashi Iwaiacf08082014-09-02 07:21:56 +02002534 case 0x10ec0900:
Kailang Yanga535ad52017-01-16 16:59:26 +08002535 case 0x10ec1220:
Takashi Iwai1d045db2011-07-07 18:23:21 +02002536 break;
2537 default:
2538 /* ALC883 and variants */
2539 alc_fix_pll_init(codec, 0x20, 0x0a, 10);
2540 break;
2541 }
2542
Takashi Iwaic9af7532019-05-10 11:01:43 +02002543 alc_pre_init(codec);
2544
Takashi Iwai1727a772013-01-10 09:52:52 +01002545 snd_hda_pick_fixup(codec, alc882_fixup_models, alc882_fixup_tbl,
Takashi Iwai912093b2012-04-11 14:03:41 +02002546 alc882_fixups);
Takashi Iwai1727a772013-01-10 09:52:52 +01002547 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002548
2549 alc_auto_parse_customize_define(codec);
2550
Takashi Iwai7504b6c2013-03-18 11:25:51 +01002551 if (has_cdefine_beep(codec))
2552 spec->gen.beep_nid = 0x01;
2553
Takashi Iwai1a97b7f2012-02-21 11:11:48 +01002554 /* automatic parse from the BIOS config */
2555 err = alc882_parse_auto_config(codec);
2556 if (err < 0)
2557 goto error;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002558
Takashi Iwaifea80fa2018-06-20 12:52:46 +02002559 if (!spec->gen.no_analog && spec->gen.beep_nid) {
2560 err = set_beep_amp(spec, 0x0b, 0x05, HDA_INPUT);
2561 if (err < 0)
2562 goto error;
2563 }
Takashi Iwai1d045db2011-07-07 18:23:21 +02002564
Takashi Iwai1727a772013-01-10 09:52:52 +01002565 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
Takashi Iwai589876e2012-02-20 15:47:55 +01002566
Takashi Iwai1d045db2011-07-07 18:23:21 +02002567 return 0;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02002568
2569 error:
2570 alc_free(codec);
2571 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002572}
2573
2574
2575/*
2576 * ALC262 support
2577 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02002578static int alc262_parse_auto_config(struct hda_codec *codec)
2579{
Takashi Iwai1d045db2011-07-07 18:23:21 +02002580 static const hda_nid_t alc262_ignore[] = { 0x1d, 0 };
Takashi Iwai3e6179b2011-07-08 16:55:13 +02002581 static const hda_nid_t alc262_ssids[] = { 0x15, 0x1b, 0x14, 0 };
2582 return alc_parse_auto_config(codec, alc262_ignore, alc262_ssids);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002583}
2584
2585/*
2586 * Pin config fixes
2587 */
2588enum {
Takashi Iwaiea4e7af2011-11-07 12:23:55 +01002589 ALC262_FIXUP_FSC_H270,
Takashi Iwai7513e6d2013-01-18 15:41:34 +01002590 ALC262_FIXUP_FSC_S7110,
Takashi Iwaiea4e7af2011-11-07 12:23:55 +01002591 ALC262_FIXUP_HP_Z200,
2592 ALC262_FIXUP_TYAN,
Takashi Iwaic4701502011-11-07 14:20:07 +01002593 ALC262_FIXUP_LENOVO_3000,
Takashi Iwaib42590b2011-11-07 14:41:01 +01002594 ALC262_FIXUP_BENQ,
2595 ALC262_FIXUP_BENQ_T31,
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002596 ALC262_FIXUP_INV_DMIC,
Mengdong Linb5c66112013-11-29 00:35:35 -05002597 ALC262_FIXUP_INTEL_BAYLEYBAY,
Takashi Iwai1d045db2011-07-07 18:23:21 +02002598};
2599
Takashi Iwai1727a772013-01-10 09:52:52 +01002600static const struct hda_fixup alc262_fixups[] = {
Takashi Iwaiea4e7af2011-11-07 12:23:55 +01002601 [ALC262_FIXUP_FSC_H270] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002602 .type = HDA_FIXUP_PINS,
2603 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02002604 { 0x14, 0x99130110 }, /* speaker */
2605 { 0x15, 0x0221142f }, /* front HP */
2606 { 0x1b, 0x0121141f }, /* rear HP */
2607 { }
2608 }
2609 },
Takashi Iwai7513e6d2013-01-18 15:41:34 +01002610 [ALC262_FIXUP_FSC_S7110] = {
2611 .type = HDA_FIXUP_PINS,
2612 .v.pins = (const struct hda_pintbl[]) {
2613 { 0x15, 0x90170110 }, /* speaker */
2614 { }
2615 },
2616 .chained = true,
2617 .chain_id = ALC262_FIXUP_BENQ,
2618 },
Takashi Iwaiea4e7af2011-11-07 12:23:55 +01002619 [ALC262_FIXUP_HP_Z200] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002620 .type = HDA_FIXUP_PINS,
2621 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02002622 { 0x16, 0x99130120 }, /* internal speaker */
2623 { }
2624 }
2625 },
Takashi Iwaiea4e7af2011-11-07 12:23:55 +01002626 [ALC262_FIXUP_TYAN] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002627 .type = HDA_FIXUP_PINS,
2628 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaiea4e7af2011-11-07 12:23:55 +01002629 { 0x14, 0x1993e1f0 }, /* int AUX */
2630 { }
2631 }
2632 },
Takashi Iwaic4701502011-11-07 14:20:07 +01002633 [ALC262_FIXUP_LENOVO_3000] = {
Takashi Iwaifd108212013-01-10 10:18:14 +01002634 .type = HDA_FIXUP_PINCTLS,
2635 .v.pins = (const struct hda_pintbl[]) {
2636 { 0x19, PIN_VREF50 },
Takashi Iwaib42590b2011-11-07 14:41:01 +01002637 {}
2638 },
2639 .chained = true,
2640 .chain_id = ALC262_FIXUP_BENQ,
2641 },
2642 [ALC262_FIXUP_BENQ] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002643 .type = HDA_FIXUP_VERBS,
Takashi Iwaib42590b2011-11-07 14:41:01 +01002644 .v.verbs = (const struct hda_verb[]) {
Takashi Iwaic4701502011-11-07 14:20:07 +01002645 { 0x20, AC_VERB_SET_COEF_INDEX, 0x07 },
2646 { 0x20, AC_VERB_SET_PROC_COEF, 0x3070 },
2647 {}
2648 }
2649 },
Takashi Iwaib42590b2011-11-07 14:41:01 +01002650 [ALC262_FIXUP_BENQ_T31] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002651 .type = HDA_FIXUP_VERBS,
Takashi Iwaib42590b2011-11-07 14:41:01 +01002652 .v.verbs = (const struct hda_verb[]) {
2653 { 0x20, AC_VERB_SET_COEF_INDEX, 0x07 },
2654 { 0x20, AC_VERB_SET_PROC_COEF, 0x3050 },
2655 {}
2656 }
2657 },
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002658 [ALC262_FIXUP_INV_DMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002659 .type = HDA_FIXUP_FUNC,
David Henningsson9d36a7d2014-10-07 10:18:42 +02002660 .v.func = alc_fixup_inv_dmic,
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002661 },
Mengdong Linb5c66112013-11-29 00:35:35 -05002662 [ALC262_FIXUP_INTEL_BAYLEYBAY] = {
2663 .type = HDA_FIXUP_FUNC,
2664 .v.func = alc_fixup_no_depop_delay,
2665 },
Takashi Iwai1d045db2011-07-07 18:23:21 +02002666};
2667
2668static const struct snd_pci_quirk alc262_fixup_tbl[] = {
Takashi Iwaiea4e7af2011-11-07 12:23:55 +01002669 SND_PCI_QUIRK(0x103c, 0x170b, "HP Z200", ALC262_FIXUP_HP_Z200),
Takashi Iwai7513e6d2013-01-18 15:41:34 +01002670 SND_PCI_QUIRK(0x10cf, 0x1397, "Fujitsu Lifebook S7110", ALC262_FIXUP_FSC_S7110),
Takashi Iwai3dcd3be2011-11-07 14:59:40 +01002671 SND_PCI_QUIRK(0x10cf, 0x142d, "Fujitsu Lifebook E8410", ALC262_FIXUP_BENQ),
Takashi Iwaiea4e7af2011-11-07 12:23:55 +01002672 SND_PCI_QUIRK(0x10f1, 0x2915, "Tyan Thunder n6650W", ALC262_FIXUP_TYAN),
Takashi Iwai275ec0c2018-06-22 12:17:45 +02002673 SND_PCI_QUIRK(0x1734, 0x1141, "FSC ESPRIMO U9210", ALC262_FIXUP_FSC_H270),
Takashi Iwaiea4e7af2011-11-07 12:23:55 +01002674 SND_PCI_QUIRK(0x1734, 0x1147, "FSC Celsius H270", ALC262_FIXUP_FSC_H270),
Takashi Iwaic4701502011-11-07 14:20:07 +01002675 SND_PCI_QUIRK(0x17aa, 0x384e, "Lenovo 3000", ALC262_FIXUP_LENOVO_3000),
Takashi Iwaib42590b2011-11-07 14:41:01 +01002676 SND_PCI_QUIRK(0x17ff, 0x0560, "Benq ED8", ALC262_FIXUP_BENQ),
2677 SND_PCI_QUIRK(0x17ff, 0x058d, "Benq T31-16", ALC262_FIXUP_BENQ_T31),
Mengdong Linb5c66112013-11-29 00:35:35 -05002678 SND_PCI_QUIRK(0x8086, 0x7270, "BayleyBay", ALC262_FIXUP_INTEL_BAYLEYBAY),
Takashi Iwai1d045db2011-07-07 18:23:21 +02002679 {}
2680};
2681
Takashi Iwai1727a772013-01-10 09:52:52 +01002682static const struct hda_model_fixup alc262_fixup_models[] = {
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002683 {.id = ALC262_FIXUP_INV_DMIC, .name = "inv-dmic"},
Takashi Iwaie43c44d2018-06-26 17:02:08 +02002684 {.id = ALC262_FIXUP_FSC_H270, .name = "fsc-h270"},
2685 {.id = ALC262_FIXUP_FSC_S7110, .name = "fsc-s7110"},
2686 {.id = ALC262_FIXUP_HP_Z200, .name = "hp-z200"},
2687 {.id = ALC262_FIXUP_TYAN, .name = "tyan"},
2688 {.id = ALC262_FIXUP_LENOVO_3000, .name = "lenovo-3000"},
2689 {.id = ALC262_FIXUP_BENQ, .name = "benq"},
2690 {.id = ALC262_FIXUP_BENQ_T31, .name = "benq-t31"},
2691 {.id = ALC262_FIXUP_INTEL_BAYLEYBAY, .name = "bayleybay"},
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002692 {}
2693};
Takashi Iwai1d045db2011-07-07 18:23:21 +02002694
Takashi Iwai1d045db2011-07-07 18:23:21 +02002695/*
2696 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02002697static int patch_alc262(struct hda_codec *codec)
2698{
2699 struct alc_spec *spec;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002700 int err;
2701
Takashi Iwai3de95172012-05-07 18:03:15 +02002702 err = alc_alloc_spec(codec, 0x0b);
2703 if (err < 0)
2704 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002705
Takashi Iwai3de95172012-05-07 18:03:15 +02002706 spec = codec->spec;
Takashi Iwai08c189f2012-12-19 15:22:24 +01002707 spec->gen.shared_mic_vref_pin = 0x18;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002708
Takashi Iwai225068a2015-05-29 10:42:14 +02002709 spec->shutup = alc_eapd_shutup;
2710
Takashi Iwai1d045db2011-07-07 18:23:21 +02002711#if 0
2712 /* pshou 07/11/05 set a zero PCM sample to DAC when FIFO is
2713 * under-run
2714 */
Takashi Iwai98b24882014-08-18 13:47:50 +02002715 alc_update_coefex_idx(codec, 0x1a, 7, 0, 0x80);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002716#endif
Takashi Iwai1d045db2011-07-07 18:23:21 +02002717 alc_fix_pll_init(codec, 0x20, 0x0a, 10);
2718
Takashi Iwaic9af7532019-05-10 11:01:43 +02002719 alc_pre_init(codec);
2720
Takashi Iwai1727a772013-01-10 09:52:52 +01002721 snd_hda_pick_fixup(codec, alc262_fixup_models, alc262_fixup_tbl,
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002722 alc262_fixups);
Takashi Iwai1727a772013-01-10 09:52:52 +01002723 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002724
Takashi Iwaiaf741c12012-05-07 18:09:48 +02002725 alc_auto_parse_customize_define(codec);
2726
Takashi Iwai7504b6c2013-03-18 11:25:51 +01002727 if (has_cdefine_beep(codec))
2728 spec->gen.beep_nid = 0x01;
2729
Takashi Iwai42399f72011-11-07 17:18:44 +01002730 /* automatic parse from the BIOS config */
2731 err = alc262_parse_auto_config(codec);
2732 if (err < 0)
2733 goto error;
Takashi Iwai3e6179b2011-07-08 16:55:13 +02002734
Takashi Iwaifea80fa2018-06-20 12:52:46 +02002735 if (!spec->gen.no_analog && spec->gen.beep_nid) {
2736 err = set_beep_amp(spec, 0x0b, 0x05, HDA_INPUT);
2737 if (err < 0)
2738 goto error;
2739 }
Takashi Iwai1d045db2011-07-07 18:23:21 +02002740
Takashi Iwai1727a772013-01-10 09:52:52 +01002741 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
Takashi Iwai589876e2012-02-20 15:47:55 +01002742
Takashi Iwai1d045db2011-07-07 18:23:21 +02002743 return 0;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02002744
2745 error:
2746 alc_free(codec);
2747 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002748}
2749
2750/*
2751 * ALC268
2752 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02002753/* bind Beep switches of both NID 0x0f and 0x10 */
Takashi Iwaia7177772017-05-10 12:04:08 +02002754static int alc268_beep_switch_put(struct snd_kcontrol *kcontrol,
2755 struct snd_ctl_elem_value *ucontrol)
2756{
2757 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2758 unsigned long pval;
2759 int err;
2760
2761 mutex_lock(&codec->control_mutex);
2762 pval = kcontrol->private_value;
2763 kcontrol->private_value = (pval & ~0xff) | 0x0f;
2764 err = snd_hda_mixer_amp_switch_put(kcontrol, ucontrol);
2765 if (err >= 0) {
2766 kcontrol->private_value = (pval & ~0xff) | 0x10;
2767 err = snd_hda_mixer_amp_switch_put(kcontrol, ucontrol);
2768 }
2769 kcontrol->private_value = pval;
2770 mutex_unlock(&codec->control_mutex);
2771 return err;
2772}
Takashi Iwai1d045db2011-07-07 18:23:21 +02002773
2774static const struct snd_kcontrol_new alc268_beep_mixer[] = {
2775 HDA_CODEC_VOLUME("Beep Playback Volume", 0x1d, 0x0, HDA_INPUT),
Takashi Iwaia7177772017-05-10 12:04:08 +02002776 {
2777 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2778 .name = "Beep Playback Switch",
2779 .subdevice = HDA_SUBDEV_AMP_FLAG,
2780 .info = snd_hda_mixer_amp_switch_info,
2781 .get = snd_hda_mixer_amp_switch_get,
2782 .put = alc268_beep_switch_put,
2783 .private_value = HDA_COMPOSE_AMP_VAL(0x0f, 3, 1, HDA_INPUT)
2784 },
Takashi Iwai1d045db2011-07-07 18:23:21 +02002785};
2786
2787/* set PCBEEP vol = 0, mute connections */
2788static const struct hda_verb alc268_beep_init_verbs[] = {
2789 {0x1d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
2790 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
2791 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
2792 { }
2793};
2794
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002795enum {
2796 ALC268_FIXUP_INV_DMIC,
Takashi Iwaicb766402012-10-20 10:55:21 +02002797 ALC268_FIXUP_HP_EAPD,
Takashi Iwai24eff322013-11-04 18:21:08 +01002798 ALC268_FIXUP_SPDIF,
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002799};
2800
Takashi Iwai1727a772013-01-10 09:52:52 +01002801static const struct hda_fixup alc268_fixups[] = {
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002802 [ALC268_FIXUP_INV_DMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002803 .type = HDA_FIXUP_FUNC,
David Henningsson9d36a7d2014-10-07 10:18:42 +02002804 .v.func = alc_fixup_inv_dmic,
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002805 },
Takashi Iwaicb766402012-10-20 10:55:21 +02002806 [ALC268_FIXUP_HP_EAPD] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01002807 .type = HDA_FIXUP_VERBS,
Takashi Iwaicb766402012-10-20 10:55:21 +02002808 .v.verbs = (const struct hda_verb[]) {
2809 {0x15, AC_VERB_SET_EAPD_BTLENABLE, 0},
2810 {}
2811 }
2812 },
Takashi Iwai24eff322013-11-04 18:21:08 +01002813 [ALC268_FIXUP_SPDIF] = {
2814 .type = HDA_FIXUP_PINS,
2815 .v.pins = (const struct hda_pintbl[]) {
2816 { 0x1e, 0x014b1180 }, /* enable SPDIF out */
2817 {}
2818 }
2819 },
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002820};
2821
Takashi Iwai1727a772013-01-10 09:52:52 +01002822static const struct hda_model_fixup alc268_fixup_models[] = {
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002823 {.id = ALC268_FIXUP_INV_DMIC, .name = "inv-dmic"},
Takashi Iwaicb766402012-10-20 10:55:21 +02002824 {.id = ALC268_FIXUP_HP_EAPD, .name = "hp-eapd"},
Takashi Iwai03bf11c2018-06-26 16:56:41 +02002825 {.id = ALC268_FIXUP_SPDIF, .name = "spdif"},
Takashi Iwaicb766402012-10-20 10:55:21 +02002826 {}
2827};
2828
2829static const struct snd_pci_quirk alc268_fixup_tbl[] = {
Takashi Iwai24eff322013-11-04 18:21:08 +01002830 SND_PCI_QUIRK(0x1025, 0x0139, "Acer TravelMate 6293", ALC268_FIXUP_SPDIF),
David Henningssonfcd8f3b2013-01-28 05:45:47 +01002831 SND_PCI_QUIRK(0x1025, 0x015b, "Acer AOA 150 (ZG5)", ALC268_FIXUP_INV_DMIC),
Takashi Iwaicb766402012-10-20 10:55:21 +02002832 /* below is codec SSID since multiple Toshiba laptops have the
2833 * same PCI SSID 1179:ff00
2834 */
2835 SND_PCI_QUIRK(0x1179, 0xff06, "Toshiba P200", ALC268_FIXUP_HP_EAPD),
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002836 {}
2837};
2838
Takashi Iwai1d045db2011-07-07 18:23:21 +02002839/*
2840 * BIOS auto configuration
2841 */
2842static int alc268_parse_auto_config(struct hda_codec *codec)
2843{
Takashi Iwai3e6179b2011-07-08 16:55:13 +02002844 static const hda_nid_t alc268_ssids[] = { 0x15, 0x1b, 0x14, 0 };
Takashi Iwai7504b6c2013-03-18 11:25:51 +01002845 return alc_parse_auto_config(codec, NULL, alc268_ssids);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002846}
2847
Takashi Iwai1d045db2011-07-07 18:23:21 +02002848/*
2849 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02002850static int patch_alc268(struct hda_codec *codec)
2851{
2852 struct alc_spec *spec;
Takashi Iwaia5cb4632018-06-20 12:50:11 +02002853 int i, err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002854
Takashi Iwai1d045db2011-07-07 18:23:21 +02002855 /* ALC268 has no aa-loopback mixer */
Takashi Iwai3de95172012-05-07 18:03:15 +02002856 err = alc_alloc_spec(codec, 0);
2857 if (err < 0)
2858 return err;
2859
2860 spec = codec->spec;
Takashi Iwai7504b6c2013-03-18 11:25:51 +01002861 spec->gen.beep_nid = 0x01;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002862
Takashi Iwai225068a2015-05-29 10:42:14 +02002863 spec->shutup = alc_eapd_shutup;
2864
Takashi Iwaic9af7532019-05-10 11:01:43 +02002865 alc_pre_init(codec);
2866
Takashi Iwai1727a772013-01-10 09:52:52 +01002867 snd_hda_pick_fixup(codec, alc268_fixup_models, alc268_fixup_tbl, alc268_fixups);
2868 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002869
Takashi Iwai6ebb8052011-08-16 15:15:40 +02002870 /* automatic parse from the BIOS config */
2871 err = alc268_parse_auto_config(codec);
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02002872 if (err < 0)
2873 goto error;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002874
Takashi Iwai7504b6c2013-03-18 11:25:51 +01002875 if (err > 0 && !spec->gen.no_analog &&
2876 spec->gen.autocfg.speaker_pins[0] != 0x1d) {
Takashi Iwaia5cb4632018-06-20 12:50:11 +02002877 for (i = 0; i < ARRAY_SIZE(alc268_beep_mixer); i++) {
2878 if (!snd_hda_gen_add_kctl(&spec->gen, NULL,
2879 &alc268_beep_mixer[i])) {
2880 err = -ENOMEM;
2881 goto error;
2882 }
2883 }
Takashi Iwai7504b6c2013-03-18 11:25:51 +01002884 snd_hda_add_verbs(codec, alc268_beep_init_verbs);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002885 if (!query_amp_caps(codec, 0x1d, HDA_INPUT))
2886 /* override the amp caps for beep generator */
2887 snd_hda_override_amp_caps(codec, 0x1d, HDA_INPUT,
2888 (0x0c << AC_AMPCAP_OFFSET_SHIFT) |
2889 (0x0c << AC_AMPCAP_NUM_STEPS_SHIFT) |
2890 (0x07 << AC_AMPCAP_STEP_SIZE_SHIFT) |
2891 (0 << AC_AMPCAP_MUTE_SHIFT));
2892 }
2893
Takashi Iwai1727a772013-01-10 09:52:52 +01002894 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
Takashi Iwai6e72aa52012-06-25 10:52:25 +02002895
Takashi Iwai1d045db2011-07-07 18:23:21 +02002896 return 0;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02002897
2898 error:
2899 alc_free(codec);
2900 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02002901}
2902
2903/*
2904 * ALC269
2905 */
Takashi Iwai08c189f2012-12-19 15:22:24 +01002906
Takashi Iwai1d045db2011-07-07 18:23:21 +02002907static const struct hda_pcm_stream alc269_44k_pcm_analog_playback = {
Takashi Iwai1d045db2011-07-07 18:23:21 +02002908 .rates = SNDRV_PCM_RATE_44100, /* fixed rate */
Takashi Iwai1d045db2011-07-07 18:23:21 +02002909};
2910
2911static const struct hda_pcm_stream alc269_44k_pcm_analog_capture = {
Takashi Iwai1d045db2011-07-07 18:23:21 +02002912 .rates = SNDRV_PCM_RATE_44100, /* fixed rate */
Takashi Iwai1d045db2011-07-07 18:23:21 +02002913};
2914
Takashi Iwai1d045db2011-07-07 18:23:21 +02002915/* different alc269-variants */
2916enum {
2917 ALC269_TYPE_ALC269VA,
2918 ALC269_TYPE_ALC269VB,
2919 ALC269_TYPE_ALC269VC,
Kailang Yangadcc70b2012-05-25 08:08:38 +02002920 ALC269_TYPE_ALC269VD,
Kailang Yang065380f2013-01-10 10:25:48 +01002921 ALC269_TYPE_ALC280,
2922 ALC269_TYPE_ALC282,
Kailang Yang2af02be2013-08-22 10:03:50 +02002923 ALC269_TYPE_ALC283,
Kailang Yang065380f2013-01-10 10:25:48 +01002924 ALC269_TYPE_ALC284,
Kailang Yang4731d5d2017-06-30 15:22:57 +08002925 ALC269_TYPE_ALC293,
Kailang Yang7fc7d042013-04-25 11:04:43 +02002926 ALC269_TYPE_ALC286,
Kailang Yang506b62c2014-12-18 17:07:44 +08002927 ALC269_TYPE_ALC298,
Kailang Yang1d04c9d2013-10-24 11:35:18 +02002928 ALC269_TYPE_ALC255,
Kailang Yang4344aec2014-12-17 17:39:05 +08002929 ALC269_TYPE_ALC256,
Kailang Yangf429e7e2017-12-05 15:38:24 +08002930 ALC269_TYPE_ALC257,
Kailang Yang0a6f0602017-06-30 16:00:48 +08002931 ALC269_TYPE_ALC215,
Kailang Yang42314302016-02-03 15:03:50 +08002932 ALC269_TYPE_ALC225,
Kailang Yangdcd4f0d2016-05-04 15:50:18 +08002933 ALC269_TYPE_ALC294,
Kailang Yang1078bef2018-11-08 16:36:15 +08002934 ALC269_TYPE_ALC300,
Kailang Yang6fbae352016-05-30 16:44:20 +08002935 ALC269_TYPE_ALC700,
Takashi Iwai1d045db2011-07-07 18:23:21 +02002936};
2937
2938/*
2939 * BIOS auto configuration
2940 */
2941static int alc269_parse_auto_config(struct hda_codec *codec)
2942{
Takashi Iwai1d045db2011-07-07 18:23:21 +02002943 static const hda_nid_t alc269_ignore[] = { 0x1d, 0 };
Takashi Iwai3e6179b2011-07-08 16:55:13 +02002944 static const hda_nid_t alc269_ssids[] = { 0, 0x1b, 0x14, 0x21 };
2945 static const hda_nid_t alc269va_ssids[] = { 0x15, 0x1b, 0x14, 0 };
2946 struct alc_spec *spec = codec->spec;
Kailang Yangadcc70b2012-05-25 08:08:38 +02002947 const hda_nid_t *ssids;
2948
2949 switch (spec->codec_variant) {
2950 case ALC269_TYPE_ALC269VA:
2951 case ALC269_TYPE_ALC269VC:
Kailang Yang065380f2013-01-10 10:25:48 +01002952 case ALC269_TYPE_ALC280:
2953 case ALC269_TYPE_ALC284:
Kailang Yang4731d5d2017-06-30 15:22:57 +08002954 case ALC269_TYPE_ALC293:
Kailang Yangadcc70b2012-05-25 08:08:38 +02002955 ssids = alc269va_ssids;
2956 break;
2957 case ALC269_TYPE_ALC269VB:
2958 case ALC269_TYPE_ALC269VD:
Kailang Yang065380f2013-01-10 10:25:48 +01002959 case ALC269_TYPE_ALC282:
Kailang Yang2af02be2013-08-22 10:03:50 +02002960 case ALC269_TYPE_ALC283:
Kailang Yang7fc7d042013-04-25 11:04:43 +02002961 case ALC269_TYPE_ALC286:
Kailang Yang506b62c2014-12-18 17:07:44 +08002962 case ALC269_TYPE_ALC298:
Kailang Yang1d04c9d2013-10-24 11:35:18 +02002963 case ALC269_TYPE_ALC255:
Kailang Yang4344aec2014-12-17 17:39:05 +08002964 case ALC269_TYPE_ALC256:
Kailang Yangf429e7e2017-12-05 15:38:24 +08002965 case ALC269_TYPE_ALC257:
Kailang Yang0a6f0602017-06-30 16:00:48 +08002966 case ALC269_TYPE_ALC215:
Kailang Yang42314302016-02-03 15:03:50 +08002967 case ALC269_TYPE_ALC225:
Kailang Yangdcd4f0d2016-05-04 15:50:18 +08002968 case ALC269_TYPE_ALC294:
Kailang Yang1078bef2018-11-08 16:36:15 +08002969 case ALC269_TYPE_ALC300:
Kailang Yang6fbae352016-05-30 16:44:20 +08002970 case ALC269_TYPE_ALC700:
Kailang Yangadcc70b2012-05-25 08:08:38 +02002971 ssids = alc269_ssids;
2972 break;
2973 default:
2974 ssids = alc269_ssids;
2975 break;
2976 }
Takashi Iwai1d045db2011-07-07 18:23:21 +02002977
Takashi Iwai3e6179b2011-07-08 16:55:13 +02002978 return alc_parse_auto_config(codec, alc269_ignore, ssids);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002979}
2980
Kailang Yang1387e2d2012-11-08 10:23:18 +01002981static void alc269vb_toggle_power_output(struct hda_codec *codec, int power_up)
Takashi Iwai1d045db2011-07-07 18:23:21 +02002982{
Takashi Iwai98b24882014-08-18 13:47:50 +02002983 alc_update_coef_idx(codec, 0x04, 1 << 11, power_up ? (1 << 11) : 0);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002984}
2985
2986static void alc269_shutup(struct hda_codec *codec)
2987{
Kailang Yangadcc70b2012-05-25 08:08:38 +02002988 struct alc_spec *spec = codec->spec;
2989
Kailang Yang1387e2d2012-11-08 10:23:18 +01002990 if (spec->codec_variant == ALC269_TYPE_ALC269VB)
2991 alc269vb_toggle_power_output(codec, 0);
2992 if (spec->codec_variant == ALC269_TYPE_ALC269VB &&
2993 (alc_get_coef0(codec) & 0x00ff) == 0x018) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02002994 msleep(150);
2995 }
Takashi Iwaic0ca5ec2019-02-20 16:15:45 +01002996 alc_shutup_pins(codec);
Takashi Iwai1d045db2011-07-07 18:23:21 +02002997}
2998
Takashi Iwai54db6c32014-08-18 15:11:19 +02002999static struct coef_fw alc282_coefs[] = {
3000 WRITE_COEF(0x03, 0x0002), /* Power Down Control */
Kailang Yang32fa7e42014-10-24 16:26:01 +08003001 UPDATE_COEF(0x05, 0xff3f, 0x0700), /* FIFO and filter clock */
Takashi Iwai54db6c32014-08-18 15:11:19 +02003002 WRITE_COEF(0x07, 0x0200), /* DMIC control */
3003 UPDATE_COEF(0x06, 0x00f0, 0), /* Analog clock */
3004 UPDATE_COEF(0x08, 0xfffc, 0x0c2c), /* JD */
3005 WRITE_COEF(0x0a, 0xcccc), /* JD offset1 */
3006 WRITE_COEF(0x0b, 0xcccc), /* JD offset2 */
3007 WRITE_COEF(0x0e, 0x6e00), /* LDO1/2/3, DAC/ADC */
3008 UPDATE_COEF(0x0f, 0xf800, 0x1000), /* JD */
3009 UPDATE_COEF(0x10, 0xfc00, 0x0c00), /* Capless */
3010 WRITE_COEF(0x6f, 0x0), /* Class D test 4 */
3011 UPDATE_COEF(0x0c, 0xfe00, 0), /* IO power down directly */
3012 WRITE_COEF(0x34, 0xa0c0), /* ANC */
3013 UPDATE_COEF(0x16, 0x0008, 0), /* AGC MUX */
3014 UPDATE_COEF(0x1d, 0x00e0, 0), /* DAC simple content protection */
3015 UPDATE_COEF(0x1f, 0x00e0, 0), /* ADC simple content protection */
3016 WRITE_COEF(0x21, 0x8804), /* DAC ADC Zero Detection */
3017 WRITE_COEF(0x63, 0x2902), /* PLL */
3018 WRITE_COEF(0x68, 0xa080), /* capless control 2 */
3019 WRITE_COEF(0x69, 0x3400), /* capless control 3 */
3020 WRITE_COEF(0x6a, 0x2f3e), /* capless control 4 */
3021 WRITE_COEF(0x6b, 0x0), /* capless control 5 */
3022 UPDATE_COEF(0x6d, 0x0fff, 0x0900), /* class D test 2 */
3023 WRITE_COEF(0x6e, 0x110a), /* class D test 3 */
3024 UPDATE_COEF(0x70, 0x00f8, 0x00d8), /* class D test 5 */
3025 WRITE_COEF(0x71, 0x0014), /* class D test 6 */
3026 WRITE_COEF(0x72, 0xc2ba), /* classD OCP */
3027 UPDATE_COEF(0x77, 0x0f80, 0), /* classD pure DC test */
3028 WRITE_COEF(0x6c, 0xfc06), /* Class D amp control */
3029 {}
3030};
3031
Kailang Yangcb149cb2014-03-18 16:45:32 +08003032static void alc282_restore_default_value(struct hda_codec *codec)
3033{
Takashi Iwai54db6c32014-08-18 15:11:19 +02003034 alc_process_coef_fw(codec, alc282_coefs);
Kailang Yangcb149cb2014-03-18 16:45:32 +08003035}
3036
Kailang Yang7b5c7a02014-03-18 16:15:54 +08003037static void alc282_init(struct hda_codec *codec)
3038{
3039 struct alc_spec *spec = codec->spec;
Takashi Iwai35a39f92019-02-01 11:19:50 +01003040 hda_nid_t hp_pin = alc_get_hp_pin(spec);
Kailang Yang7b5c7a02014-03-18 16:15:54 +08003041 bool hp_pin_sense;
3042 int coef78;
3043
Kailang Yangcb149cb2014-03-18 16:45:32 +08003044 alc282_restore_default_value(codec);
3045
Kailang Yang7b5c7a02014-03-18 16:15:54 +08003046 if (!hp_pin)
3047 return;
3048 hp_pin_sense = snd_hda_jack_detect(codec, hp_pin);
3049 coef78 = alc_read_coef_idx(codec, 0x78);
3050
3051 /* Index 0x78 Direct Drive HP AMP LPM Control 1 */
3052 /* Headphone capless set to high power mode */
3053 alc_write_coef_idx(codec, 0x78, 0x9004);
3054
3055 if (hp_pin_sense)
3056 msleep(2);
3057
3058 snd_hda_codec_write(codec, hp_pin, 0,
3059 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
3060
3061 if (hp_pin_sense)
3062 msleep(85);
3063
3064 snd_hda_codec_write(codec, hp_pin, 0,
3065 AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
3066
3067 if (hp_pin_sense)
3068 msleep(100);
3069
3070 /* Headphone capless set to normal mode */
3071 alc_write_coef_idx(codec, 0x78, coef78);
3072}
3073
3074static void alc282_shutup(struct hda_codec *codec)
3075{
3076 struct alc_spec *spec = codec->spec;
Takashi Iwai35a39f92019-02-01 11:19:50 +01003077 hda_nid_t hp_pin = alc_get_hp_pin(spec);
Kailang Yang7b5c7a02014-03-18 16:15:54 +08003078 bool hp_pin_sense;
3079 int coef78;
3080
3081 if (!hp_pin) {
3082 alc269_shutup(codec);
3083 return;
3084 }
3085
3086 hp_pin_sense = snd_hda_jack_detect(codec, hp_pin);
3087 coef78 = alc_read_coef_idx(codec, 0x78);
3088 alc_write_coef_idx(codec, 0x78, 0x9004);
3089
3090 if (hp_pin_sense)
3091 msleep(2);
3092
3093 snd_hda_codec_write(codec, hp_pin, 0,
3094 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
3095
3096 if (hp_pin_sense)
3097 msleep(85);
3098
Takashi Iwaic0ca5ec2019-02-20 16:15:45 +01003099 if (!spec->no_shutup_pins)
3100 snd_hda_codec_write(codec, hp_pin, 0,
3101 AC_VERB_SET_PIN_WIDGET_CONTROL, 0x0);
Kailang Yang7b5c7a02014-03-18 16:15:54 +08003102
3103 if (hp_pin_sense)
3104 msleep(100);
3105
3106 alc_auto_setup_eapd(codec, false);
Takashi Iwaic0ca5ec2019-02-20 16:15:45 +01003107 alc_shutup_pins(codec);
Kailang Yang7b5c7a02014-03-18 16:15:54 +08003108 alc_write_coef_idx(codec, 0x78, coef78);
3109}
3110
Takashi Iwai54db6c32014-08-18 15:11:19 +02003111static struct coef_fw alc283_coefs[] = {
3112 WRITE_COEF(0x03, 0x0002), /* Power Down Control */
Kailang Yang56779862014-10-24 16:17:51 +08003113 UPDATE_COEF(0x05, 0xff3f, 0x0700), /* FIFO and filter clock */
Takashi Iwai54db6c32014-08-18 15:11:19 +02003114 WRITE_COEF(0x07, 0x0200), /* DMIC control */
3115 UPDATE_COEF(0x06, 0x00f0, 0), /* Analog clock */
3116 UPDATE_COEF(0x08, 0xfffc, 0x0c2c), /* JD */
3117 WRITE_COEF(0x0a, 0xcccc), /* JD offset1 */
3118 WRITE_COEF(0x0b, 0xcccc), /* JD offset2 */
3119 WRITE_COEF(0x0e, 0x6fc0), /* LDO1/2/3, DAC/ADC */
3120 UPDATE_COEF(0x0f, 0xf800, 0x1000), /* JD */
3121 UPDATE_COEF(0x10, 0xfc00, 0x0c00), /* Capless */
3122 WRITE_COEF(0x3a, 0x0), /* Class D test 4 */
3123 UPDATE_COEF(0x0c, 0xfe00, 0x0), /* IO power down directly */
3124 WRITE_COEF(0x22, 0xa0c0), /* ANC */
3125 UPDATE_COEFEX(0x53, 0x01, 0x000f, 0x0008), /* AGC MUX */
3126 UPDATE_COEF(0x1d, 0x00e0, 0), /* DAC simple content protection */
3127 UPDATE_COEF(0x1f, 0x00e0, 0), /* ADC simple content protection */
3128 WRITE_COEF(0x21, 0x8804), /* DAC ADC Zero Detection */
3129 WRITE_COEF(0x2e, 0x2902), /* PLL */
3130 WRITE_COEF(0x33, 0xa080), /* capless control 2 */
3131 WRITE_COEF(0x34, 0x3400), /* capless control 3 */
3132 WRITE_COEF(0x35, 0x2f3e), /* capless control 4 */
3133 WRITE_COEF(0x36, 0x0), /* capless control 5 */
3134 UPDATE_COEF(0x38, 0x0fff, 0x0900), /* class D test 2 */
3135 WRITE_COEF(0x39, 0x110a), /* class D test 3 */
3136 UPDATE_COEF(0x3b, 0x00f8, 0x00d8), /* class D test 5 */
3137 WRITE_COEF(0x3c, 0x0014), /* class D test 6 */
3138 WRITE_COEF(0x3d, 0xc2ba), /* classD OCP */
3139 UPDATE_COEF(0x42, 0x0f80, 0x0), /* classD pure DC test */
3140 WRITE_COEF(0x49, 0x0), /* test mode */
3141 UPDATE_COEF(0x40, 0xf800, 0x9800), /* Class D DC enable */
3142 UPDATE_COEF(0x42, 0xf000, 0x2000), /* DC offset */
3143 WRITE_COEF(0x37, 0xfc06), /* Class D amp control */
Kailang Yang56779862014-10-24 16:17:51 +08003144 UPDATE_COEF(0x1b, 0x8000, 0), /* HP JD control */
Takashi Iwai54db6c32014-08-18 15:11:19 +02003145 {}
3146};
3147
Kailang Yang6bd55b02014-03-17 13:51:27 +08003148static void alc283_restore_default_value(struct hda_codec *codec)
3149{
Takashi Iwai54db6c32014-08-18 15:11:19 +02003150 alc_process_coef_fw(codec, alc283_coefs);
Kailang Yang6bd55b02014-03-17 13:51:27 +08003151}
3152
Kailang Yang2af02be2013-08-22 10:03:50 +02003153static void alc283_init(struct hda_codec *codec)
3154{
3155 struct alc_spec *spec = codec->spec;
Takashi Iwai35a39f92019-02-01 11:19:50 +01003156 hda_nid_t hp_pin = alc_get_hp_pin(spec);
Kailang Yang2af02be2013-08-22 10:03:50 +02003157 bool hp_pin_sense;
Kailang Yang2af02be2013-08-22 10:03:50 +02003158
Kailang Yang6bd55b02014-03-17 13:51:27 +08003159 alc283_restore_default_value(codec);
3160
Kailang Yang2af02be2013-08-22 10:03:50 +02003161 if (!hp_pin)
3162 return;
Kailang Yanga59d7192015-04-08 16:34:00 +08003163
3164 msleep(30);
Kailang Yang2af02be2013-08-22 10:03:50 +02003165 hp_pin_sense = snd_hda_jack_detect(codec, hp_pin);
3166
3167 /* Index 0x43 Direct Drive HP AMP LPM Control 1 */
3168 /* Headphone capless set to high power mode */
3169 alc_write_coef_idx(codec, 0x43, 0x9004);
3170
3171 snd_hda_codec_write(codec, hp_pin, 0,
3172 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
3173
3174 if (hp_pin_sense)
3175 msleep(85);
3176
3177 snd_hda_codec_write(codec, hp_pin, 0,
3178 AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
3179
3180 if (hp_pin_sense)
3181 msleep(85);
3182 /* Index 0x46 Combo jack auto switch control 2 */
3183 /* 3k pull low control for Headset jack. */
Takashi Iwai98b24882014-08-18 13:47:50 +02003184 alc_update_coef_idx(codec, 0x46, 3 << 12, 0);
Kailang Yang2af02be2013-08-22 10:03:50 +02003185 /* Headphone capless set to normal mode */
3186 alc_write_coef_idx(codec, 0x43, 0x9614);
3187}
3188
3189static void alc283_shutup(struct hda_codec *codec)
3190{
3191 struct alc_spec *spec = codec->spec;
Takashi Iwai35a39f92019-02-01 11:19:50 +01003192 hda_nid_t hp_pin = alc_get_hp_pin(spec);
Kailang Yang2af02be2013-08-22 10:03:50 +02003193 bool hp_pin_sense;
Kailang Yang2af02be2013-08-22 10:03:50 +02003194
3195 if (!hp_pin) {
3196 alc269_shutup(codec);
3197 return;
3198 }
3199
3200 hp_pin_sense = snd_hda_jack_detect(codec, hp_pin);
3201
3202 alc_write_coef_idx(codec, 0x43, 0x9004);
3203
Harsha Priyab450b172014-10-09 11:04:56 +00003204 /*depop hp during suspend*/
3205 alc_write_coef_idx(codec, 0x06, 0x2100);
3206
Kailang Yang2af02be2013-08-22 10:03:50 +02003207 snd_hda_codec_write(codec, hp_pin, 0,
3208 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
3209
3210 if (hp_pin_sense)
Kailang Yang88011c092013-10-24 15:45:24 +02003211 msleep(100);
Kailang Yang2af02be2013-08-22 10:03:50 +02003212
Takashi Iwaic0ca5ec2019-02-20 16:15:45 +01003213 if (!spec->no_shutup_pins)
3214 snd_hda_codec_write(codec, hp_pin, 0,
3215 AC_VERB_SET_PIN_WIDGET_CONTROL, 0x0);
Kailang Yang2af02be2013-08-22 10:03:50 +02003216
Takashi Iwai98b24882014-08-18 13:47:50 +02003217 alc_update_coef_idx(codec, 0x46, 0, 3 << 12);
Kailang Yang2af02be2013-08-22 10:03:50 +02003218
3219 if (hp_pin_sense)
Kailang Yang88011c092013-10-24 15:45:24 +02003220 msleep(100);
Kailang Yang0435b3f2014-04-08 17:14:14 +08003221 alc_auto_setup_eapd(codec, false);
Takashi Iwaic0ca5ec2019-02-20 16:15:45 +01003222 alc_shutup_pins(codec);
Kailang Yang2af02be2013-08-22 10:03:50 +02003223 alc_write_coef_idx(codec, 0x43, 0x9614);
3224}
3225
Kailang Yang4a219ef2017-06-16 16:54:35 +08003226static void alc256_init(struct hda_codec *codec)
3227{
3228 struct alc_spec *spec = codec->spec;
Takashi Iwai35a39f92019-02-01 11:19:50 +01003229 hda_nid_t hp_pin = alc_get_hp_pin(spec);
Kailang Yang4a219ef2017-06-16 16:54:35 +08003230 bool hp_pin_sense;
3231
3232 if (!hp_pin)
Kailang Yang6447c962019-05-08 16:27:03 +08003233 hp_pin = 0x21;
Kailang Yang4a219ef2017-06-16 16:54:35 +08003234
3235 msleep(30);
3236
3237 hp_pin_sense = snd_hda_jack_detect(codec, hp_pin);
3238
3239 if (hp_pin_sense)
3240 msleep(2);
3241
3242 alc_update_coefex_idx(codec, 0x57, 0x04, 0x0007, 0x1); /* Low power */
Kailang Yang6447c962019-05-08 16:27:03 +08003243 if (spec->ultra_low_power) {
3244 alc_update_coef_idx(codec, 0x03, 1<<1, 1<<1);
3245 alc_update_coef_idx(codec, 0x08, 3<<2, 3<<2);
3246 alc_update_coef_idx(codec, 0x08, 7<<4, 0);
3247 alc_update_coef_idx(codec, 0x3b, 1<<15, 0);
3248 alc_update_coef_idx(codec, 0x0e, 7<<6, 7<<6);
3249 msleep(30);
3250 }
Kailang Yang4a219ef2017-06-16 16:54:35 +08003251
3252 snd_hda_codec_write(codec, hp_pin, 0,
3253 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
3254
Kailang Yang6447c962019-05-08 16:27:03 +08003255 if (hp_pin_sense || spec->ultra_low_power)
Kailang Yang4a219ef2017-06-16 16:54:35 +08003256 msleep(85);
3257
3258 snd_hda_codec_write(codec, hp_pin, 0,
3259 AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
3260
Kailang Yang6447c962019-05-08 16:27:03 +08003261 if (hp_pin_sense || spec->ultra_low_power)
Kailang Yang4a219ef2017-06-16 16:54:35 +08003262 msleep(100);
3263
3264 alc_update_coef_idx(codec, 0x46, 3 << 12, 0);
3265 alc_update_coefex_idx(codec, 0x57, 0x04, 0x0007, 0x4); /* Hight power */
Kailang Yang88d42b22018-03-14 16:08:57 +08003266 alc_update_coefex_idx(codec, 0x53, 0x02, 0x8000, 1 << 15); /* Clear bit */
3267 alc_update_coefex_idx(codec, 0x53, 0x02, 0x8000, 0 << 15);
Kailang Yang4a219ef2017-06-16 16:54:35 +08003268}
3269
3270static void alc256_shutup(struct hda_codec *codec)
3271{
3272 struct alc_spec *spec = codec->spec;
Takashi Iwai35a39f92019-02-01 11:19:50 +01003273 hda_nid_t hp_pin = alc_get_hp_pin(spec);
Kailang Yang4a219ef2017-06-16 16:54:35 +08003274 bool hp_pin_sense;
3275
Kailang Yang6447c962019-05-08 16:27:03 +08003276 if (!hp_pin)
3277 hp_pin = 0x21;
Kailang Yang4a219ef2017-06-16 16:54:35 +08003278
3279 hp_pin_sense = snd_hda_jack_detect(codec, hp_pin);
3280
3281 if (hp_pin_sense)
3282 msleep(2);
3283
3284 snd_hda_codec_write(codec, hp_pin, 0,
3285 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
3286
Kailang Yang6447c962019-05-08 16:27:03 +08003287 if (hp_pin_sense || spec->ultra_low_power)
Kailang Yang4a219ef2017-06-16 16:54:35 +08003288 msleep(85);
3289
Takashi Iwai1c9609e2018-01-19 14:18:34 +01003290 /* 3k pull low control for Headset jack. */
3291 /* NOTE: call this before clearing the pin, otherwise codec stalls */
3292 alc_update_coef_idx(codec, 0x46, 0, 3 << 12);
3293
Takashi Iwaic0ca5ec2019-02-20 16:15:45 +01003294 if (!spec->no_shutup_pins)
3295 snd_hda_codec_write(codec, hp_pin, 0,
3296 AC_VERB_SET_PIN_WIDGET_CONTROL, 0x0);
Kailang Yang4a219ef2017-06-16 16:54:35 +08003297
Kailang Yang6447c962019-05-08 16:27:03 +08003298 if (hp_pin_sense || spec->ultra_low_power)
Kailang Yang4a219ef2017-06-16 16:54:35 +08003299 msleep(100);
3300
3301 alc_auto_setup_eapd(codec, false);
Takashi Iwaic0ca5ec2019-02-20 16:15:45 +01003302 alc_shutup_pins(codec);
Kailang Yang6447c962019-05-08 16:27:03 +08003303 if (spec->ultra_low_power) {
3304 msleep(50);
3305 alc_update_coef_idx(codec, 0x03, 1<<1, 0);
3306 alc_update_coef_idx(codec, 0x08, 7<<4, 7<<4);
3307 alc_update_coef_idx(codec, 0x08, 3<<2, 0);
3308 alc_update_coef_idx(codec, 0x3b, 1<<15, 1<<15);
3309 alc_update_coef_idx(codec, 0x0e, 7<<6, 0);
3310 msleep(30);
3311 }
Kailang Yang4a219ef2017-06-16 16:54:35 +08003312}
3313
Kailang Yangda911b12018-01-05 16:50:08 +08003314static void alc225_init(struct hda_codec *codec)
3315{
3316 struct alc_spec *spec = codec->spec;
Takashi Iwai35a39f92019-02-01 11:19:50 +01003317 hda_nid_t hp_pin = alc_get_hp_pin(spec);
Kailang Yangda911b12018-01-05 16:50:08 +08003318 bool hp1_pin_sense, hp2_pin_sense;
3319
3320 if (!hp_pin)
Kailang Yangd3ba58b2019-05-06 15:09:42 +08003321 hp_pin = 0x21;
Kailang Yangda911b12018-01-05 16:50:08 +08003322 msleep(30);
3323
3324 hp1_pin_sense = snd_hda_jack_detect(codec, hp_pin);
3325 hp2_pin_sense = snd_hda_jack_detect(codec, 0x16);
3326
3327 if (hp1_pin_sense || hp2_pin_sense)
3328 msleep(2);
3329
3330 alc_update_coefex_idx(codec, 0x57, 0x04, 0x0007, 0x1); /* Low power */
Kailang Yangd3ba58b2019-05-06 15:09:42 +08003331 if (spec->ultra_low_power) {
3332 alc_update_coef_idx(codec, 0x08, 0x0f << 2, 3<<2);
3333 alc_update_coef_idx(codec, 0x0e, 7<<6, 7<<6);
3334 alc_update_coef_idx(codec, 0x33, 1<<11, 0);
3335 msleep(30);
3336 }
Kailang Yangda911b12018-01-05 16:50:08 +08003337
Kailang Yangd3ba58b2019-05-06 15:09:42 +08003338 if (hp1_pin_sense || spec->ultra_low_power)
Kailang Yangda911b12018-01-05 16:50:08 +08003339 snd_hda_codec_write(codec, hp_pin, 0,
3340 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
3341 if (hp2_pin_sense)
3342 snd_hda_codec_write(codec, 0x16, 0,
3343 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
3344
Kailang Yangd3ba58b2019-05-06 15:09:42 +08003345 if (hp1_pin_sense || hp2_pin_sense || spec->ultra_low_power)
Kailang Yangda911b12018-01-05 16:50:08 +08003346 msleep(85);
3347
Kailang Yangd3ba58b2019-05-06 15:09:42 +08003348 if (hp1_pin_sense || spec->ultra_low_power)
Kailang Yangda911b12018-01-05 16:50:08 +08003349 snd_hda_codec_write(codec, hp_pin, 0,
3350 AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
3351 if (hp2_pin_sense)
3352 snd_hda_codec_write(codec, 0x16, 0,
3353 AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
3354
Kailang Yangd3ba58b2019-05-06 15:09:42 +08003355 if (hp1_pin_sense || hp2_pin_sense || spec->ultra_low_power)
Kailang Yangda911b12018-01-05 16:50:08 +08003356 msleep(100);
3357
3358 alc_update_coef_idx(codec, 0x4a, 3 << 10, 0);
3359 alc_update_coefex_idx(codec, 0x57, 0x04, 0x0007, 0x4); /* Hight power */
3360}
3361
3362static void alc225_shutup(struct hda_codec *codec)
3363{
3364 struct alc_spec *spec = codec->spec;
Takashi Iwai35a39f92019-02-01 11:19:50 +01003365 hda_nid_t hp_pin = alc_get_hp_pin(spec);
Kailang Yangda911b12018-01-05 16:50:08 +08003366 bool hp1_pin_sense, hp2_pin_sense;
3367
Kailang Yangd3ba58b2019-05-06 15:09:42 +08003368 if (!hp_pin)
3369 hp_pin = 0x21;
Kailang Yangda911b12018-01-05 16:50:08 +08003370 /* 3k pull low control for Headset jack. */
3371 alc_update_coef_idx(codec, 0x4a, 0, 3 << 10);
3372
3373 hp1_pin_sense = snd_hda_jack_detect(codec, hp_pin);
3374 hp2_pin_sense = snd_hda_jack_detect(codec, 0x16);
3375
3376 if (hp1_pin_sense || hp2_pin_sense)
3377 msleep(2);
3378
Kailang Yangd3ba58b2019-05-06 15:09:42 +08003379 if (hp1_pin_sense || spec->ultra_low_power)
Kailang Yangda911b12018-01-05 16:50:08 +08003380 snd_hda_codec_write(codec, hp_pin, 0,
3381 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
3382 if (hp2_pin_sense)
3383 snd_hda_codec_write(codec, 0x16, 0,
3384 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
3385
Kailang Yangd3ba58b2019-05-06 15:09:42 +08003386 if (hp1_pin_sense || hp2_pin_sense || spec->ultra_low_power)
Kailang Yangda911b12018-01-05 16:50:08 +08003387 msleep(85);
3388
Kailang Yangd3ba58b2019-05-06 15:09:42 +08003389 if (hp1_pin_sense || spec->ultra_low_power)
Kailang Yangda911b12018-01-05 16:50:08 +08003390 snd_hda_codec_write(codec, hp_pin, 0,
3391 AC_VERB_SET_PIN_WIDGET_CONTROL, 0x0);
3392 if (hp2_pin_sense)
3393 snd_hda_codec_write(codec, 0x16, 0,
3394 AC_VERB_SET_PIN_WIDGET_CONTROL, 0x0);
3395
Kailang Yangd3ba58b2019-05-06 15:09:42 +08003396 if (hp1_pin_sense || hp2_pin_sense || spec->ultra_low_power)
Kailang Yangda911b12018-01-05 16:50:08 +08003397 msleep(100);
3398
3399 alc_auto_setup_eapd(codec, false);
Takashi Iwaic0ca5ec2019-02-20 16:15:45 +01003400 alc_shutup_pins(codec);
Kailang Yangd3ba58b2019-05-06 15:09:42 +08003401 if (spec->ultra_low_power) {
3402 msleep(50);
3403 alc_update_coef_idx(codec, 0x08, 0x0f << 2, 0x0c << 2);
3404 alc_update_coef_idx(codec, 0x0e, 7<<6, 0);
3405 alc_update_coef_idx(codec, 0x33, 1<<11, 1<<11);
3406 alc_update_coef_idx(codec, 0x4a, 3<<4, 2<<4);
3407 msleep(30);
3408 }
Kailang Yangda911b12018-01-05 16:50:08 +08003409}
3410
Kailang Yangc2d6af52017-06-21 14:50:54 +08003411static void alc_default_init(struct hda_codec *codec)
3412{
3413 struct alc_spec *spec = codec->spec;
Takashi Iwai35a39f92019-02-01 11:19:50 +01003414 hda_nid_t hp_pin = alc_get_hp_pin(spec);
Kailang Yangc2d6af52017-06-21 14:50:54 +08003415 bool hp_pin_sense;
3416
3417 if (!hp_pin)
3418 return;
3419
3420 msleep(30);
3421
3422 hp_pin_sense = snd_hda_jack_detect(codec, hp_pin);
3423
3424 if (hp_pin_sense)
3425 msleep(2);
3426
3427 snd_hda_codec_write(codec, hp_pin, 0,
3428 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
3429
3430 if (hp_pin_sense)
3431 msleep(85);
3432
3433 snd_hda_codec_write(codec, hp_pin, 0,
3434 AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
3435
3436 if (hp_pin_sense)
3437 msleep(100);
3438}
3439
3440static void alc_default_shutup(struct hda_codec *codec)
3441{
3442 struct alc_spec *spec = codec->spec;
Takashi Iwai35a39f92019-02-01 11:19:50 +01003443 hda_nid_t hp_pin = alc_get_hp_pin(spec);
Kailang Yangc2d6af52017-06-21 14:50:54 +08003444 bool hp_pin_sense;
3445
3446 if (!hp_pin) {
3447 alc269_shutup(codec);
3448 return;
3449 }
3450
3451 hp_pin_sense = snd_hda_jack_detect(codec, hp_pin);
3452
3453 if (hp_pin_sense)
3454 msleep(2);
3455
3456 snd_hda_codec_write(codec, hp_pin, 0,
3457 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
3458
3459 if (hp_pin_sense)
3460 msleep(85);
3461
Takashi Iwaic0ca5ec2019-02-20 16:15:45 +01003462 if (!spec->no_shutup_pins)
3463 snd_hda_codec_write(codec, hp_pin, 0,
3464 AC_VERB_SET_PIN_WIDGET_CONTROL, 0x0);
Kailang Yangc2d6af52017-06-21 14:50:54 +08003465
3466 if (hp_pin_sense)
3467 msleep(100);
3468
3469 alc_auto_setup_eapd(codec, false);
Takashi Iwaic0ca5ec2019-02-20 16:15:45 +01003470 alc_shutup_pins(codec);
Kailang Yangc2d6af52017-06-21 14:50:54 +08003471}
3472
Kailang Yang693abe12019-01-29 15:38:21 +08003473static void alc294_hp_init(struct hda_codec *codec)
3474{
3475 struct alc_spec *spec = codec->spec;
Takashi Iwai35a39f92019-02-01 11:19:50 +01003476 hda_nid_t hp_pin = alc_get_hp_pin(spec);
Kailang Yang693abe12019-01-29 15:38:21 +08003477 int i, val;
3478
3479 if (!hp_pin)
3480 return;
3481
3482 snd_hda_codec_write(codec, hp_pin, 0,
3483 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
3484
3485 msleep(100);
3486
Takashi Iwaic0ca5ec2019-02-20 16:15:45 +01003487 if (!spec->no_shutup_pins)
3488 snd_hda_codec_write(codec, hp_pin, 0,
3489 AC_VERB_SET_PIN_WIDGET_CONTROL, 0x0);
Kailang Yang693abe12019-01-29 15:38:21 +08003490
3491 alc_update_coef_idx(codec, 0x6f, 0x000f, 0);/* Set HP depop to manual mode */
3492 alc_update_coefex_idx(codec, 0x58, 0x00, 0x8000, 0x8000); /* HP depop procedure start */
3493
3494 /* Wait for depop procedure finish */
3495 val = alc_read_coefex_idx(codec, 0x58, 0x01);
3496 for (i = 0; i < 20 && val & 0x0080; i++) {
3497 msleep(50);
3498 val = alc_read_coefex_idx(codec, 0x58, 0x01);
3499 }
3500 /* Set HP depop to auto mode */
3501 alc_update_coef_idx(codec, 0x6f, 0x000f, 0x000b);
3502 msleep(50);
3503}
3504
3505static void alc294_init(struct hda_codec *codec)
3506{
3507 struct alc_spec *spec = codec->spec;
3508
Takashi Iwaif6ef4e02019-01-29 14:14:51 +01003509 /* required only at boot or S4 resume time */
3510 if (!spec->done_hp_init ||
3511 codec->core.dev.power.power_state.event == PM_EVENT_RESTORE) {
Kailang Yang693abe12019-01-29 15:38:21 +08003512 alc294_hp_init(codec);
3513 spec->done_hp_init = true;
3514 }
3515 alc_default_init(codec);
3516}
3517
Kailang Yangad60d502013-06-28 12:03:01 +02003518static void alc5505_coef_set(struct hda_codec *codec, unsigned int index_reg,
3519 unsigned int val)
3520{
3521 snd_hda_codec_write(codec, 0x51, 0, AC_VERB_SET_COEF_INDEX, index_reg >> 1);
3522 snd_hda_codec_write(codec, 0x51, 0, AC_VERB_SET_PROC_COEF, val & 0xffff); /* LSB */
3523 snd_hda_codec_write(codec, 0x51, 0, AC_VERB_SET_PROC_COEF, val >> 16); /* MSB */
3524}
3525
3526static int alc5505_coef_get(struct hda_codec *codec, unsigned int index_reg)
3527{
3528 unsigned int val;
3529
3530 snd_hda_codec_write(codec, 0x51, 0, AC_VERB_SET_COEF_INDEX, index_reg >> 1);
3531 val = snd_hda_codec_read(codec, 0x51, 0, AC_VERB_GET_PROC_COEF, 0)
3532 & 0xffff;
3533 val |= snd_hda_codec_read(codec, 0x51, 0, AC_VERB_GET_PROC_COEF, 0)
3534 << 16;
3535 return val;
3536}
3537
3538static void alc5505_dsp_halt(struct hda_codec *codec)
3539{
3540 unsigned int val;
3541
3542 alc5505_coef_set(codec, 0x3000, 0x000c); /* DSP CPU stop */
3543 alc5505_coef_set(codec, 0x880c, 0x0008); /* DDR enter self refresh */
3544 alc5505_coef_set(codec, 0x61c0, 0x11110080); /* Clock control for PLL and CPU */
3545 alc5505_coef_set(codec, 0x6230, 0xfc0d4011); /* Disable Input OP */
3546 alc5505_coef_set(codec, 0x61b4, 0x040a2b03); /* Stop PLL2 */
3547 alc5505_coef_set(codec, 0x61b0, 0x00005b17); /* Stop PLL1 */
3548 alc5505_coef_set(codec, 0x61b8, 0x04133303); /* Stop PLL3 */
3549 val = alc5505_coef_get(codec, 0x6220);
3550 alc5505_coef_set(codec, 0x6220, (val | 0x3000)); /* switch Ringbuffer clock to DBUS clock */
3551}
3552
3553static void alc5505_dsp_back_from_halt(struct hda_codec *codec)
3554{
3555 alc5505_coef_set(codec, 0x61b8, 0x04133302);
3556 alc5505_coef_set(codec, 0x61b0, 0x00005b16);
3557 alc5505_coef_set(codec, 0x61b4, 0x040a2b02);
3558 alc5505_coef_set(codec, 0x6230, 0xf80d4011);
3559 alc5505_coef_set(codec, 0x6220, 0x2002010f);
3560 alc5505_coef_set(codec, 0x880c, 0x00000004);
3561}
3562
3563static void alc5505_dsp_init(struct hda_codec *codec)
3564{
3565 unsigned int val;
3566
3567 alc5505_dsp_halt(codec);
3568 alc5505_dsp_back_from_halt(codec);
3569 alc5505_coef_set(codec, 0x61b0, 0x5b14); /* PLL1 control */
3570 alc5505_coef_set(codec, 0x61b0, 0x5b16);
3571 alc5505_coef_set(codec, 0x61b4, 0x04132b00); /* PLL2 control */
3572 alc5505_coef_set(codec, 0x61b4, 0x04132b02);
3573 alc5505_coef_set(codec, 0x61b8, 0x041f3300); /* PLL3 control*/
3574 alc5505_coef_set(codec, 0x61b8, 0x041f3302);
3575 snd_hda_codec_write(codec, 0x51, 0, AC_VERB_SET_CODEC_RESET, 0); /* Function reset */
3576 alc5505_coef_set(codec, 0x61b8, 0x041b3302);
3577 alc5505_coef_set(codec, 0x61b8, 0x04173302);
3578 alc5505_coef_set(codec, 0x61b8, 0x04163302);
3579 alc5505_coef_set(codec, 0x8800, 0x348b328b); /* DRAM control */
3580 alc5505_coef_set(codec, 0x8808, 0x00020022); /* DRAM control */
3581 alc5505_coef_set(codec, 0x8818, 0x00000400); /* DRAM control */
3582
3583 val = alc5505_coef_get(codec, 0x6200) >> 16; /* Read revision ID */
3584 if (val <= 3)
3585 alc5505_coef_set(codec, 0x6220, 0x2002010f); /* I/O PAD Configuration */
3586 else
3587 alc5505_coef_set(codec, 0x6220, 0x6002018f);
3588
3589 alc5505_coef_set(codec, 0x61ac, 0x055525f0); /**/
3590 alc5505_coef_set(codec, 0x61c0, 0x12230080); /* Clock control */
3591 alc5505_coef_set(codec, 0x61b4, 0x040e2b02); /* PLL2 control */
3592 alc5505_coef_set(codec, 0x61bc, 0x010234f8); /* OSC Control */
3593 alc5505_coef_set(codec, 0x880c, 0x00000004); /* DRAM Function control */
3594 alc5505_coef_set(codec, 0x880c, 0x00000003);
3595 alc5505_coef_set(codec, 0x880c, 0x00000010);
Takashi Iwaicd63a5f2013-07-05 12:13:59 +02003596
3597#ifdef HALT_REALTEK_ALC5505
3598 alc5505_dsp_halt(codec);
3599#endif
Kailang Yangad60d502013-06-28 12:03:01 +02003600}
3601
Takashi Iwaicd63a5f2013-07-05 12:13:59 +02003602#ifdef HALT_REALTEK_ALC5505
3603#define alc5505_dsp_suspend(codec) /* NOP */
3604#define alc5505_dsp_resume(codec) /* NOP */
3605#else
3606#define alc5505_dsp_suspend(codec) alc5505_dsp_halt(codec)
3607#define alc5505_dsp_resume(codec) alc5505_dsp_back_from_halt(codec)
3608#endif
3609
Takashi Iwai2a439522011-07-26 09:52:50 +02003610#ifdef CONFIG_PM
Kailang Yangad60d502013-06-28 12:03:01 +02003611static int alc269_suspend(struct hda_codec *codec)
3612{
3613 struct alc_spec *spec = codec->spec;
3614
3615 if (spec->has_alc5505_dsp)
Takashi Iwaicd63a5f2013-07-05 12:13:59 +02003616 alc5505_dsp_suspend(codec);
Kailang Yangad60d502013-06-28 12:03:01 +02003617 return alc_suspend(codec);
3618}
3619
Takashi Iwai1d045db2011-07-07 18:23:21 +02003620static int alc269_resume(struct hda_codec *codec)
3621{
Kailang Yangadcc70b2012-05-25 08:08:38 +02003622 struct alc_spec *spec = codec->spec;
3623
Kailang Yang1387e2d2012-11-08 10:23:18 +01003624 if (spec->codec_variant == ALC269_TYPE_ALC269VB)
3625 alc269vb_toggle_power_output(codec, 0);
3626 if (spec->codec_variant == ALC269_TYPE_ALC269VB &&
Kailang Yangadcc70b2012-05-25 08:08:38 +02003627 (alc_get_coef0(codec) & 0x00ff) == 0x018) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02003628 msleep(150);
3629 }
3630
3631 codec->patch_ops.init(codec);
3632
Kailang Yang1387e2d2012-11-08 10:23:18 +01003633 if (spec->codec_variant == ALC269_TYPE_ALC269VB)
3634 alc269vb_toggle_power_output(codec, 1);
3635 if (spec->codec_variant == ALC269_TYPE_ALC269VB &&
Kailang Yangadcc70b2012-05-25 08:08:38 +02003636 (alc_get_coef0(codec) & 0x00ff) == 0x017) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02003637 msleep(200);
3638 }
3639
Takashi Iwaieeecd9d2015-02-25 15:18:50 +01003640 regcache_sync(codec->core.regmap);
Takashi Iwai1d045db2011-07-07 18:23:21 +02003641 hda_call_check_power_status(codec, 0x01);
Hui Wangf4753712014-08-19 12:07:03 +08003642
3643 /* on some machine, the BIOS will clear the codec gpio data when enter
3644 * suspend, and won't restore the data after resume, so we restore it
3645 * in the driver.
3646 */
Takashi Iwaid261eec2018-06-19 22:32:29 +02003647 if (spec->gpio_data)
3648 alc_write_gpio_data(codec);
Hui Wangf4753712014-08-19 12:07:03 +08003649
Kailang Yangad60d502013-06-28 12:03:01 +02003650 if (spec->has_alc5505_dsp)
Takashi Iwaicd63a5f2013-07-05 12:13:59 +02003651 alc5505_dsp_resume(codec);
Kailang Yangc5177c82013-07-24 14:39:49 +02003652
Takashi Iwai1d045db2011-07-07 18:23:21 +02003653 return 0;
3654}
Takashi Iwai2a439522011-07-26 09:52:50 +02003655#endif /* CONFIG_PM */
Takashi Iwai1d045db2011-07-07 18:23:21 +02003656
David Henningsson108cc102012-07-20 10:37:25 +02003657static void alc269_fixup_pincfg_no_hp_to_lineout(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01003658 const struct hda_fixup *fix, int action)
David Henningsson108cc102012-07-20 10:37:25 +02003659{
3660 struct alc_spec *spec = codec->spec;
3661
Takashi Iwai1727a772013-01-10 09:52:52 +01003662 if (action == HDA_FIXUP_ACT_PRE_PROBE)
David Henningsson108cc102012-07-20 10:37:25 +02003663 spec->parse_flags = HDA_PINCFG_NO_HP_FIXUP;
3664}
3665
Jan-Marek Glogowskifdcc9682018-02-14 11:29:15 +01003666static void alc269_fixup_pincfg_U7x7_headset_mic(struct hda_codec *codec,
3667 const struct hda_fixup *fix,
3668 int action)
3669{
3670 unsigned int cfg_headphone = snd_hda_codec_get_pincfg(codec, 0x21);
3671 unsigned int cfg_headset_mic = snd_hda_codec_get_pincfg(codec, 0x19);
3672
3673 if (cfg_headphone && cfg_headset_mic == 0x411111f0)
3674 snd_hda_codec_set_pincfg(codec, 0x19,
3675 (cfg_headphone & ~AC_DEFCFG_DEVICE) |
3676 (AC_JACK_MIC_IN << AC_DEFCFG_DEVICE_SHIFT));
3677}
3678
Takashi Iwai1d045db2011-07-07 18:23:21 +02003679static void alc269_fixup_hweq(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01003680 const struct hda_fixup *fix, int action)
Takashi Iwai1d045db2011-07-07 18:23:21 +02003681{
Takashi Iwai98b24882014-08-18 13:47:50 +02003682 if (action == HDA_FIXUP_ACT_INIT)
3683 alc_update_coef_idx(codec, 0x1e, 0, 0x80);
Takashi Iwai1d045db2011-07-07 18:23:21 +02003684}
3685
David Henningsson7c478f02013-10-11 10:18:46 +02003686static void alc269_fixup_headset_mic(struct hda_codec *codec,
3687 const struct hda_fixup *fix, int action)
3688{
3689 struct alc_spec *spec = codec->spec;
3690
3691 if (action == HDA_FIXUP_ACT_PRE_PROBE)
3692 spec->parse_flags |= HDA_PINCFG_HEADSET_MIC;
3693}
3694
Takashi Iwai1d045db2011-07-07 18:23:21 +02003695static void alc271_fixup_dmic(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01003696 const struct hda_fixup *fix, int action)
Takashi Iwai1d045db2011-07-07 18:23:21 +02003697{
3698 static const struct hda_verb verbs[] = {
3699 {0x20, AC_VERB_SET_COEF_INDEX, 0x0d},
3700 {0x20, AC_VERB_SET_PROC_COEF, 0x4000},
3701 {}
3702 };
3703 unsigned int cfg;
3704
Takashi Iwai7639a062015-03-03 10:07:24 +01003705 if (strcmp(codec->core.chip_name, "ALC271X") &&
3706 strcmp(codec->core.chip_name, "ALC269VB"))
Takashi Iwai1d045db2011-07-07 18:23:21 +02003707 return;
3708 cfg = snd_hda_codec_get_pincfg(codec, 0x12);
3709 if (get_defcfg_connect(cfg) == AC_JACK_PORT_FIXED)
3710 snd_hda_sequence_write(codec, verbs);
3711}
3712
Takashi Iwai017f2a12011-07-09 14:42:25 +02003713static void alc269_fixup_pcm_44k(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01003714 const struct hda_fixup *fix, int action)
Takashi Iwai017f2a12011-07-09 14:42:25 +02003715{
3716 struct alc_spec *spec = codec->spec;
3717
Takashi Iwai1727a772013-01-10 09:52:52 +01003718 if (action != HDA_FIXUP_ACT_PROBE)
Takashi Iwai017f2a12011-07-09 14:42:25 +02003719 return;
3720
3721 /* Due to a hardware problem on Lenovo Ideadpad, we need to
3722 * fix the sample rate of analog I/O to 44.1kHz
3723 */
Takashi Iwai08c189f2012-12-19 15:22:24 +01003724 spec->gen.stream_analog_playback = &alc269_44k_pcm_analog_playback;
3725 spec->gen.stream_analog_capture = &alc269_44k_pcm_analog_capture;
Takashi Iwai017f2a12011-07-09 14:42:25 +02003726}
3727
Takashi Iwaiadabb3e2011-08-03 07:48:37 +02003728static void alc269_fixup_stereo_dmic(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01003729 const struct hda_fixup *fix, int action)
Takashi Iwaiadabb3e2011-08-03 07:48:37 +02003730{
Takashi Iwaiadabb3e2011-08-03 07:48:37 +02003731 /* The digital-mic unit sends PDM (differential signal) instead of
3732 * the standard PCM, thus you can't record a valid mono stream as is.
3733 * Below is a workaround specific to ALC269 to control the dmic
3734 * signal source as mono.
3735 */
Takashi Iwai98b24882014-08-18 13:47:50 +02003736 if (action == HDA_FIXUP_ACT_INIT)
3737 alc_update_coef_idx(codec, 0x07, 0, 0x80);
Takashi Iwaiadabb3e2011-08-03 07:48:37 +02003738}
3739
Takashi Iwai24519912011-08-16 15:08:49 +02003740static void alc269_quanta_automute(struct hda_codec *codec)
3741{
Takashi Iwai08c189f2012-12-19 15:22:24 +01003742 snd_hda_gen_update_outputs(codec);
Takashi Iwai24519912011-08-16 15:08:49 +02003743
Takashi Iwai1687ccc2014-08-18 13:49:35 +02003744 alc_write_coef_idx(codec, 0x0c, 0x680);
3745 alc_write_coef_idx(codec, 0x0c, 0x480);
Takashi Iwai24519912011-08-16 15:08:49 +02003746}
3747
3748static void alc269_fixup_quanta_mute(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01003749 const struct hda_fixup *fix, int action)
Takashi Iwai24519912011-08-16 15:08:49 +02003750{
3751 struct alc_spec *spec = codec->spec;
Takashi Iwai1727a772013-01-10 09:52:52 +01003752 if (action != HDA_FIXUP_ACT_PROBE)
Takashi Iwai24519912011-08-16 15:08:49 +02003753 return;
Takashi Iwai08c189f2012-12-19 15:22:24 +01003754 spec->gen.automute_hook = alc269_quanta_automute;
Takashi Iwai24519912011-08-16 15:08:49 +02003755}
3756
David Henningssond240d1d2013-04-15 12:50:02 +02003757static void alc269_x101_hp_automute_hook(struct hda_codec *codec,
Takashi Iwai1a4f69d2014-09-11 15:22:46 +02003758 struct hda_jack_callback *jack)
David Henningssond240d1d2013-04-15 12:50:02 +02003759{
3760 struct alc_spec *spec = codec->spec;
3761 int vref;
3762 msleep(200);
3763 snd_hda_gen_hp_automute(codec, jack);
3764
3765 vref = spec->gen.hp_jack_present ? PIN_VREF80 : 0;
3766 msleep(100);
3767 snd_hda_codec_write(codec, 0x18, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
3768 vref);
3769 msleep(500);
3770 snd_hda_codec_write(codec, 0x18, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
3771 vref);
3772}
3773
3774static void alc269_fixup_x101_headset_mic(struct hda_codec *codec,
3775 const struct hda_fixup *fix, int action)
3776{
3777 struct alc_spec *spec = codec->spec;
3778 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
3779 spec->parse_flags |= HDA_PINCFG_HEADSET_MIC;
3780 spec->gen.hp_automute_hook = alc269_x101_hp_automute_hook;
3781 }
3782}
3783
3784
Takashi Iwai08fb0d02013-01-10 17:33:58 +01003785/* update mute-LED according to the speaker mute state via mic VREF pin */
3786static void alc269_fixup_mic_mute_hook(void *private_data, int enabled)
David Henningsson6d3cd5d2013-01-07 12:03:47 +01003787{
3788 struct hda_codec *codec = private_data;
Takashi Iwai08fb0d02013-01-10 17:33:58 +01003789 struct alc_spec *spec = codec->spec;
3790 unsigned int pinval;
3791
3792 if (spec->mute_led_polarity)
3793 enabled = !enabled;
Takashi Iwai415d5552014-04-03 11:51:21 +02003794 pinval = snd_hda_codec_get_pin_target(codec, spec->mute_led_nid);
3795 pinval &= ~AC_PINCTL_VREFEN;
3796 pinval |= enabled ? AC_PINCTL_VREF_HIZ : AC_PINCTL_VREF_80;
Takashi Iwaie40bdb02018-03-17 22:40:18 +01003797 if (spec->mute_led_nid) {
3798 /* temporarily power up/down for setting VREF */
3799 snd_hda_power_up_pm(codec);
Takashi Iwai08fb0d02013-01-10 17:33:58 +01003800 snd_hda_set_pin_ctl_cache(codec, spec->mute_led_nid, pinval);
Takashi Iwaie40bdb02018-03-17 22:40:18 +01003801 snd_hda_power_down_pm(codec);
3802 }
David Henningsson6d3cd5d2013-01-07 12:03:47 +01003803}
3804
David Henningssond5b6b652013-11-06 10:50:44 +01003805/* Make sure the led works even in runtime suspend */
3806static unsigned int led_power_filter(struct hda_codec *codec,
3807 hda_nid_t nid,
3808 unsigned int power_state)
3809{
3810 struct alc_spec *spec = codec->spec;
3811
Hui Wang50dd9052014-07-08 17:56:15 +08003812 if (power_state != AC_PWRST_D3 || nid == 0 ||
3813 (nid != spec->mute_led_nid && nid != spec->cap_mute_led_nid))
David Henningssond5b6b652013-11-06 10:50:44 +01003814 return power_state;
3815
3816 /* Set pin ctl again, it might have just been set to 0 */
3817 snd_hda_set_pin_ctl(codec, nid,
3818 snd_hda_codec_get_pin_target(codec, nid));
3819
Takashi Iwaicffd3962015-04-09 10:30:25 +02003820 return snd_hda_gen_path_power_filter(codec, nid, power_state);
David Henningssond5b6b652013-11-06 10:50:44 +01003821}
3822
Takashi Iwai08fb0d02013-01-10 17:33:58 +01003823static void alc269_fixup_hp_mute_led(struct hda_codec *codec,
3824 const struct hda_fixup *fix, int action)
David Henningsson6d3cd5d2013-01-07 12:03:47 +01003825{
3826 struct alc_spec *spec = codec->spec;
Takashi Iwai08fb0d02013-01-10 17:33:58 +01003827 const struct dmi_device *dev = NULL;
3828
3829 if (action != HDA_FIXUP_ACT_PRE_PROBE)
3830 return;
3831
3832 while ((dev = dmi_find_device(DMI_DEV_TYPE_OEM_STRING, NULL, dev))) {
3833 int pol, pin;
3834 if (sscanf(dev->name, "HP_Mute_LED_%d_%x", &pol, &pin) != 2)
3835 continue;
3836 if (pin < 0x0a || pin >= 0x10)
3837 break;
3838 spec->mute_led_polarity = pol;
3839 spec->mute_led_nid = pin - 0x0a + 0x18;
3840 spec->gen.vmaster_mute.hook = alc269_fixup_mic_mute_hook;
Takashi Iwaifd25a972012-12-20 14:57:18 +01003841 spec->gen.vmaster_mute_enum = 1;
David Henningssond5b6b652013-11-06 10:50:44 +01003842 codec->power_filter = led_power_filter;
Takashi Iwai4e76a882014-02-25 12:21:03 +01003843 codec_dbg(codec,
3844 "Detected mute LED for %x:%d\n", spec->mute_led_nid,
Takashi Iwai08fb0d02013-01-10 17:33:58 +01003845 spec->mute_led_polarity);
David Henningsson6d3cd5d2013-01-07 12:03:47 +01003846 break;
3847 }
3848}
3849
Takashi Iwai85c467d2018-05-29 11:38:38 +02003850static void alc269_fixup_hp_mute_led_micx(struct hda_codec *codec,
3851 const struct hda_fixup *fix,
3852 int action, hda_nid_t pin)
David Henningssond06ac142013-02-18 11:41:55 +01003853{
3854 struct alc_spec *spec = codec->spec;
Takashi Iwai85c467d2018-05-29 11:38:38 +02003855
David Henningssond06ac142013-02-18 11:41:55 +01003856 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
3857 spec->mute_led_polarity = 0;
Takashi Iwai85c467d2018-05-29 11:38:38 +02003858 spec->mute_led_nid = pin;
David Henningssond06ac142013-02-18 11:41:55 +01003859 spec->gen.vmaster_mute.hook = alc269_fixup_mic_mute_hook;
3860 spec->gen.vmaster_mute_enum = 1;
David Henningssond5b6b652013-11-06 10:50:44 +01003861 codec->power_filter = led_power_filter;
David Henningssond06ac142013-02-18 11:41:55 +01003862 }
3863}
3864
Takashi Iwai85c467d2018-05-29 11:38:38 +02003865static void alc269_fixup_hp_mute_led_mic1(struct hda_codec *codec,
3866 const struct hda_fixup *fix, int action)
3867{
3868 alc269_fixup_hp_mute_led_micx(codec, fix, action, 0x18);
3869}
3870
Takashi Iwai08fb0d02013-01-10 17:33:58 +01003871static void alc269_fixup_hp_mute_led_mic2(struct hda_codec *codec,
3872 const struct hda_fixup *fix, int action)
Takashi Iwai420b0fe2012-03-12 12:35:27 +01003873{
Takashi Iwai85c467d2018-05-29 11:38:38 +02003874 alc269_fixup_hp_mute_led_micx(codec, fix, action, 0x19);
Takashi Iwai420b0fe2012-03-12 12:35:27 +01003875}
3876
Tom Briden7f783bd2017-03-25 10:12:01 +00003877static void alc269_fixup_hp_mute_led_mic3(struct hda_codec *codec,
3878 const struct hda_fixup *fix, int action)
3879{
Takashi Iwai85c467d2018-05-29 11:38:38 +02003880 alc269_fixup_hp_mute_led_micx(codec, fix, action, 0x1b);
Tom Briden7f783bd2017-03-25 10:12:01 +00003881}
3882
Takashi Iwai0f32fd192014-11-19 12:16:14 +01003883/* update LED status via GPIO */
3884static void alc_update_gpio_led(struct hda_codec *codec, unsigned int mask,
3885 bool enabled)
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003886{
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003887 struct alc_spec *spec = codec->spec;
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003888
Takashi Iwai0f32fd192014-11-19 12:16:14 +01003889 if (spec->mute_led_polarity)
3890 enabled = !enabled;
Takashi Iwaid261eec2018-06-19 22:32:29 +02003891 alc_update_gpio_data(codec, mask, !enabled); /* muted -> LED on */
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003892}
3893
Takashi Iwai0f32fd192014-11-19 12:16:14 +01003894/* turn on/off mute LED via GPIO per vmaster hook */
3895static void alc_fixup_gpio_mute_hook(void *private_data, int enabled)
3896{
3897 struct hda_codec *codec = private_data;
3898 struct alc_spec *spec = codec->spec;
3899
3900 alc_update_gpio_led(codec, spec->gpio_mute_led_mask, enabled);
3901}
3902
3903/* turn on/off mic-mute LED via GPIO per capture hook */
Takashi Iwaid03abec2018-06-19 12:29:13 +02003904static void alc_gpio_micmute_update(struct hda_codec *codec)
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003905{
3906 struct alc_spec *spec = codec->spec;
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003907
Takashi Iwaid03abec2018-06-19 12:29:13 +02003908 alc_update_gpio_led(codec, spec->gpio_mic_led_mask,
3909 spec->gen.micmute_led.led_value);
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003910}
3911
Takashi Iwai01e4a272018-06-19 22:47:30 +02003912/* setup mute and mic-mute GPIO bits, add hooks appropriately */
3913static void alc_fixup_hp_gpio_led(struct hda_codec *codec,
3914 int action,
3915 unsigned int mute_mask,
3916 unsigned int micmute_mask)
3917{
3918 struct alc_spec *spec = codec->spec;
3919
3920 alc_fixup_gpio(codec, action, mute_mask | micmute_mask);
3921
3922 if (action != HDA_FIXUP_ACT_PRE_PROBE)
3923 return;
3924 if (mute_mask) {
3925 spec->gpio_mute_led_mask = mute_mask;
3926 spec->gen.vmaster_mute.hook = alc_fixup_gpio_mute_hook;
3927 }
3928 if (micmute_mask) {
3929 spec->gpio_mic_led_mask = micmute_mask;
3930 snd_hda_gen_add_micmute_led(codec, alc_gpio_micmute_update);
3931 }
3932}
3933
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003934static void alc269_fixup_hp_gpio_led(struct hda_codec *codec,
3935 const struct hda_fixup *fix, int action)
3936{
Takashi Iwai01e4a272018-06-19 22:47:30 +02003937 alc_fixup_hp_gpio_led(codec, action, 0x08, 0x10);
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01003938}
3939
Kailang Yangeaa8e5e2014-11-21 15:49:11 +08003940static void alc286_fixup_hp_gpio_led(struct hda_codec *codec,
3941 const struct hda_fixup *fix, int action)
3942{
Takashi Iwai01e4a272018-06-19 22:47:30 +02003943 alc_fixup_hp_gpio_led(codec, action, 0x02, 0x20);
Takashi Iwai1d045db2011-07-07 18:23:21 +02003944}
3945
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08003946/* turn on/off mic-mute LED per capture hook */
Takashi Iwaid03abec2018-06-19 12:29:13 +02003947static void alc_cap_micmute_update(struct hda_codec *codec)
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08003948{
3949 struct alc_spec *spec = codec->spec;
Takashi Iwaid03abec2018-06-19 12:29:13 +02003950 unsigned int pinval;
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08003951
Takashi Iwaid03abec2018-06-19 12:29:13 +02003952 if (!spec->cap_mute_led_nid)
3953 return;
Hui Wangfc1fad92014-07-08 17:56:14 +08003954 pinval = snd_hda_codec_get_pin_target(codec, spec->cap_mute_led_nid);
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08003955 pinval &= ~AC_PINCTL_VREFEN;
Takashi Iwaid03abec2018-06-19 12:29:13 +02003956 if (spec->gen.micmute_led.led_value)
3957 pinval |= AC_PINCTL_VREF_80;
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08003958 else
Takashi Iwaid03abec2018-06-19 12:29:13 +02003959 pinval |= AC_PINCTL_VREF_HIZ;
3960 snd_hda_set_pin_ctl_cache(codec, spec->cap_mute_led_nid, pinval);
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08003961}
3962
3963static void alc269_fixup_hp_gpio_mic1_led(struct hda_codec *codec,
3964 const struct hda_fixup *fix, int action)
3965{
3966 struct alc_spec *spec = codec->spec;
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08003967
Takashi Iwai01e4a272018-06-19 22:47:30 +02003968 alc_fixup_hp_gpio_led(codec, action, 0x08, 0);
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08003969 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwai01e4a272018-06-19 22:47:30 +02003970 /* Like hp_gpio_mic1_led, but also needs GPIO4 low to
3971 * enable headphone amp
3972 */
3973 spec->gpio_mask |= 0x10;
3974 spec->gpio_dir |= 0x10;
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08003975 spec->cap_mute_led_nid = 0x18;
Takashi Iwaid03abec2018-06-19 12:29:13 +02003976 snd_hda_gen_add_micmute_led(codec, alc_cap_micmute_update);
Hui Wang50dd9052014-07-08 17:56:15 +08003977 codec->power_filter = led_power_filter;
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08003978 }
3979}
3980
David Henningsson7a5255f2014-10-30 08:26:01 +01003981static void alc280_fixup_hp_gpio4(struct hda_codec *codec,
3982 const struct hda_fixup *fix, int action)
3983{
David Henningsson7a5255f2014-10-30 08:26:01 +01003984 struct alc_spec *spec = codec->spec;
David Henningsson7a5255f2014-10-30 08:26:01 +01003985
Takashi Iwai01e4a272018-06-19 22:47:30 +02003986 alc_fixup_hp_gpio_led(codec, action, 0x08, 0);
David Henningsson7a5255f2014-10-30 08:26:01 +01003987 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
David Henningsson7a5255f2014-10-30 08:26:01 +01003988 spec->cap_mute_led_nid = 0x18;
Takashi Iwaid03abec2018-06-19 12:29:13 +02003989 snd_hda_gen_add_micmute_led(codec, alc_cap_micmute_update);
David Henningsson7a5255f2014-10-30 08:26:01 +01003990 codec->power_filter = led_power_filter;
3991 }
3992}
3993
Takashi Iwai6a30aba2018-04-27 17:17:35 +02003994#if IS_REACHABLE(CONFIG_INPUT)
David Henningsson33f4acd2015-01-07 15:50:13 +01003995static void gpio2_mic_hotkey_event(struct hda_codec *codec,
3996 struct hda_jack_callback *event)
3997{
3998 struct alc_spec *spec = codec->spec;
3999
4000 /* GPIO2 just toggles on a keypress/keyrelease cycle. Therefore
4001 send both key on and key off event for every interrupt. */
Hui Wangc7b60a82015-12-28 11:35:25 +08004002 input_report_key(spec->kb_dev, spec->alc_mute_keycode_map[ALC_KEY_MICMUTE_INDEX], 1);
David Henningsson33f4acd2015-01-07 15:50:13 +01004003 input_sync(spec->kb_dev);
Hui Wangc7b60a82015-12-28 11:35:25 +08004004 input_report_key(spec->kb_dev, spec->alc_mute_keycode_map[ALC_KEY_MICMUTE_INDEX], 0);
David Henningsson33f4acd2015-01-07 15:50:13 +01004005 input_sync(spec->kb_dev);
4006}
David Henningsson33f4acd2015-01-07 15:50:13 +01004007
Kailang3694cb22015-12-28 11:35:24 +08004008static int alc_register_micmute_input_device(struct hda_codec *codec)
4009{
4010 struct alc_spec *spec = codec->spec;
Hui Wangc7b60a82015-12-28 11:35:25 +08004011 int i;
Kailang3694cb22015-12-28 11:35:24 +08004012
4013 spec->kb_dev = input_allocate_device();
4014 if (!spec->kb_dev) {
4015 codec_err(codec, "Out of memory (input_allocate_device)\n");
4016 return -ENOMEM;
4017 }
Hui Wangc7b60a82015-12-28 11:35:25 +08004018
4019 spec->alc_mute_keycode_map[ALC_KEY_MICMUTE_INDEX] = KEY_MICMUTE;
4020
Kailang3694cb22015-12-28 11:35:24 +08004021 spec->kb_dev->name = "Microphone Mute Button";
4022 spec->kb_dev->evbit[0] = BIT_MASK(EV_KEY);
Hui Wangc7b60a82015-12-28 11:35:25 +08004023 spec->kb_dev->keycodesize = sizeof(spec->alc_mute_keycode_map[0]);
4024 spec->kb_dev->keycodemax = ARRAY_SIZE(spec->alc_mute_keycode_map);
4025 spec->kb_dev->keycode = spec->alc_mute_keycode_map;
4026 for (i = 0; i < ARRAY_SIZE(spec->alc_mute_keycode_map); i++)
4027 set_bit(spec->alc_mute_keycode_map[i], spec->kb_dev->keybit);
Kailang3694cb22015-12-28 11:35:24 +08004028
4029 if (input_register_device(spec->kb_dev)) {
4030 codec_err(codec, "input_register_device failed\n");
4031 input_free_device(spec->kb_dev);
4032 spec->kb_dev = NULL;
4033 return -ENOMEM;
4034 }
4035
4036 return 0;
4037}
4038
Takashi Iwai01e4a272018-06-19 22:47:30 +02004039/* GPIO1 = set according to SKU external amp
4040 * GPIO2 = mic mute hotkey
4041 * GPIO3 = mute LED
4042 * GPIO4 = mic mute LED
4043 */
David Henningsson33f4acd2015-01-07 15:50:13 +01004044static void alc280_fixup_hp_gpio2_mic_hotkey(struct hda_codec *codec,
4045 const struct hda_fixup *fix, int action)
4046{
David Henningsson33f4acd2015-01-07 15:50:13 +01004047 struct alc_spec *spec = codec->spec;
4048
Takashi Iwai01e4a272018-06-19 22:47:30 +02004049 alc_fixup_hp_gpio_led(codec, action, 0x08, 0x10);
David Henningsson33f4acd2015-01-07 15:50:13 +01004050 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwai1c76aa52018-06-21 16:37:54 +02004051 spec->init_amp = ALC_INIT_DEFAULT;
Kailang3694cb22015-12-28 11:35:24 +08004052 if (alc_register_micmute_input_device(codec) != 0)
David Henningsson33f4acd2015-01-07 15:50:13 +01004053 return;
David Henningsson33f4acd2015-01-07 15:50:13 +01004054
Takashi Iwai01e4a272018-06-19 22:47:30 +02004055 spec->gpio_mask |= 0x06;
4056 spec->gpio_dir |= 0x02;
4057 spec->gpio_data |= 0x02;
Takashi Iwai7639a062015-03-03 10:07:24 +01004058 snd_hda_codec_write_cache(codec, codec->core.afg, 0,
David Henningsson33f4acd2015-01-07 15:50:13 +01004059 AC_VERB_SET_GPIO_UNSOLICITED_RSP_MASK, 0x04);
Takashi Iwai7639a062015-03-03 10:07:24 +01004060 snd_hda_jack_detect_enable_callback(codec, codec->core.afg,
David Henningsson33f4acd2015-01-07 15:50:13 +01004061 gpio2_mic_hotkey_event);
David Henningsson33f4acd2015-01-07 15:50:13 +01004062 return;
4063 }
4064
4065 if (!spec->kb_dev)
4066 return;
4067
4068 switch (action) {
David Henningsson33f4acd2015-01-07 15:50:13 +01004069 case HDA_FIXUP_ACT_FREE:
4070 input_unregister_device(spec->kb_dev);
David Henningsson33f4acd2015-01-07 15:50:13 +01004071 spec->kb_dev = NULL;
4072 }
David Henningsson33f4acd2015-01-07 15:50:13 +01004073}
4074
Takashi Iwai01e4a272018-06-19 22:47:30 +02004075/* Line2 = mic mute hotkey
4076 * GPIO2 = mic mute LED
4077 */
Kailang3694cb22015-12-28 11:35:24 +08004078static void alc233_fixup_lenovo_line2_mic_hotkey(struct hda_codec *codec,
4079 const struct hda_fixup *fix, int action)
4080{
Kailang3694cb22015-12-28 11:35:24 +08004081 struct alc_spec *spec = codec->spec;
4082
Takashi Iwai01e4a272018-06-19 22:47:30 +02004083 alc_fixup_hp_gpio_led(codec, action, 0, 0x04);
Kailang3694cb22015-12-28 11:35:24 +08004084 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwai1c76aa52018-06-21 16:37:54 +02004085 spec->init_amp = ALC_INIT_DEFAULT;
Kailang3694cb22015-12-28 11:35:24 +08004086 if (alc_register_micmute_input_device(codec) != 0)
4087 return;
4088
Kailang3694cb22015-12-28 11:35:24 +08004089 snd_hda_jack_detect_enable_callback(codec, 0x1b,
4090 gpio2_mic_hotkey_event);
Kailang3694cb22015-12-28 11:35:24 +08004091 return;
4092 }
4093
4094 if (!spec->kb_dev)
4095 return;
4096
4097 switch (action) {
Kailang3694cb22015-12-28 11:35:24 +08004098 case HDA_FIXUP_ACT_FREE:
4099 input_unregister_device(spec->kb_dev);
4100 spec->kb_dev = NULL;
4101 }
4102}
Takashi Iwaic4696522018-01-15 10:44:35 +01004103#else /* INPUT */
4104#define alc280_fixup_hp_gpio2_mic_hotkey NULL
4105#define alc233_fixup_lenovo_line2_mic_hotkey NULL
4106#endif /* INPUT */
Kailang3694cb22015-12-28 11:35:24 +08004107
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08004108static void alc269_fixup_hp_line1_mic1_led(struct hda_codec *codec,
4109 const struct hda_fixup *fix, int action)
4110{
4111 struct alc_spec *spec = codec->spec;
4112
Takashi Iwai1bce62a2018-06-19 23:15:59 +02004113 alc269_fixup_hp_mute_led_micx(codec, fix, action, 0x1a);
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08004114 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08004115 spec->cap_mute_led_nid = 0x18;
Takashi Iwaid03abec2018-06-19 12:29:13 +02004116 snd_hda_gen_add_micmute_led(codec, alc_cap_micmute_update);
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08004117 }
4118}
4119
Kailang Yang5a367672017-07-21 15:23:53 +08004120static struct coef_fw alc225_pre_hsmode[] = {
4121 UPDATE_COEF(0x4a, 1<<8, 0),
4122 UPDATE_COEFEX(0x57, 0x05, 1<<14, 0),
4123 UPDATE_COEF(0x63, 3<<14, 3<<14),
4124 UPDATE_COEF(0x4a, 3<<4, 2<<4),
4125 UPDATE_COEF(0x4a, 3<<10, 3<<10),
4126 UPDATE_COEF(0x45, 0x3f<<10, 0x34<<10),
4127 UPDATE_COEF(0x4a, 3<<10, 0),
4128 {}
4129};
4130
David Henningsson73bdd592013-04-15 15:44:14 +02004131static void alc_headset_mode_unplugged(struct hda_codec *codec)
4132{
Takashi Iwai54db6c32014-08-18 15:11:19 +02004133 static struct coef_fw coef0255[] = {
Takashi Iwai54db6c32014-08-18 15:11:19 +02004134 WRITE_COEF(0x45, 0xd089), /* UAJ function set to menual mode */
4135 UPDATE_COEFEX(0x57, 0x05, 1<<14, 0), /* Direct Drive HP Amp control(Set to verb control)*/
4136 WRITE_COEF(0x06, 0x6104), /* Set MIC2 Vref gate with HP */
4137 WRITE_COEFEX(0x57, 0x03, 0x8aa6), /* Direct Drive HP Amp control */
4138 {}
4139 };
Kailang Yange69e7e02016-05-30 15:58:28 +08004140 static struct coef_fw coef0255_1[] = {
4141 WRITE_COEF(0x1b, 0x0c0b), /* LDO and MISC control */
4142 {}
4143 };
4144 static struct coef_fw coef0256[] = {
4145 WRITE_COEF(0x1b, 0x0c4b), /* LDO and MISC control */
4146 {}
4147 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004148 static struct coef_fw coef0233[] = {
4149 WRITE_COEF(0x1b, 0x0c0b),
4150 WRITE_COEF(0x45, 0xc429),
4151 UPDATE_COEF(0x35, 0x4000, 0),
4152 WRITE_COEF(0x06, 0x2104),
4153 WRITE_COEF(0x1a, 0x0001),
4154 WRITE_COEF(0x26, 0x0004),
4155 WRITE_COEF(0x32, 0x42a3),
4156 {}
4157 };
Kailang Yangf3b70332015-04-08 15:01:17 +08004158 static struct coef_fw coef0288[] = {
4159 UPDATE_COEF(0x4f, 0xfcc0, 0xc400),
4160 UPDATE_COEF(0x50, 0x2000, 0x2000),
4161 UPDATE_COEF(0x56, 0x0006, 0x0006),
4162 UPDATE_COEF(0x66, 0x0008, 0),
4163 UPDATE_COEF(0x67, 0x2000, 0),
4164 {}
4165 };
Kailang Yang89542932017-07-17 15:03:43 +08004166 static struct coef_fw coef0298[] = {
4167 UPDATE_COEF(0x19, 0x1300, 0x0300),
4168 {}
4169 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004170 static struct coef_fw coef0292[] = {
4171 WRITE_COEF(0x76, 0x000e),
4172 WRITE_COEF(0x6c, 0x2400),
4173 WRITE_COEF(0x18, 0x7308),
4174 WRITE_COEF(0x6b, 0xc429),
4175 {}
4176 };
4177 static struct coef_fw coef0293[] = {
4178 UPDATE_COEF(0x10, 7<<8, 6<<8), /* SET Line1 JD to 0 */
4179 UPDATE_COEFEX(0x57, 0x05, 1<<15|1<<13, 0x0), /* SET charge pump by verb */
4180 UPDATE_COEFEX(0x57, 0x03, 1<<10, 1<<10), /* SET EN_OSW to 1 */
4181 UPDATE_COEF(0x1a, 1<<3, 1<<3), /* Combo JD gating with LINE1-VREFO */
4182 WRITE_COEF(0x45, 0xc429), /* Set to TRS type */
4183 UPDATE_COEF(0x4a, 0x000f, 0x000e), /* Combo Jack auto detect */
4184 {}
4185 };
4186 static struct coef_fw coef0668[] = {
4187 WRITE_COEF(0x15, 0x0d40),
4188 WRITE_COEF(0xb7, 0x802b),
4189 {}
4190 };
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004191 static struct coef_fw coef0225[] = {
Kailang Yang5a367672017-07-21 15:23:53 +08004192 UPDATE_COEF(0x63, 3<<14, 0),
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004193 {}
4194 };
Kailang Yang71683c32017-06-20 16:33:50 +08004195 static struct coef_fw coef0274[] = {
4196 UPDATE_COEF(0x4a, 0x0100, 0),
4197 UPDATE_COEFEX(0x57, 0x05, 0x4000, 0),
4198 UPDATE_COEF(0x6b, 0xf000, 0x5000),
4199 UPDATE_COEF(0x4a, 0x0010, 0),
4200 UPDATE_COEF(0x4a, 0x0c00, 0x0c00),
4201 WRITE_COEF(0x45, 0x5289),
4202 UPDATE_COEF(0x4a, 0x0c00, 0),
4203 {}
4204 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004205
Takashi Iwai7639a062015-03-03 10:07:24 +01004206 switch (codec->core.vendor_id) {
Kailang Yang9a22a8f52013-11-08 15:54:49 +08004207 case 0x10ec0255:
Kailang Yange69e7e02016-05-30 15:58:28 +08004208 alc_process_coef_fw(codec, coef0255_1);
4209 alc_process_coef_fw(codec, coef0255);
4210 break;
Kailang Yang736f20a2017-10-20 15:06:34 +08004211 case 0x10ec0236:
Kailang Yang7081adf2015-03-30 17:05:37 +08004212 case 0x10ec0256:
Kailang Yange69e7e02016-05-30 15:58:28 +08004213 alc_process_coef_fw(codec, coef0256);
Takashi Iwai54db6c32014-08-18 15:11:19 +02004214 alc_process_coef_fw(codec, coef0255);
Kailang Yang9a22a8f52013-11-08 15:54:49 +08004215 break;
Kailang Yang71683c32017-06-20 16:33:50 +08004216 case 0x10ec0234:
4217 case 0x10ec0274:
4218 case 0x10ec0294:
4219 alc_process_coef_fw(codec, coef0274);
4220 break;
Kailang Yang13fd08a2014-05-21 16:49:48 +08004221 case 0x10ec0233:
David Henningsson73bdd592013-04-15 15:44:14 +02004222 case 0x10ec0283:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004223 alc_process_coef_fw(codec, coef0233);
David Henningsson73bdd592013-04-15 15:44:14 +02004224 break;
Kailang Yangf3b70332015-04-08 15:01:17 +08004225 case 0x10ec0286:
4226 case 0x10ec0288:
Kailang Yang89542932017-07-17 15:03:43 +08004227 alc_process_coef_fw(codec, coef0288);
4228 break;
Kailang Yang1a5bc8d2015-05-12 17:11:10 +08004229 case 0x10ec0298:
Kailang Yang89542932017-07-17 15:03:43 +08004230 alc_process_coef_fw(codec, coef0298);
Kailang Yangf3b70332015-04-08 15:01:17 +08004231 alc_process_coef_fw(codec, coef0288);
4232 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004233 case 0x10ec0292:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004234 alc_process_coef_fw(codec, coef0292);
David Henningsson73bdd592013-04-15 15:44:14 +02004235 break;
Kailang Yanga22aa262014-04-23 17:34:28 +08004236 case 0x10ec0293:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004237 alc_process_coef_fw(codec, coef0293);
Kailang Yanga22aa262014-04-23 17:34:28 +08004238 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004239 case 0x10ec0668:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004240 alc_process_coef_fw(codec, coef0668);
David Henningsson73bdd592013-04-15 15:44:14 +02004241 break;
Kailang Yangc2b691e2018-01-17 15:22:55 +08004242 case 0x10ec0215:
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004243 case 0x10ec0225:
Kailang Yangc2b691e2018-01-17 15:22:55 +08004244 case 0x10ec0285:
Kailang Yang7d727862016-05-24 16:46:07 +08004245 case 0x10ec0295:
Kailang Yangc2b691e2018-01-17 15:22:55 +08004246 case 0x10ec0289:
Kailang Yang28f1f9b2017-01-04 14:49:07 +08004247 case 0x10ec0299:
Kailang Yang4d4b0c52019-01-09 16:23:37 +08004248 alc_process_coef_fw(codec, alc225_pre_hsmode);
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004249 alc_process_coef_fw(codec, coef0225);
4250 break;
Kailang Yang78f4f7c2016-06-07 11:31:34 +08004251 case 0x10ec0867:
4252 alc_update_coefex_idx(codec, 0x57, 0x5, 1<<14, 0);
4253 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004254 }
Takashi Iwai4e76a882014-02-25 12:21:03 +01004255 codec_dbg(codec, "Headset jack set to unplugged mode.\n");
David Henningsson73bdd592013-04-15 15:44:14 +02004256}
4257
4258
4259static void alc_headset_mode_mic_in(struct hda_codec *codec, hda_nid_t hp_pin,
4260 hda_nid_t mic_pin)
4261{
Takashi Iwai54db6c32014-08-18 15:11:19 +02004262 static struct coef_fw coef0255[] = {
4263 WRITE_COEFEX(0x57, 0x03, 0x8aa6),
4264 WRITE_COEF(0x06, 0x6100), /* Set MIC2 Vref gate to normal */
4265 {}
4266 };
4267 static struct coef_fw coef0233[] = {
4268 UPDATE_COEF(0x35, 0, 1<<14),
4269 WRITE_COEF(0x06, 0x2100),
4270 WRITE_COEF(0x1a, 0x0021),
4271 WRITE_COEF(0x26, 0x008c),
4272 {}
4273 };
Kailang Yangf3b70332015-04-08 15:01:17 +08004274 static struct coef_fw coef0288[] = {
Kailang Yang89542932017-07-17 15:03:43 +08004275 UPDATE_COEF(0x4f, 0x00c0, 0),
Kailang Yangf3b70332015-04-08 15:01:17 +08004276 UPDATE_COEF(0x50, 0x2000, 0),
4277 UPDATE_COEF(0x56, 0x0006, 0),
4278 UPDATE_COEF(0x4f, 0xfcc0, 0xc400),
4279 UPDATE_COEF(0x66, 0x0008, 0x0008),
4280 UPDATE_COEF(0x67, 0x2000, 0x2000),
4281 {}
4282 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004283 static struct coef_fw coef0292[] = {
4284 WRITE_COEF(0x19, 0xa208),
4285 WRITE_COEF(0x2e, 0xacf0),
4286 {}
4287 };
4288 static struct coef_fw coef0293[] = {
4289 UPDATE_COEFEX(0x57, 0x05, 0, 1<<15|1<<13), /* SET charge pump by verb */
4290 UPDATE_COEFEX(0x57, 0x03, 1<<10, 0), /* SET EN_OSW to 0 */
4291 UPDATE_COEF(0x1a, 1<<3, 0), /* Combo JD gating without LINE1-VREFO */
4292 {}
4293 };
4294 static struct coef_fw coef0688[] = {
4295 WRITE_COEF(0xb7, 0x802b),
4296 WRITE_COEF(0xb5, 0x1040),
4297 UPDATE_COEF(0xc3, 0, 1<<12),
4298 {}
4299 };
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004300 static struct coef_fw coef0225[] = {
4301 UPDATE_COEFEX(0x57, 0x05, 1<<14, 1<<14),
4302 UPDATE_COEF(0x4a, 3<<4, 2<<4),
4303 UPDATE_COEF(0x63, 3<<14, 0),
4304 {}
4305 };
Kailang Yang71683c32017-06-20 16:33:50 +08004306 static struct coef_fw coef0274[] = {
4307 UPDATE_COEFEX(0x57, 0x05, 0x4000, 0x4000),
4308 UPDATE_COEF(0x4a, 0x0010, 0),
4309 UPDATE_COEF(0x6b, 0xf000, 0),
4310 {}
4311 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004312
Takashi Iwai7639a062015-03-03 10:07:24 +01004313 switch (codec->core.vendor_id) {
Kailang Yang736f20a2017-10-20 15:06:34 +08004314 case 0x10ec0236:
Kailang Yang9a22a8f52013-11-08 15:54:49 +08004315 case 0x10ec0255:
Kailang Yang7081adf2015-03-30 17:05:37 +08004316 case 0x10ec0256:
Kailang Yang9a22a8f52013-11-08 15:54:49 +08004317 alc_write_coef_idx(codec, 0x45, 0xc489);
4318 snd_hda_set_pin_ctl_cache(codec, hp_pin, 0);
Takashi Iwai54db6c32014-08-18 15:11:19 +02004319 alc_process_coef_fw(codec, coef0255);
Kailang Yang9a22a8f52013-11-08 15:54:49 +08004320 snd_hda_set_pin_ctl_cache(codec, mic_pin, PIN_VREF50);
4321 break;
Kailang Yang71683c32017-06-20 16:33:50 +08004322 case 0x10ec0234:
4323 case 0x10ec0274:
4324 case 0x10ec0294:
4325 alc_write_coef_idx(codec, 0x45, 0x4689);
4326 snd_hda_set_pin_ctl_cache(codec, hp_pin, 0);
4327 alc_process_coef_fw(codec, coef0274);
4328 snd_hda_set_pin_ctl_cache(codec, mic_pin, PIN_VREF50);
4329 break;
Kailang Yang13fd08a2014-05-21 16:49:48 +08004330 case 0x10ec0233:
David Henningsson73bdd592013-04-15 15:44:14 +02004331 case 0x10ec0283:
4332 alc_write_coef_idx(codec, 0x45, 0xc429);
4333 snd_hda_set_pin_ctl_cache(codec, hp_pin, 0);
Takashi Iwai54db6c32014-08-18 15:11:19 +02004334 alc_process_coef_fw(codec, coef0233);
David Henningsson73bdd592013-04-15 15:44:14 +02004335 snd_hda_set_pin_ctl_cache(codec, mic_pin, PIN_VREF50);
4336 break;
Kailang Yangf3b70332015-04-08 15:01:17 +08004337 case 0x10ec0286:
4338 case 0x10ec0288:
Kailang Yang1a5bc8d2015-05-12 17:11:10 +08004339 case 0x10ec0298:
Kailang Yangf3b70332015-04-08 15:01:17 +08004340 snd_hda_set_pin_ctl_cache(codec, hp_pin, 0);
4341 alc_process_coef_fw(codec, coef0288);
4342 snd_hda_set_pin_ctl_cache(codec, mic_pin, PIN_VREF50);
4343 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004344 case 0x10ec0292:
4345 snd_hda_set_pin_ctl_cache(codec, hp_pin, 0);
Takashi Iwai54db6c32014-08-18 15:11:19 +02004346 alc_process_coef_fw(codec, coef0292);
David Henningsson73bdd592013-04-15 15:44:14 +02004347 break;
Kailang Yanga22aa262014-04-23 17:34:28 +08004348 case 0x10ec0293:
4349 /* Set to TRS mode */
4350 alc_write_coef_idx(codec, 0x45, 0xc429);
4351 snd_hda_set_pin_ctl_cache(codec, hp_pin, 0);
Takashi Iwai54db6c32014-08-18 15:11:19 +02004352 alc_process_coef_fw(codec, coef0293);
Kailang Yanga22aa262014-04-23 17:34:28 +08004353 snd_hda_set_pin_ctl_cache(codec, mic_pin, PIN_VREF50);
4354 break;
Kailang Yang78f4f7c2016-06-07 11:31:34 +08004355 case 0x10ec0867:
4356 alc_update_coefex_idx(codec, 0x57, 0x5, 0, 1<<14);
4357 /* fallthru */
Kailang Yang9eb5d0e2017-01-17 15:40:25 +08004358 case 0x10ec0221:
David Henningsson1f8b46c2015-05-12 14:38:15 +02004359 case 0x10ec0662:
4360 snd_hda_set_pin_ctl_cache(codec, hp_pin, 0);
4361 snd_hda_set_pin_ctl_cache(codec, mic_pin, PIN_VREF50);
4362 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004363 case 0x10ec0668:
4364 alc_write_coef_idx(codec, 0x11, 0x0001);
4365 snd_hda_set_pin_ctl_cache(codec, hp_pin, 0);
Takashi Iwai54db6c32014-08-18 15:11:19 +02004366 alc_process_coef_fw(codec, coef0688);
David Henningsson73bdd592013-04-15 15:44:14 +02004367 snd_hda_set_pin_ctl_cache(codec, mic_pin, PIN_VREF50);
4368 break;
Kailang Yangc2b691e2018-01-17 15:22:55 +08004369 case 0x10ec0215:
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004370 case 0x10ec0225:
Kailang Yangc2b691e2018-01-17 15:22:55 +08004371 case 0x10ec0285:
Kailang Yang7d727862016-05-24 16:46:07 +08004372 case 0x10ec0295:
Kailang Yangc2b691e2018-01-17 15:22:55 +08004373 case 0x10ec0289:
Kailang Yang28f1f9b2017-01-04 14:49:07 +08004374 case 0x10ec0299:
Kailang Yang5a367672017-07-21 15:23:53 +08004375 alc_process_coef_fw(codec, alc225_pre_hsmode);
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004376 alc_update_coef_idx(codec, 0x45, 0x3f<<10, 0x31<<10);
4377 snd_hda_set_pin_ctl_cache(codec, hp_pin, 0);
4378 alc_process_coef_fw(codec, coef0225);
4379 snd_hda_set_pin_ctl_cache(codec, mic_pin, PIN_VREF50);
4380 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004381 }
Takashi Iwai4e76a882014-02-25 12:21:03 +01004382 codec_dbg(codec, "Headset jack set to mic-in mode.\n");
David Henningsson73bdd592013-04-15 15:44:14 +02004383}
4384
4385static void alc_headset_mode_default(struct hda_codec *codec)
4386{
David Henningsson2ae95572016-02-25 09:37:05 +01004387 static struct coef_fw coef0225[] = {
Kailang Yang5a367672017-07-21 15:23:53 +08004388 UPDATE_COEF(0x45, 0x3f<<10, 0x30<<10),
4389 UPDATE_COEF(0x45, 0x3f<<10, 0x31<<10),
4390 UPDATE_COEF(0x49, 3<<8, 0<<8),
4391 UPDATE_COEF(0x4a, 3<<4, 3<<4),
4392 UPDATE_COEF(0x63, 3<<14, 0),
4393 UPDATE_COEF(0x67, 0xf000, 0x3000),
David Henningsson2ae95572016-02-25 09:37:05 +01004394 {}
4395 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004396 static struct coef_fw coef0255[] = {
4397 WRITE_COEF(0x45, 0xc089),
4398 WRITE_COEF(0x45, 0xc489),
4399 WRITE_COEFEX(0x57, 0x03, 0x8ea6),
4400 WRITE_COEF(0x49, 0x0049),
4401 {}
4402 };
4403 static struct coef_fw coef0233[] = {
4404 WRITE_COEF(0x06, 0x2100),
4405 WRITE_COEF(0x32, 0x4ea3),
4406 {}
4407 };
Kailang Yangf3b70332015-04-08 15:01:17 +08004408 static struct coef_fw coef0288[] = {
4409 UPDATE_COEF(0x4f, 0xfcc0, 0xc400), /* Set to TRS type */
4410 UPDATE_COEF(0x50, 0x2000, 0x2000),
4411 UPDATE_COEF(0x56, 0x0006, 0x0006),
4412 UPDATE_COEF(0x66, 0x0008, 0),
4413 UPDATE_COEF(0x67, 0x2000, 0),
4414 {}
4415 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004416 static struct coef_fw coef0292[] = {
4417 WRITE_COEF(0x76, 0x000e),
4418 WRITE_COEF(0x6c, 0x2400),
4419 WRITE_COEF(0x6b, 0xc429),
4420 WRITE_COEF(0x18, 0x7308),
4421 {}
4422 };
4423 static struct coef_fw coef0293[] = {
4424 UPDATE_COEF(0x4a, 0x000f, 0x000e), /* Combo Jack auto detect */
4425 WRITE_COEF(0x45, 0xC429), /* Set to TRS type */
4426 UPDATE_COEF(0x1a, 1<<3, 0), /* Combo JD gating without LINE1-VREFO */
4427 {}
4428 };
4429 static struct coef_fw coef0688[] = {
4430 WRITE_COEF(0x11, 0x0041),
4431 WRITE_COEF(0x15, 0x0d40),
4432 WRITE_COEF(0xb7, 0x802b),
4433 {}
4434 };
Kailang Yang71683c32017-06-20 16:33:50 +08004435 static struct coef_fw coef0274[] = {
4436 WRITE_COEF(0x45, 0x4289),
4437 UPDATE_COEF(0x4a, 0x0010, 0x0010),
4438 UPDATE_COEF(0x6b, 0x0f00, 0),
4439 UPDATE_COEF(0x49, 0x0300, 0x0300),
4440 {}
4441 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004442
Takashi Iwai7639a062015-03-03 10:07:24 +01004443 switch (codec->core.vendor_id) {
Kailang Yangc2b691e2018-01-17 15:22:55 +08004444 case 0x10ec0215:
David Henningsson2ae95572016-02-25 09:37:05 +01004445 case 0x10ec0225:
Kailang Yangc2b691e2018-01-17 15:22:55 +08004446 case 0x10ec0285:
Kailang Yang7d727862016-05-24 16:46:07 +08004447 case 0x10ec0295:
Kailang Yangc2b691e2018-01-17 15:22:55 +08004448 case 0x10ec0289:
Kailang Yang28f1f9b2017-01-04 14:49:07 +08004449 case 0x10ec0299:
Kailang Yang5a367672017-07-21 15:23:53 +08004450 alc_process_coef_fw(codec, alc225_pre_hsmode);
David Henningsson2ae95572016-02-25 09:37:05 +01004451 alc_process_coef_fw(codec, coef0225);
4452 break;
Kailang Yang736f20a2017-10-20 15:06:34 +08004453 case 0x10ec0236:
Kailang Yang9a22a8f52013-11-08 15:54:49 +08004454 case 0x10ec0255:
Kailang Yang7081adf2015-03-30 17:05:37 +08004455 case 0x10ec0256:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004456 alc_process_coef_fw(codec, coef0255);
Kailang Yang9a22a8f52013-11-08 15:54:49 +08004457 break;
Kailang Yang71683c32017-06-20 16:33:50 +08004458 case 0x10ec0234:
4459 case 0x10ec0274:
4460 case 0x10ec0294:
4461 alc_process_coef_fw(codec, coef0274);
4462 break;
Kailang Yang13fd08a2014-05-21 16:49:48 +08004463 case 0x10ec0233:
David Henningsson73bdd592013-04-15 15:44:14 +02004464 case 0x10ec0283:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004465 alc_process_coef_fw(codec, coef0233);
David Henningsson73bdd592013-04-15 15:44:14 +02004466 break;
Kailang Yangf3b70332015-04-08 15:01:17 +08004467 case 0x10ec0286:
4468 case 0x10ec0288:
Kailang Yang1a5bc8d2015-05-12 17:11:10 +08004469 case 0x10ec0298:
Kailang Yangf3b70332015-04-08 15:01:17 +08004470 alc_process_coef_fw(codec, coef0288);
4471 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004472 case 0x10ec0292:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004473 alc_process_coef_fw(codec, coef0292);
David Henningsson73bdd592013-04-15 15:44:14 +02004474 break;
Kailang Yanga22aa262014-04-23 17:34:28 +08004475 case 0x10ec0293:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004476 alc_process_coef_fw(codec, coef0293);
Kailang Yanga22aa262014-04-23 17:34:28 +08004477 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004478 case 0x10ec0668:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004479 alc_process_coef_fw(codec, coef0688);
David Henningsson73bdd592013-04-15 15:44:14 +02004480 break;
Kailang Yang78f4f7c2016-06-07 11:31:34 +08004481 case 0x10ec0867:
4482 alc_update_coefex_idx(codec, 0x57, 0x5, 1<<14, 0);
4483 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004484 }
Takashi Iwai4e76a882014-02-25 12:21:03 +01004485 codec_dbg(codec, "Headset jack set to headphone (default) mode.\n");
David Henningsson73bdd592013-04-15 15:44:14 +02004486}
4487
4488/* Iphone type */
4489static void alc_headset_mode_ctia(struct hda_codec *codec)
4490{
Kailang Yang89542932017-07-17 15:03:43 +08004491 int val;
4492
Takashi Iwai54db6c32014-08-18 15:11:19 +02004493 static struct coef_fw coef0255[] = {
4494 WRITE_COEF(0x45, 0xd489), /* Set to CTIA type */
4495 WRITE_COEF(0x1b, 0x0c2b),
4496 WRITE_COEFEX(0x57, 0x03, 0x8ea6),
4497 {}
4498 };
Kailang Yange69e7e02016-05-30 15:58:28 +08004499 static struct coef_fw coef0256[] = {
4500 WRITE_COEF(0x45, 0xd489), /* Set to CTIA type */
4501 WRITE_COEF(0x1b, 0x0c6b),
4502 WRITE_COEFEX(0x57, 0x03, 0x8ea6),
4503 {}
4504 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004505 static struct coef_fw coef0233[] = {
4506 WRITE_COEF(0x45, 0xd429),
4507 WRITE_COEF(0x1b, 0x0c2b),
4508 WRITE_COEF(0x32, 0x4ea3),
4509 {}
4510 };
Kailang Yangf3b70332015-04-08 15:01:17 +08004511 static struct coef_fw coef0288[] = {
4512 UPDATE_COEF(0x50, 0x2000, 0x2000),
4513 UPDATE_COEF(0x56, 0x0006, 0x0006),
4514 UPDATE_COEF(0x66, 0x0008, 0),
4515 UPDATE_COEF(0x67, 0x2000, 0),
4516 {}
4517 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004518 static struct coef_fw coef0292[] = {
4519 WRITE_COEF(0x6b, 0xd429),
4520 WRITE_COEF(0x76, 0x0008),
4521 WRITE_COEF(0x18, 0x7388),
4522 {}
4523 };
4524 static struct coef_fw coef0293[] = {
4525 WRITE_COEF(0x45, 0xd429), /* Set to ctia type */
4526 UPDATE_COEF(0x10, 7<<8, 7<<8), /* SET Line1 JD to 1 */
4527 {}
4528 };
4529 static struct coef_fw coef0688[] = {
4530 WRITE_COEF(0x11, 0x0001),
4531 WRITE_COEF(0x15, 0x0d60),
4532 WRITE_COEF(0xc3, 0x0000),
4533 {}
4534 };
Kailang Yang5a367672017-07-21 15:23:53 +08004535 static struct coef_fw coef0225_1[] = {
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004536 UPDATE_COEF(0x45, 0x3f<<10, 0x35<<10),
Kailang Yang5a367672017-07-21 15:23:53 +08004537 UPDATE_COEF(0x63, 3<<14, 2<<14),
4538 {}
4539 };
4540 static struct coef_fw coef0225_2[] = {
4541 UPDATE_COEF(0x45, 0x3f<<10, 0x35<<10),
4542 UPDATE_COEF(0x63, 3<<14, 1<<14),
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004543 {}
4544 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004545
Takashi Iwai7639a062015-03-03 10:07:24 +01004546 switch (codec->core.vendor_id) {
Kailang Yang9a22a8f52013-11-08 15:54:49 +08004547 case 0x10ec0255:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004548 alc_process_coef_fw(codec, coef0255);
Kailang Yang9a22a8f52013-11-08 15:54:49 +08004549 break;
Kailang Yang736f20a2017-10-20 15:06:34 +08004550 case 0x10ec0236:
Kailang Yange69e7e02016-05-30 15:58:28 +08004551 case 0x10ec0256:
4552 alc_process_coef_fw(codec, coef0256);
4553 break;
Kailang Yang71683c32017-06-20 16:33:50 +08004554 case 0x10ec0234:
4555 case 0x10ec0274:
4556 case 0x10ec0294:
4557 alc_write_coef_idx(codec, 0x45, 0xd689);
4558 break;
Kailang Yang13fd08a2014-05-21 16:49:48 +08004559 case 0x10ec0233:
David Henningsson73bdd592013-04-15 15:44:14 +02004560 case 0x10ec0283:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004561 alc_process_coef_fw(codec, coef0233);
David Henningsson73bdd592013-04-15 15:44:14 +02004562 break;
Kailang Yang1a5bc8d2015-05-12 17:11:10 +08004563 case 0x10ec0298:
Kailang Yang89542932017-07-17 15:03:43 +08004564 val = alc_read_coef_idx(codec, 0x50);
4565 if (val & (1 << 12)) {
4566 alc_update_coef_idx(codec, 0x8e, 0x0070, 0x0020);
4567 alc_update_coef_idx(codec, 0x4f, 0xfcc0, 0xd400);
4568 msleep(300);
4569 } else {
4570 alc_update_coef_idx(codec, 0x8e, 0x0070, 0x0010);
4571 alc_update_coef_idx(codec, 0x4f, 0xfcc0, 0xd400);
4572 msleep(300);
4573 }
4574 break;
Kailang Yangf3b70332015-04-08 15:01:17 +08004575 case 0x10ec0286:
4576 case 0x10ec0288:
4577 alc_update_coef_idx(codec, 0x4f, 0xfcc0, 0xd400);
4578 msleep(300);
4579 alc_process_coef_fw(codec, coef0288);
4580 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004581 case 0x10ec0292:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004582 alc_process_coef_fw(codec, coef0292);
David Henningsson73bdd592013-04-15 15:44:14 +02004583 break;
Kailang Yanga22aa262014-04-23 17:34:28 +08004584 case 0x10ec0293:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004585 alc_process_coef_fw(codec, coef0293);
Kailang Yanga22aa262014-04-23 17:34:28 +08004586 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004587 case 0x10ec0668:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004588 alc_process_coef_fw(codec, coef0688);
David Henningsson73bdd592013-04-15 15:44:14 +02004589 break;
Kailang Yangc2b691e2018-01-17 15:22:55 +08004590 case 0x10ec0215:
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004591 case 0x10ec0225:
Kailang Yangc2b691e2018-01-17 15:22:55 +08004592 case 0x10ec0285:
Kailang Yang7d727862016-05-24 16:46:07 +08004593 case 0x10ec0295:
Kailang Yangc2b691e2018-01-17 15:22:55 +08004594 case 0x10ec0289:
Kailang Yang28f1f9b2017-01-04 14:49:07 +08004595 case 0x10ec0299:
Kailang Yang5a367672017-07-21 15:23:53 +08004596 val = alc_read_coef_idx(codec, 0x45);
4597 if (val & (1 << 9))
4598 alc_process_coef_fw(codec, coef0225_2);
4599 else
4600 alc_process_coef_fw(codec, coef0225_1);
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004601 break;
Kailang Yang78f4f7c2016-06-07 11:31:34 +08004602 case 0x10ec0867:
4603 alc_update_coefex_idx(codec, 0x57, 0x5, 1<<14, 0);
4604 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004605 }
Takashi Iwai4e76a882014-02-25 12:21:03 +01004606 codec_dbg(codec, "Headset jack set to iPhone-style headset mode.\n");
David Henningsson73bdd592013-04-15 15:44:14 +02004607}
4608
4609/* Nokia type */
4610static void alc_headset_mode_omtp(struct hda_codec *codec)
4611{
Takashi Iwai54db6c32014-08-18 15:11:19 +02004612 static struct coef_fw coef0255[] = {
4613 WRITE_COEF(0x45, 0xe489), /* Set to OMTP Type */
4614 WRITE_COEF(0x1b, 0x0c2b),
4615 WRITE_COEFEX(0x57, 0x03, 0x8ea6),
4616 {}
4617 };
Kailang Yange69e7e02016-05-30 15:58:28 +08004618 static struct coef_fw coef0256[] = {
4619 WRITE_COEF(0x45, 0xe489), /* Set to OMTP Type */
4620 WRITE_COEF(0x1b, 0x0c6b),
4621 WRITE_COEFEX(0x57, 0x03, 0x8ea6),
4622 {}
4623 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004624 static struct coef_fw coef0233[] = {
4625 WRITE_COEF(0x45, 0xe429),
4626 WRITE_COEF(0x1b, 0x0c2b),
4627 WRITE_COEF(0x32, 0x4ea3),
4628 {}
4629 };
Kailang Yangf3b70332015-04-08 15:01:17 +08004630 static struct coef_fw coef0288[] = {
4631 UPDATE_COEF(0x50, 0x2000, 0x2000),
4632 UPDATE_COEF(0x56, 0x0006, 0x0006),
4633 UPDATE_COEF(0x66, 0x0008, 0),
4634 UPDATE_COEF(0x67, 0x2000, 0),
4635 {}
4636 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004637 static struct coef_fw coef0292[] = {
4638 WRITE_COEF(0x6b, 0xe429),
4639 WRITE_COEF(0x76, 0x0008),
4640 WRITE_COEF(0x18, 0x7388),
4641 {}
4642 };
4643 static struct coef_fw coef0293[] = {
4644 WRITE_COEF(0x45, 0xe429), /* Set to omtp type */
4645 UPDATE_COEF(0x10, 7<<8, 7<<8), /* SET Line1 JD to 1 */
4646 {}
4647 };
4648 static struct coef_fw coef0688[] = {
4649 WRITE_COEF(0x11, 0x0001),
4650 WRITE_COEF(0x15, 0x0d50),
4651 WRITE_COEF(0xc3, 0x0000),
4652 {}
4653 };
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004654 static struct coef_fw coef0225[] = {
4655 UPDATE_COEF(0x45, 0x3f<<10, 0x39<<10),
Kailang Yang5a367672017-07-21 15:23:53 +08004656 UPDATE_COEF(0x63, 3<<14, 2<<14),
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004657 {}
4658 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004659
Takashi Iwai7639a062015-03-03 10:07:24 +01004660 switch (codec->core.vendor_id) {
Kailang Yang9a22a8f52013-11-08 15:54:49 +08004661 case 0x10ec0255:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004662 alc_process_coef_fw(codec, coef0255);
Kailang Yang9a22a8f52013-11-08 15:54:49 +08004663 break;
Kailang Yang736f20a2017-10-20 15:06:34 +08004664 case 0x10ec0236:
Kailang Yange69e7e02016-05-30 15:58:28 +08004665 case 0x10ec0256:
4666 alc_process_coef_fw(codec, coef0256);
4667 break;
Kailang Yang71683c32017-06-20 16:33:50 +08004668 case 0x10ec0234:
4669 case 0x10ec0274:
4670 case 0x10ec0294:
4671 alc_write_coef_idx(codec, 0x45, 0xe689);
4672 break;
Kailang Yang13fd08a2014-05-21 16:49:48 +08004673 case 0x10ec0233:
David Henningsson73bdd592013-04-15 15:44:14 +02004674 case 0x10ec0283:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004675 alc_process_coef_fw(codec, coef0233);
David Henningsson73bdd592013-04-15 15:44:14 +02004676 break;
Kailang Yang1a5bc8d2015-05-12 17:11:10 +08004677 case 0x10ec0298:
4678 alc_update_coef_idx(codec, 0x8e, 0x0070, 0x0010);/* Headset output enable */
Kailang Yang89542932017-07-17 15:03:43 +08004679 alc_update_coef_idx(codec, 0x4f, 0xfcc0, 0xe400);
4680 msleep(300);
4681 break;
Kailang Yangf3b70332015-04-08 15:01:17 +08004682 case 0x10ec0286:
4683 case 0x10ec0288:
4684 alc_update_coef_idx(codec, 0x4f, 0xfcc0, 0xe400);
4685 msleep(300);
4686 alc_process_coef_fw(codec, coef0288);
4687 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004688 case 0x10ec0292:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004689 alc_process_coef_fw(codec, coef0292);
David Henningsson73bdd592013-04-15 15:44:14 +02004690 break;
Kailang Yanga22aa262014-04-23 17:34:28 +08004691 case 0x10ec0293:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004692 alc_process_coef_fw(codec, coef0293);
Kailang Yanga22aa262014-04-23 17:34:28 +08004693 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004694 case 0x10ec0668:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004695 alc_process_coef_fw(codec, coef0688);
David Henningsson73bdd592013-04-15 15:44:14 +02004696 break;
Kailang Yangc2b691e2018-01-17 15:22:55 +08004697 case 0x10ec0215:
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004698 case 0x10ec0225:
Kailang Yangc2b691e2018-01-17 15:22:55 +08004699 case 0x10ec0285:
Kailang Yang7d727862016-05-24 16:46:07 +08004700 case 0x10ec0295:
Kailang Yangc2b691e2018-01-17 15:22:55 +08004701 case 0x10ec0289:
Kailang Yang28f1f9b2017-01-04 14:49:07 +08004702 case 0x10ec0299:
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004703 alc_process_coef_fw(codec, coef0225);
4704 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004705 }
Takashi Iwai4e76a882014-02-25 12:21:03 +01004706 codec_dbg(codec, "Headset jack set to Nokia-style headset mode.\n");
David Henningsson73bdd592013-04-15 15:44:14 +02004707}
4708
4709static void alc_determine_headset_type(struct hda_codec *codec)
4710{
4711 int val;
4712 bool is_ctia = false;
4713 struct alc_spec *spec = codec->spec;
Takashi Iwai54db6c32014-08-18 15:11:19 +02004714 static struct coef_fw coef0255[] = {
4715 WRITE_COEF(0x45, 0xd089), /* combo jack auto switch control(Check type)*/
4716 WRITE_COEF(0x49, 0x0149), /* combo jack auto switch control(Vref
4717 conteol) */
4718 {}
4719 };
Kailang Yangf3b70332015-04-08 15:01:17 +08004720 static struct coef_fw coef0288[] = {
4721 UPDATE_COEF(0x4f, 0xfcc0, 0xd400), /* Check Type */
4722 {}
4723 };
Kailang Yang89542932017-07-17 15:03:43 +08004724 static struct coef_fw coef0298[] = {
4725 UPDATE_COEF(0x50, 0x2000, 0x2000),
4726 UPDATE_COEF(0x56, 0x0006, 0x0006),
4727 UPDATE_COEF(0x66, 0x0008, 0),
4728 UPDATE_COEF(0x67, 0x2000, 0),
4729 UPDATE_COEF(0x19, 0x1300, 0x1300),
4730 {}
4731 };
Takashi Iwai54db6c32014-08-18 15:11:19 +02004732 static struct coef_fw coef0293[] = {
4733 UPDATE_COEF(0x4a, 0x000f, 0x0008), /* Combo Jack auto detect */
4734 WRITE_COEF(0x45, 0xD429), /* Set to ctia type */
4735 {}
4736 };
4737 static struct coef_fw coef0688[] = {
4738 WRITE_COEF(0x11, 0x0001),
4739 WRITE_COEF(0xb7, 0x802b),
4740 WRITE_COEF(0x15, 0x0d60),
4741 WRITE_COEF(0xc3, 0x0c00),
4742 {}
4743 };
Kailang Yang71683c32017-06-20 16:33:50 +08004744 static struct coef_fw coef0274[] = {
4745 UPDATE_COEF(0x4a, 0x0010, 0),
4746 UPDATE_COEF(0x4a, 0x8000, 0),
4747 WRITE_COEF(0x45, 0xd289),
4748 UPDATE_COEF(0x49, 0x0300, 0x0300),
4749 {}
4750 };
David Henningsson73bdd592013-04-15 15:44:14 +02004751
Takashi Iwai7639a062015-03-03 10:07:24 +01004752 switch (codec->core.vendor_id) {
Kailang Yang736f20a2017-10-20 15:06:34 +08004753 case 0x10ec0236:
Kailang Yang9a22a8f52013-11-08 15:54:49 +08004754 case 0x10ec0255:
Kailang Yang7081adf2015-03-30 17:05:37 +08004755 case 0x10ec0256:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004756 alc_process_coef_fw(codec, coef0255);
Kailang Yang9a22a8f52013-11-08 15:54:49 +08004757 msleep(300);
4758 val = alc_read_coef_idx(codec, 0x46);
4759 is_ctia = (val & 0x0070) == 0x0070;
4760 break;
Kailang Yang71683c32017-06-20 16:33:50 +08004761 case 0x10ec0234:
4762 case 0x10ec0274:
4763 case 0x10ec0294:
4764 alc_process_coef_fw(codec, coef0274);
4765 msleep(80);
4766 val = alc_read_coef_idx(codec, 0x46);
4767 is_ctia = (val & 0x00f0) == 0x00f0;
4768 break;
Kailang Yang13fd08a2014-05-21 16:49:48 +08004769 case 0x10ec0233:
David Henningsson73bdd592013-04-15 15:44:14 +02004770 case 0x10ec0283:
4771 alc_write_coef_idx(codec, 0x45, 0xd029);
4772 msleep(300);
4773 val = alc_read_coef_idx(codec, 0x46);
4774 is_ctia = (val & 0x0070) == 0x0070;
4775 break;
Kailang Yang1a5bc8d2015-05-12 17:11:10 +08004776 case 0x10ec0298:
Kailang Yang89542932017-07-17 15:03:43 +08004777 snd_hda_codec_write(codec, 0x21, 0,
4778 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
4779 msleep(100);
4780 snd_hda_codec_write(codec, 0x21, 0,
4781 AC_VERB_SET_PIN_WIDGET_CONTROL, 0x0);
4782 msleep(200);
4783
4784 val = alc_read_coef_idx(codec, 0x50);
4785 if (val & (1 << 12)) {
4786 alc_update_coef_idx(codec, 0x8e, 0x0070, 0x0020);
4787 alc_process_coef_fw(codec, coef0288);
4788 msleep(350);
4789 val = alc_read_coef_idx(codec, 0x50);
4790 is_ctia = (val & 0x0070) == 0x0070;
4791 } else {
4792 alc_update_coef_idx(codec, 0x8e, 0x0070, 0x0010);
4793 alc_process_coef_fw(codec, coef0288);
4794 msleep(350);
4795 val = alc_read_coef_idx(codec, 0x50);
4796 is_ctia = (val & 0x0070) == 0x0070;
4797 }
4798 alc_process_coef_fw(codec, coef0298);
4799 snd_hda_codec_write(codec, 0x21, 0,
4800 AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP);
4801 msleep(75);
4802 snd_hda_codec_write(codec, 0x21, 0,
4803 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE);
4804 break;
Kailang Yangf3b70332015-04-08 15:01:17 +08004805 case 0x10ec0286:
4806 case 0x10ec0288:
4807 alc_process_coef_fw(codec, coef0288);
4808 msleep(350);
4809 val = alc_read_coef_idx(codec, 0x50);
4810 is_ctia = (val & 0x0070) == 0x0070;
4811 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004812 case 0x10ec0292:
4813 alc_write_coef_idx(codec, 0x6b, 0xd429);
4814 msleep(300);
4815 val = alc_read_coef_idx(codec, 0x6c);
4816 is_ctia = (val & 0x001c) == 0x001c;
4817 break;
Kailang Yanga22aa262014-04-23 17:34:28 +08004818 case 0x10ec0293:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004819 alc_process_coef_fw(codec, coef0293);
Kailang Yanga22aa262014-04-23 17:34:28 +08004820 msleep(300);
4821 val = alc_read_coef_idx(codec, 0x46);
4822 is_ctia = (val & 0x0070) == 0x0070;
4823 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004824 case 0x10ec0668:
Takashi Iwai54db6c32014-08-18 15:11:19 +02004825 alc_process_coef_fw(codec, coef0688);
David Henningsson73bdd592013-04-15 15:44:14 +02004826 msleep(300);
4827 val = alc_read_coef_idx(codec, 0xbe);
4828 is_ctia = (val & 0x1c02) == 0x1c02;
4829 break;
Kailang Yangc2b691e2018-01-17 15:22:55 +08004830 case 0x10ec0215:
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004831 case 0x10ec0225:
Kailang Yangc2b691e2018-01-17 15:22:55 +08004832 case 0x10ec0285:
Kailang Yang7d727862016-05-24 16:46:07 +08004833 case 0x10ec0295:
Kailang Yangc2b691e2018-01-17 15:22:55 +08004834 case 0x10ec0289:
Kailang Yang28f1f9b2017-01-04 14:49:07 +08004835 case 0x10ec0299:
Kailang Yangda911b12018-01-05 16:50:08 +08004836 snd_hda_codec_write(codec, 0x21, 0,
4837 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
4838 msleep(80);
4839 snd_hda_codec_write(codec, 0x21, 0,
4840 AC_VERB_SET_PIN_WIDGET_CONTROL, 0x0);
4841
Kailang Yang5a367672017-07-21 15:23:53 +08004842 alc_process_coef_fw(codec, alc225_pre_hsmode);
4843 alc_update_coef_idx(codec, 0x67, 0xf000, 0x1000);
4844 val = alc_read_coef_idx(codec, 0x45);
4845 if (val & (1 << 9)) {
4846 alc_update_coef_idx(codec, 0x45, 0x3f<<10, 0x34<<10);
4847 alc_update_coef_idx(codec, 0x49, 3<<8, 2<<8);
4848 msleep(800);
4849 val = alc_read_coef_idx(codec, 0x46);
4850 is_ctia = (val & 0x00f0) == 0x00f0;
4851 } else {
4852 alc_update_coef_idx(codec, 0x45, 0x3f<<10, 0x34<<10);
4853 alc_update_coef_idx(codec, 0x49, 3<<8, 1<<8);
4854 msleep(800);
4855 val = alc_read_coef_idx(codec, 0x46);
4856 is_ctia = (val & 0x00f0) == 0x00f0;
4857 }
4858 alc_update_coef_idx(codec, 0x4a, 7<<6, 7<<6);
4859 alc_update_coef_idx(codec, 0x4a, 3<<4, 3<<4);
4860 alc_update_coef_idx(codec, 0x67, 0xf000, 0x3000);
Kailang Yangda911b12018-01-05 16:50:08 +08004861
4862 snd_hda_codec_write(codec, 0x21, 0,
4863 AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
4864 msleep(80);
4865 snd_hda_codec_write(codec, 0x21, 0,
4866 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE);
Kailang Yang4cc9b9d2016-02-03 15:09:35 +08004867 break;
Kailang Yang78f4f7c2016-06-07 11:31:34 +08004868 case 0x10ec0867:
4869 is_ctia = true;
4870 break;
David Henningsson73bdd592013-04-15 15:44:14 +02004871 }
4872
Takashi Iwai4e76a882014-02-25 12:21:03 +01004873 codec_dbg(codec, "Headset jack detected iPhone-style headset: %s\n",
David Henningsson73bdd592013-04-15 15:44:14 +02004874 is_ctia ? "yes" : "no");
4875 spec->current_headset_type = is_ctia ? ALC_HEADSET_TYPE_CTIA : ALC_HEADSET_TYPE_OMTP;
4876}
4877
4878static void alc_update_headset_mode(struct hda_codec *codec)
4879{
4880 struct alc_spec *spec = codec->spec;
4881
4882 hda_nid_t mux_pin = spec->gen.imux_pins[spec->gen.cur_mux[0]];
Takashi Iwai35a39f92019-02-01 11:19:50 +01004883 hda_nid_t hp_pin = alc_get_hp_pin(spec);
David Henningsson73bdd592013-04-15 15:44:14 +02004884
4885 int new_headset_mode;
4886
4887 if (!snd_hda_jack_detect(codec, hp_pin))
4888 new_headset_mode = ALC_HEADSET_MODE_UNPLUGGED;
4889 else if (mux_pin == spec->headset_mic_pin)
4890 new_headset_mode = ALC_HEADSET_MODE_HEADSET;
4891 else if (mux_pin == spec->headphone_mic_pin)
4892 new_headset_mode = ALC_HEADSET_MODE_MIC;
4893 else
4894 new_headset_mode = ALC_HEADSET_MODE_HEADPHONE;
4895
David Henningsson5959a6b2013-11-12 11:10:57 +01004896 if (new_headset_mode == spec->current_headset_mode) {
4897 snd_hda_gen_update_outputs(codec);
David Henningsson73bdd592013-04-15 15:44:14 +02004898 return;
David Henningsson5959a6b2013-11-12 11:10:57 +01004899 }
David Henningsson73bdd592013-04-15 15:44:14 +02004900
4901 switch (new_headset_mode) {
4902 case ALC_HEADSET_MODE_UNPLUGGED:
4903 alc_headset_mode_unplugged(codec);
4904 spec->gen.hp_jack_present = false;
4905 break;
4906 case ALC_HEADSET_MODE_HEADSET:
4907 if (spec->current_headset_type == ALC_HEADSET_TYPE_UNKNOWN)
4908 alc_determine_headset_type(codec);
4909 if (spec->current_headset_type == ALC_HEADSET_TYPE_CTIA)
4910 alc_headset_mode_ctia(codec);
4911 else if (spec->current_headset_type == ALC_HEADSET_TYPE_OMTP)
4912 alc_headset_mode_omtp(codec);
4913 spec->gen.hp_jack_present = true;
4914 break;
4915 case ALC_HEADSET_MODE_MIC:
4916 alc_headset_mode_mic_in(codec, hp_pin, spec->headphone_mic_pin);
4917 spec->gen.hp_jack_present = false;
4918 break;
4919 case ALC_HEADSET_MODE_HEADPHONE:
4920 alc_headset_mode_default(codec);
4921 spec->gen.hp_jack_present = true;
4922 break;
4923 }
4924 if (new_headset_mode != ALC_HEADSET_MODE_MIC) {
4925 snd_hda_set_pin_ctl_cache(codec, hp_pin,
4926 AC_PINCTL_OUT_EN | AC_PINCTL_HP_EN);
David Henningsson1f8b46c2015-05-12 14:38:15 +02004927 if (spec->headphone_mic_pin && spec->headphone_mic_pin != hp_pin)
David Henningsson73bdd592013-04-15 15:44:14 +02004928 snd_hda_set_pin_ctl_cache(codec, spec->headphone_mic_pin,
4929 PIN_VREFHIZ);
4930 }
4931 spec->current_headset_mode = new_headset_mode;
4932
4933 snd_hda_gen_update_outputs(codec);
4934}
4935
4936static void alc_update_headset_mode_hook(struct hda_codec *codec,
Takashi Iwai7fe30712014-01-30 17:59:02 +01004937 struct snd_kcontrol *kcontrol,
4938 struct snd_ctl_elem_value *ucontrol)
David Henningsson73bdd592013-04-15 15:44:14 +02004939{
4940 alc_update_headset_mode(codec);
4941}
4942
Takashi Iwai1a4f69d2014-09-11 15:22:46 +02004943static void alc_update_headset_jack_cb(struct hda_codec *codec,
4944 struct hda_jack_callback *jack)
David Henningsson73bdd592013-04-15 15:44:14 +02004945{
4946 struct alc_spec *spec = codec->spec;
David Henningsson5db4d342013-11-22 12:17:06 +01004947 spec->current_headset_type = ALC_HEADSET_TYPE_UNKNOWN;
David Henningsson73bdd592013-04-15 15:44:14 +02004948 snd_hda_gen_hp_automute(codec, jack);
4949}
4950
4951static void alc_probe_headset_mode(struct hda_codec *codec)
4952{
4953 int i;
4954 struct alc_spec *spec = codec->spec;
4955 struct auto_pin_cfg *cfg = &spec->gen.autocfg;
4956
4957 /* Find mic pins */
4958 for (i = 0; i < cfg->num_inputs; i++) {
4959 if (cfg->inputs[i].is_headset_mic && !spec->headset_mic_pin)
4960 spec->headset_mic_pin = cfg->inputs[i].pin;
4961 if (cfg->inputs[i].is_headphone_mic && !spec->headphone_mic_pin)
4962 spec->headphone_mic_pin = cfg->inputs[i].pin;
4963 }
4964
Takashi Iwai0bed2aa2018-06-19 12:45:53 +02004965 WARN_ON(spec->gen.cap_sync_hook);
David Henningsson73bdd592013-04-15 15:44:14 +02004966 spec->gen.cap_sync_hook = alc_update_headset_mode_hook;
4967 spec->gen.automute_hook = alc_update_headset_mode;
4968 spec->gen.hp_automute_hook = alc_update_headset_jack_cb;
4969}
4970
4971static void alc_fixup_headset_mode(struct hda_codec *codec,
4972 const struct hda_fixup *fix, int action)
4973{
4974 struct alc_spec *spec = codec->spec;
4975
4976 switch (action) {
4977 case HDA_FIXUP_ACT_PRE_PROBE:
4978 spec->parse_flags |= HDA_PINCFG_HEADSET_MIC | HDA_PINCFG_HEADPHONE_MIC;
4979 break;
4980 case HDA_FIXUP_ACT_PROBE:
4981 alc_probe_headset_mode(codec);
4982 break;
4983 case HDA_FIXUP_ACT_INIT:
4984 spec->current_headset_mode = 0;
4985 alc_update_headset_mode(codec);
4986 break;
4987 }
4988}
4989
4990static void alc_fixup_headset_mode_no_hp_mic(struct hda_codec *codec,
4991 const struct hda_fixup *fix, int action)
4992{
4993 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
4994 struct alc_spec *spec = codec->spec;
4995 spec->parse_flags |= HDA_PINCFG_HEADSET_MIC;
4996 }
4997 else
4998 alc_fixup_headset_mode(codec, fix, action);
4999}
5000
Kailang Yang31278992014-03-03 15:27:22 +08005001static void alc255_set_default_jack_type(struct hda_codec *codec)
5002{
5003 /* Set to iphone type */
Kailang Yange69e7e02016-05-30 15:58:28 +08005004 static struct coef_fw alc255fw[] = {
Takashi Iwai54db6c32014-08-18 15:11:19 +02005005 WRITE_COEF(0x1b, 0x880b),
5006 WRITE_COEF(0x45, 0xd089),
5007 WRITE_COEF(0x1b, 0x080b),
5008 WRITE_COEF(0x46, 0x0004),
5009 WRITE_COEF(0x1b, 0x0c0b),
5010 {}
5011 };
Kailang Yange69e7e02016-05-30 15:58:28 +08005012 static struct coef_fw alc256fw[] = {
5013 WRITE_COEF(0x1b, 0x884b),
5014 WRITE_COEF(0x45, 0xd089),
5015 WRITE_COEF(0x1b, 0x084b),
5016 WRITE_COEF(0x46, 0x0004),
5017 WRITE_COEF(0x1b, 0x0c4b),
5018 {}
5019 };
5020 switch (codec->core.vendor_id) {
5021 case 0x10ec0255:
5022 alc_process_coef_fw(codec, alc255fw);
5023 break;
Kailang Yang736f20a2017-10-20 15:06:34 +08005024 case 0x10ec0236:
Kailang Yange69e7e02016-05-30 15:58:28 +08005025 case 0x10ec0256:
5026 alc_process_coef_fw(codec, alc256fw);
5027 break;
5028 }
Kailang Yang31278992014-03-03 15:27:22 +08005029 msleep(30);
5030}
5031
Kailang Yang9a22a8f52013-11-08 15:54:49 +08005032static void alc_fixup_headset_mode_alc255(struct hda_codec *codec,
5033 const struct hda_fixup *fix, int action)
5034{
5035 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Kailang Yang31278992014-03-03 15:27:22 +08005036 alc255_set_default_jack_type(codec);
Kailang Yang9a22a8f52013-11-08 15:54:49 +08005037 }
5038 alc_fixup_headset_mode(codec, fix, action);
5039}
5040
Kailang Yang31278992014-03-03 15:27:22 +08005041static void alc_fixup_headset_mode_alc255_no_hp_mic(struct hda_codec *codec,
5042 const struct hda_fixup *fix, int action)
5043{
5044 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
5045 struct alc_spec *spec = codec->spec;
5046 spec->parse_flags |= HDA_PINCFG_HEADSET_MIC;
5047 alc255_set_default_jack_type(codec);
5048 }
5049 else
5050 alc_fixup_headset_mode(codec, fix, action);
5051}
5052
Kailang Yange1e62b92015-04-08 16:01:22 +08005053static void alc288_update_headset_jack_cb(struct hda_codec *codec,
5054 struct hda_jack_callback *jack)
5055{
5056 struct alc_spec *spec = codec->spec;
Kailang Yange1e62b92015-04-08 16:01:22 +08005057
5058 alc_update_headset_jack_cb(codec, jack);
5059 /* Headset Mic enable or disable, only for Dell Dino */
Takashi Iwaid44a6862018-06-19 23:04:03 +02005060 alc_update_gpio_data(codec, 0x40, spec->gen.hp_jack_present);
Kailang Yange1e62b92015-04-08 16:01:22 +08005061}
5062
5063static void alc_fixup_headset_mode_dell_alc288(struct hda_codec *codec,
5064 const struct hda_fixup *fix, int action)
5065{
5066 alc_fixup_headset_mode(codec, fix, action);
5067 if (action == HDA_FIXUP_ACT_PROBE) {
5068 struct alc_spec *spec = codec->spec;
Takashi Iwaid44a6862018-06-19 23:04:03 +02005069 /* toggled via hp_automute_hook */
5070 spec->gpio_mask |= 0x40;
5071 spec->gpio_dir |= 0x40;
Kailang Yange1e62b92015-04-08 16:01:22 +08005072 spec->gen.hp_automute_hook = alc288_update_headset_jack_cb;
5073 }
5074}
5075
Hui Wang493a52a2014-01-14 14:07:36 +08005076static void alc_fixup_auto_mute_via_amp(struct hda_codec *codec,
5077 const struct hda_fixup *fix, int action)
5078{
5079 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
5080 struct alc_spec *spec = codec->spec;
5081 spec->gen.auto_mute_via_amp = 1;
5082 }
5083}
5084
Takashi Iwai9b745ab2014-03-07 08:37:19 +01005085static void alc_fixup_no_shutup(struct hda_codec *codec,
5086 const struct hda_fixup *fix, int action)
5087{
Takashi Iwaiefe55732018-06-15 11:55:02 +02005088 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwai9b745ab2014-03-07 08:37:19 +01005089 struct alc_spec *spec = codec->spec;
Takashi Iwaic0ca5ec2019-02-20 16:15:45 +01005090 spec->no_shutup_pins = 1;
Takashi Iwai9b745ab2014-03-07 08:37:19 +01005091 }
5092}
5093
Gabriele Mazzotta5e6db662014-05-20 18:58:26 +02005094static void alc_fixup_disable_aamix(struct hda_codec *codec,
5095 const struct hda_fixup *fix, int action)
5096{
5097 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
5098 struct alc_spec *spec = codec->spec;
5099 /* Disable AA-loopback as it causes white noise */
5100 spec->gen.mixer_nid = 0;
5101 }
5102}
5103
Takashi Iwai7f57d802015-09-24 17:36:51 +02005104/* fixup for Thinkpad docks: add dock pins, avoid HP parser fixup */
5105static void alc_fixup_tpt440_dock(struct hda_codec *codec,
5106 const struct hda_fixup *fix, int action)
5107{
5108 static const struct hda_pintbl pincfgs[] = {
5109 { 0x16, 0x21211010 }, /* dock headphone */
5110 { 0x19, 0x21a11010 }, /* dock mic */
5111 { }
5112 };
5113 struct alc_spec *spec = codec->spec;
5114
5115 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwai70a09762015-12-15 14:59:58 +01005116 spec->reboot_notify = alc_d3_at_reboot; /* reduce noise */
Takashi Iwai7f57d802015-09-24 17:36:51 +02005117 spec->parse_flags = HDA_PINCFG_NO_HP_FIXUP;
5118 codec->power_save_node = 0; /* avoid click noises */
5119 snd_hda_apply_pincfgs(codec, pincfgs);
5120 }
5121}
5122
Kailang Yang61fcf8e2018-02-02 15:26:46 +08005123static void alc_fixup_tpt470_dock(struct hda_codec *codec,
5124 const struct hda_fixup *fix, int action)
5125{
5126 static const struct hda_pintbl pincfgs[] = {
5127 { 0x17, 0x21211010 }, /* dock headphone */
5128 { 0x19, 0x21a11010 }, /* dock mic */
5129 { }
5130 };
Takashi Iwai54947cd2018-12-03 10:44:15 +01005131 /* Assure the speaker pin to be coupled with DAC NID 0x03; otherwise
5132 * the speaker output becomes too low by some reason on Thinkpads with
5133 * ALC298 codec
5134 */
5135 static hda_nid_t preferred_pairs[] = {
5136 0x14, 0x03, 0x17, 0x02, 0x21, 0x02,
5137 0
5138 };
Kailang Yang61fcf8e2018-02-02 15:26:46 +08005139 struct alc_spec *spec = codec->spec;
5140
5141 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwai54947cd2018-12-03 10:44:15 +01005142 spec->gen.preferred_dacs = preferred_pairs;
Kailang Yang61fcf8e2018-02-02 15:26:46 +08005143 spec->parse_flags = HDA_PINCFG_NO_HP_FIXUP;
Takashi Iwai71db96d2018-02-26 15:36:38 +01005144 snd_hda_apply_pincfgs(codec, pincfgs);
5145 } else if (action == HDA_FIXUP_ACT_INIT) {
Kailang Yang61fcf8e2018-02-02 15:26:46 +08005146 /* Enable DOCK device */
5147 snd_hda_codec_write(codec, 0x17, 0,
5148 AC_VERB_SET_CONFIG_DEFAULT_BYTES_3, 0);
5149 /* Enable DOCK device */
5150 snd_hda_codec_write(codec, 0x19, 0,
5151 AC_VERB_SET_CONFIG_DEFAULT_BYTES_3, 0);
Kailang Yang61fcf8e2018-02-02 15:26:46 +08005152 }
5153}
5154
Gabriele Mazzotta9476d3692015-04-19 19:00:40 +02005155static void alc_shutup_dell_xps13(struct hda_codec *codec)
Gabriele Mazzotta033b0a72014-05-26 17:11:46 +02005156{
5157 struct alc_spec *spec = codec->spec;
Takashi Iwai35a39f92019-02-01 11:19:50 +01005158 int hp_pin = alc_get_hp_pin(spec);
Gabriele Mazzotta033b0a72014-05-26 17:11:46 +02005159
Gabriele Mazzotta9476d3692015-04-19 19:00:40 +02005160 /* Prevent pop noises when headphones are plugged in */
5161 snd_hda_codec_write(codec, hp_pin, 0,
5162 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
5163 msleep(20);
Gabriele Mazzotta033b0a72014-05-26 17:11:46 +02005164}
5165
5166static void alc_fixup_dell_xps13(struct hda_codec *codec,
5167 const struct hda_fixup *fix, int action)
5168{
Takashi Iwai3e1b0c42015-04-27 10:43:22 +02005169 struct alc_spec *spec = codec->spec;
5170 struct hda_input_mux *imux = &spec->gen.input_mux;
5171 int i;
Gabriele Mazzottaf38663a2014-08-08 19:34:40 +02005172
Takashi Iwai3e1b0c42015-04-27 10:43:22 +02005173 switch (action) {
5174 case HDA_FIXUP_ACT_PRE_PROBE:
5175 /* mic pin 0x19 must be initialized with Vref Hi-Z, otherwise
5176 * it causes a click noise at start up
5177 */
5178 snd_hda_codec_set_pin_target(codec, 0x19, PIN_VREFHIZ);
Takashi Iwaiefe55732018-06-15 11:55:02 +02005179 spec->shutup = alc_shutup_dell_xps13;
Takashi Iwai3e1b0c42015-04-27 10:43:22 +02005180 break;
5181 case HDA_FIXUP_ACT_PROBE:
Gabriele Mazzottaf38663a2014-08-08 19:34:40 +02005182 /* Make the internal mic the default input source. */
5183 for (i = 0; i < imux->num_items; i++) {
5184 if (spec->gen.imux_pins[i] == 0x12) {
5185 spec->gen.cur_mux[0] = i;
5186 break;
5187 }
5188 }
Takashi Iwai3e1b0c42015-04-27 10:43:22 +02005189 break;
Gabriele Mazzotta033b0a72014-05-26 17:11:46 +02005190 }
5191}
5192
David Henningsson1f8b46c2015-05-12 14:38:15 +02005193static void alc_fixup_headset_mode_alc662(struct hda_codec *codec,
5194 const struct hda_fixup *fix, int action)
5195{
5196 struct alc_spec *spec = codec->spec;
5197
5198 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
5199 spec->parse_flags |= HDA_PINCFG_HEADSET_MIC;
5200 spec->gen.hp_mic = 1; /* Mic-in is same pin as headphone */
David Henningssonb40eda62015-05-28 09:40:23 +02005201
5202 /* Disable boost for mic-in permanently. (This code is only called
5203 from quirks that guarantee that the headphone is at NID 0x1b.) */
5204 snd_hda_codec_write(codec, 0x1b, 0, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000);
5205 snd_hda_override_wcaps(codec, 0x1b, get_wcaps(codec, 0x1b) & ~AC_WCAP_IN_AMP);
David Henningsson1f8b46c2015-05-12 14:38:15 +02005206 } else
5207 alc_fixup_headset_mode(codec, fix, action);
5208}
5209
David Henningsson73bdd592013-04-15 15:44:14 +02005210static void alc_fixup_headset_mode_alc668(struct hda_codec *codec,
5211 const struct hda_fixup *fix, int action)
5212{
5213 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
David Henningsson73bdd592013-04-15 15:44:14 +02005214 alc_write_coef_idx(codec, 0xc4, 0x8000);
Takashi Iwai98b24882014-08-18 13:47:50 +02005215 alc_update_coef_idx(codec, 0xc2, ~0xfe, 0);
David Henningsson73bdd592013-04-15 15:44:14 +02005216 snd_hda_set_pin_ctl_cache(codec, 0x18, 0);
5217 }
5218 alc_fixup_headset_mode(codec, fix, action);
5219}
5220
Chih-Chung Changbde7bc62013-08-05 16:38:42 +08005221/* Returns the nid of the external mic input pin, or 0 if it cannot be found. */
5222static int find_ext_mic_pin(struct hda_codec *codec)
5223{
5224 struct alc_spec *spec = codec->spec;
5225 struct auto_pin_cfg *cfg = &spec->gen.autocfg;
5226 hda_nid_t nid;
5227 unsigned int defcfg;
5228 int i;
5229
5230 for (i = 0; i < cfg->num_inputs; i++) {
5231 if (cfg->inputs[i].type != AUTO_PIN_MIC)
5232 continue;
5233 nid = cfg->inputs[i].pin;
5234 defcfg = snd_hda_codec_get_pincfg(codec, nid);
5235 if (snd_hda_get_input_pin_attr(defcfg) == INPUT_PIN_ATTR_INT)
5236 continue;
5237 return nid;
5238 }
5239
5240 return 0;
5241}
5242
Dylan Reid08a978d2012-11-18 22:56:40 -08005243static void alc271_hp_gate_mic_jack(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01005244 const struct hda_fixup *fix,
Dylan Reid08a978d2012-11-18 22:56:40 -08005245 int action)
5246{
5247 struct alc_spec *spec = codec->spec;
5248
Takashi Iwai0db75792013-01-23 13:57:20 +01005249 if (action == HDA_FIXUP_ACT_PROBE) {
Chih-Chung Changbde7bc62013-08-05 16:38:42 +08005250 int mic_pin = find_ext_mic_pin(codec);
Takashi Iwai35a39f92019-02-01 11:19:50 +01005251 int hp_pin = alc_get_hp_pin(spec);
Chih-Chung Changbde7bc62013-08-05 16:38:42 +08005252
5253 if (snd_BUG_ON(!mic_pin || !hp_pin))
Takashi Iwai0db75792013-01-23 13:57:20 +01005254 return;
Chih-Chung Changbde7bc62013-08-05 16:38:42 +08005255 snd_hda_jack_set_gating_jack(codec, mic_pin, hp_pin);
Takashi Iwai0db75792013-01-23 13:57:20 +01005256 }
Dylan Reid08a978d2012-11-18 22:56:40 -08005257}
David Henningsson693b6132012-06-22 19:12:10 +02005258
David Henningsson3e0d6112013-04-22 14:30:14 +02005259static void alc269_fixup_limit_int_mic_boost(struct hda_codec *codec,
5260 const struct hda_fixup *fix,
5261 int action)
5262{
5263 struct alc_spec *spec = codec->spec;
5264 struct auto_pin_cfg *cfg = &spec->gen.autocfg;
5265 int i;
5266
5267 /* The mic boosts on level 2 and 3 are too noisy
5268 on the internal mic input.
5269 Therefore limit the boost to 0 or 1. */
5270
5271 if (action != HDA_FIXUP_ACT_PROBE)
5272 return;
5273
5274 for (i = 0; i < cfg->num_inputs; i++) {
5275 hda_nid_t nid = cfg->inputs[i].pin;
5276 unsigned int defcfg;
5277 if (cfg->inputs[i].type != AUTO_PIN_MIC)
5278 continue;
5279 defcfg = snd_hda_codec_get_pincfg(codec, nid);
5280 if (snd_hda_get_input_pin_attr(defcfg) != INPUT_PIN_ATTR_INT)
5281 continue;
5282
5283 snd_hda_override_amp_caps(codec, nid, HDA_INPUT,
5284 (0x00 << AC_AMPCAP_OFFSET_SHIFT) |
5285 (0x01 << AC_AMPCAP_NUM_STEPS_SHIFT) |
5286 (0x2f << AC_AMPCAP_STEP_SIZE_SHIFT) |
5287 (0 << AC_AMPCAP_MUTE_SHIFT));
5288 }
5289}
5290
Kailang Yangcd217a62013-08-22 10:15:24 +02005291static void alc283_hp_automute_hook(struct hda_codec *codec,
Takashi Iwai1a4f69d2014-09-11 15:22:46 +02005292 struct hda_jack_callback *jack)
Kailang Yangcd217a62013-08-22 10:15:24 +02005293{
5294 struct alc_spec *spec = codec->spec;
5295 int vref;
5296
5297 msleep(200);
5298 snd_hda_gen_hp_automute(codec, jack);
5299
5300 vref = spec->gen.hp_jack_present ? PIN_VREF80 : 0;
5301
5302 msleep(600);
5303 snd_hda_codec_write(codec, 0x19, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
5304 vref);
5305}
5306
Kailang Yangcd217a62013-08-22 10:15:24 +02005307static void alc283_fixup_chromebook(struct hda_codec *codec,
5308 const struct hda_fixup *fix, int action)
5309{
5310 struct alc_spec *spec = codec->spec;
Kailang Yangcd217a62013-08-22 10:15:24 +02005311
5312 switch (action) {
5313 case HDA_FIXUP_ACT_PRE_PROBE:
Kailang Yang0202e992013-12-02 15:20:15 +08005314 snd_hda_override_wcaps(codec, 0x03, 0);
Takashi Iwaid2e92702013-10-30 07:50:53 +01005315 /* Disable AA-loopback as it causes white noise */
5316 spec->gen.mixer_nid = 0;
Kailang Yang38070212013-11-01 15:57:35 +08005317 break;
5318 case HDA_FIXUP_ACT_INIT:
Kailang Yangde9481c2014-04-07 16:41:52 +08005319 /* MIC2-VREF control */
5320 /* Set to manual mode */
Takashi Iwai98b24882014-08-18 13:47:50 +02005321 alc_update_coef_idx(codec, 0x06, 0x000c, 0);
Kailang Yang0202e992013-12-02 15:20:15 +08005322 /* Enable Line1 input control by verb */
Takashi Iwai98b24882014-08-18 13:47:50 +02005323 alc_update_coef_idx(codec, 0x1a, 0, 1 << 4);
Kailang Yang0202e992013-12-02 15:20:15 +08005324 break;
5325 }
5326}
5327
5328static void alc283_fixup_sense_combo_jack(struct hda_codec *codec,
5329 const struct hda_fixup *fix, int action)
5330{
5331 struct alc_spec *spec = codec->spec;
Kailang Yang0202e992013-12-02 15:20:15 +08005332
5333 switch (action) {
5334 case HDA_FIXUP_ACT_PRE_PROBE:
Kailang Yangcd217a62013-08-22 10:15:24 +02005335 spec->gen.hp_automute_hook = alc283_hp_automute_hook;
5336 break;
5337 case HDA_FIXUP_ACT_INIT:
5338 /* MIC2-VREF control */
5339 /* Set to manual mode */
Takashi Iwai98b24882014-08-18 13:47:50 +02005340 alc_update_coef_idx(codec, 0x06, 0x000c, 0);
Kailang Yangcd217a62013-08-22 10:15:24 +02005341 break;
5342 }
5343}
5344
Takashi Iwai7bba2152013-09-06 15:45:38 +02005345/* mute tablet speaker pin (0x14) via dock plugging in addition */
5346static void asus_tx300_automute(struct hda_codec *codec)
5347{
5348 struct alc_spec *spec = codec->spec;
5349 snd_hda_gen_update_outputs(codec);
5350 if (snd_hda_jack_detect(codec, 0x1b))
5351 spec->gen.mute_bits |= (1ULL << 0x14);
5352}
5353
5354static void alc282_fixup_asus_tx300(struct hda_codec *codec,
5355 const struct hda_fixup *fix, int action)
5356{
5357 struct alc_spec *spec = codec->spec;
Takashi Iwai7bba2152013-09-06 15:45:38 +02005358 static const struct hda_pintbl dock_pins[] = {
5359 { 0x1b, 0x21114000 }, /* dock speaker pin */
5360 {}
5361 };
Takashi Iwai7bba2152013-09-06 15:45:38 +02005362
5363 switch (action) {
5364 case HDA_FIXUP_ACT_PRE_PROBE:
Takashi Iwai1c76aa52018-06-21 16:37:54 +02005365 spec->init_amp = ALC_INIT_DEFAULT;
Takashi Iwaiae065f12018-06-19 23:00:03 +02005366 /* TX300 needs to set up GPIO2 for the speaker amp */
5367 alc_setup_gpio(codec, 0x04);
Takashi Iwai7bba2152013-09-06 15:45:38 +02005368 snd_hda_apply_pincfgs(codec, dock_pins);
5369 spec->gen.auto_mute_via_amp = 1;
5370 spec->gen.automute_hook = asus_tx300_automute;
5371 snd_hda_jack_detect_enable_callback(codec, 0x1b,
Takashi Iwai7bba2152013-09-06 15:45:38 +02005372 snd_hda_gen_hp_automute);
5373 break;
Takashi Iwai5579cd62018-06-19 22:22:41 +02005374 case HDA_FIXUP_ACT_PROBE:
5375 spec->init_amp = ALC_INIT_DEFAULT;
5376 break;
Takashi Iwai7bba2152013-09-06 15:45:38 +02005377 case HDA_FIXUP_ACT_BUILD:
5378 /* this is a bit tricky; give more sane names for the main
5379 * (tablet) speaker and the dock speaker, respectively
5380 */
Takashi Iwai56798e62017-04-11 08:10:52 +02005381 rename_ctl(codec, "Speaker Playback Switch",
5382 "Dock Speaker Playback Switch");
5383 rename_ctl(codec, "Bass Speaker Playback Switch",
5384 "Speaker Playback Switch");
Takashi Iwai7bba2152013-09-06 15:45:38 +02005385 break;
5386 }
5387}
5388
David Henningsson338cae52013-10-07 10:39:59 +02005389static void alc290_fixup_mono_speakers(struct hda_codec *codec,
5390 const struct hda_fixup *fix, int action)
5391{
David Henningsson0f4881d2013-12-20 16:08:13 +01005392 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
5393 /* DAC node 0x03 is giving mono output. We therefore want to
5394 make sure 0x14 (front speaker) and 0x15 (headphones) use the
5395 stereo DAC, while leaving 0x17 (bass speaker) for node 0x03. */
5396 hda_nid_t conn1[2] = { 0x0c };
5397 snd_hda_override_conn_list(codec, 0x14, 1, conn1);
5398 snd_hda_override_conn_list(codec, 0x15, 1, conn1);
5399 }
David Henningsson338cae52013-10-07 10:39:59 +02005400}
5401
Hui Wangdd9aa332016-08-01 10:20:32 +08005402static void alc298_fixup_speaker_volume(struct hda_codec *codec,
5403 const struct hda_fixup *fix, int action)
5404{
5405 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
5406 /* The speaker is routed to the Node 0x06 by a mistake, as a result
5407 we can't adjust the speaker's volume since this node does not has
5408 Amp-out capability. we change the speaker's route to:
5409 Node 0x02 (Audio Output) -> Node 0x0c (Audio Mixer) -> Node 0x17 (
5410 Pin Complex), since Node 0x02 has Amp-out caps, we can adjust
5411 speaker's volume now. */
5412
5413 hda_nid_t conn1[1] = { 0x0c };
5414 snd_hda_override_conn_list(codec, 0x17, 1, conn1);
5415 }
5416}
5417
Takashi Iwaie312a862018-03-06 12:14:17 +01005418/* disable DAC3 (0x06) selection on NID 0x17 as it has no volume amp control */
5419static void alc295_fixup_disable_dac3(struct hda_codec *codec,
5420 const struct hda_fixup *fix, int action)
5421{
5422 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
5423 hda_nid_t conn[2] = { 0x02, 0x03 };
5424 snd_hda_override_conn_list(codec, 0x17, 2, conn);
5425 }
5426}
5427
Keith Packard98973f22015-07-15 12:14:39 -07005428/* Hook to update amp GPIO4 for automute */
5429static void alc280_hp_gpio4_automute_hook(struct hda_codec *codec,
5430 struct hda_jack_callback *jack)
5431{
5432 struct alc_spec *spec = codec->spec;
5433
5434 snd_hda_gen_hp_automute(codec, jack);
5435 /* mute_led_polarity is set to 0, so we pass inverted value here */
5436 alc_update_gpio_led(codec, 0x10, !spec->gen.hp_jack_present);
5437}
5438
5439/* Manage GPIOs for HP EliteBook Folio 9480m.
5440 *
5441 * GPIO4 is the headphone amplifier power control
5442 * GPIO3 is the audio output mute indicator LED
5443 */
5444
5445static void alc280_fixup_hp_9480m(struct hda_codec *codec,
5446 const struct hda_fixup *fix,
5447 int action)
5448{
5449 struct alc_spec *spec = codec->spec;
Keith Packard98973f22015-07-15 12:14:39 -07005450
Takashi Iwai01e4a272018-06-19 22:47:30 +02005451 alc_fixup_hp_gpio_led(codec, action, 0x08, 0);
Keith Packard98973f22015-07-15 12:14:39 -07005452 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwai01e4a272018-06-19 22:47:30 +02005453 /* amp at GPIO4; toggled via alc280_hp_gpio4_automute_hook() */
5454 spec->gpio_mask |= 0x10;
5455 spec->gpio_dir |= 0x10;
Keith Packard98973f22015-07-15 12:14:39 -07005456 spec->gen.hp_automute_hook = alc280_hp_gpio4_automute_hook;
Keith Packard98973f22015-07-15 12:14:39 -07005457 }
5458}
5459
Takashi Iwaiae065f12018-06-19 23:00:03 +02005460static void alc275_fixup_gpio4_off(struct hda_codec *codec,
5461 const struct hda_fixup *fix,
5462 int action)
5463{
5464 struct alc_spec *spec = codec->spec;
5465
5466 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
5467 spec->gpio_mask |= 0x04;
5468 spec->gpio_dir |= 0x04;
5469 /* set data bit low */
5470 }
5471}
5472
Kailang Yangca169cc2017-04-25 16:17:40 +08005473static void alc233_alc662_fixup_lenovo_dual_codecs(struct hda_codec *codec,
5474 const struct hda_fixup *fix,
5475 int action)
5476{
5477 alc_fixup_dual_codecs(codec, fix, action);
5478 switch (action) {
5479 case HDA_FIXUP_ACT_PRE_PROBE:
5480 /* override card longname to provide a unique UCM profile */
5481 strcpy(codec->card->longname, "HDAudio-Lenovo-DualCodecs");
5482 break;
5483 case HDA_FIXUP_ACT_BUILD:
5484 /* rename Capture controls depending on the codec */
5485 rename_ctl(codec, "Capture Volume",
5486 codec->addr == 0 ?
5487 "Rear-Panel Capture Volume" :
5488 "Front-Panel Capture Volume");
5489 rename_ctl(codec, "Capture Switch",
5490 codec->addr == 0 ?
5491 "Rear-Panel Capture Switch" :
5492 "Front-Panel Capture Switch");
5493 break;
5494 }
5495}
5496
Kailang Yang92266652017-12-14 15:28:58 +08005497/* Forcibly assign NID 0x03 to HP/LO while NID 0x02 to SPK for EQ */
5498static void alc274_fixup_bind_dacs(struct hda_codec *codec,
5499 const struct hda_fixup *fix, int action)
5500{
5501 struct alc_spec *spec = codec->spec;
5502 static hda_nid_t preferred_pairs[] = {
5503 0x21, 0x03, 0x1b, 0x03, 0x16, 0x02,
5504 0
5505 };
5506
5507 if (action != HDA_FIXUP_ACT_PRE_PROBE)
5508 return;
5509
5510 spec->gen.preferred_dacs = preferred_pairs;
Kailang Yang0700d3d2019-04-26 16:13:54 +08005511 spec->gen.auto_mute_via_amp = 1;
5512 codec->power_save_node = 0;
Kailang Yang92266652017-12-14 15:28:58 +08005513}
5514
Hui Wangc4cfcf62018-11-26 14:17:16 +08005515/* The DAC of NID 0x3 will introduce click/pop noise on headphones, so invalidate it */
5516static void alc285_fixup_invalidate_dacs(struct hda_codec *codec,
5517 const struct hda_fixup *fix, int action)
5518{
5519 if (action != HDA_FIXUP_ACT_PRE_PROBE)
5520 return;
5521
5522 snd_hda_override_wcaps(codec, 0x03, 0);
5523}
5524
Kailang Yange8547472018-11-28 15:32:45 +08005525static const struct hda_jack_keymap alc_headset_btn_keymap[] = {
5526 { SND_JACK_BTN_0, KEY_PLAYPAUSE },
5527 { SND_JACK_BTN_1, KEY_VOICECOMMAND },
5528 { SND_JACK_BTN_2, KEY_VOLUMEUP },
5529 { SND_JACK_BTN_3, KEY_VOLUMEDOWN },
5530 {}
5531};
5532
5533static void alc_headset_btn_callback(struct hda_codec *codec,
5534 struct hda_jack_callback *jack)
5535{
5536 int report = 0;
5537
5538 if (jack->unsol_res & (7 << 13))
5539 report |= SND_JACK_BTN_0;
5540
5541 if (jack->unsol_res & (1 << 16 | 3 << 8))
5542 report |= SND_JACK_BTN_1;
5543
5544 /* Volume up key */
5545 if (jack->unsol_res & (7 << 23))
5546 report |= SND_JACK_BTN_2;
5547
5548 /* Volume down key */
5549 if (jack->unsol_res & (7 << 10))
5550 report |= SND_JACK_BTN_3;
5551
5552 jack->jack->button_state = report;
5553}
5554
Kailang Yang8983eb62019-04-03 15:31:49 +08005555static void alc_fixup_headset_jack(struct hda_codec *codec,
Kailang Yange8547472018-11-28 15:32:45 +08005556 const struct hda_fixup *fix, int action)
5557{
5558
5559 switch (action) {
5560 case HDA_FIXUP_ACT_PRE_PROBE:
5561 snd_hda_jack_detect_enable_callback(codec, 0x55,
5562 alc_headset_btn_callback);
5563 snd_hda_jack_add_kctl(codec, 0x55, "Headset Jack", false,
5564 SND_JACK_HEADSET, alc_headset_btn_keymap);
5565 break;
5566 case HDA_FIXUP_ACT_INIT:
5567 switch (codec->core.vendor_id) {
5568 case 0x10ec0225:
5569 case 0x10ec0295:
5570 case 0x10ec0299:
5571 alc_write_coef_idx(codec, 0x48, 0xd011);
5572 alc_update_coef_idx(codec, 0x49, 0x007f, 0x0045);
5573 alc_update_coef_idx(codec, 0x44, 0x007f << 8, 0x0045 << 8);
5574 break;
5575 case 0x10ec0236:
5576 case 0x10ec0256:
5577 alc_write_coef_idx(codec, 0x48, 0xd011);
5578 alc_update_coef_idx(codec, 0x49, 0x007f, 0x0045);
5579 break;
5580 }
5581 break;
5582 }
5583}
5584
Kailang Yang8983eb62019-04-03 15:31:49 +08005585static void alc295_fixup_chromebook(struct hda_codec *codec,
5586 const struct hda_fixup *fix, int action)
5587{
Kailang Yangd3ba58b2019-05-06 15:09:42 +08005588 struct alc_spec *spec = codec->spec;
5589
Kailang Yang8983eb62019-04-03 15:31:49 +08005590 switch (action) {
Kailang Yangd3ba58b2019-05-06 15:09:42 +08005591 case HDA_FIXUP_ACT_PRE_PROBE:
5592 spec->ultra_low_power = true;
5593 break;
Kailang Yang8983eb62019-04-03 15:31:49 +08005594 case HDA_FIXUP_ACT_INIT:
5595 switch (codec->core.vendor_id) {
5596 case 0x10ec0295:
5597 alc_update_coef_idx(codec, 0x4a, 0x8000, 1 << 15); /* Reset HP JD */
5598 alc_update_coef_idx(codec, 0x4a, 0x8000, 0 << 15);
5599 break;
5600 case 0x10ec0236:
5601 alc_update_coef_idx(codec, 0x1b, 0x8000, 1 << 15); /* Reset HP JD */
5602 alc_update_coef_idx(codec, 0x1b, 0x8000, 0 << 15);
5603 break;
5604 }
5605 break;
5606 }
5607}
5608
Kailang Yangd1dd4212019-01-09 17:05:24 +08005609static void alc_fixup_disable_mic_vref(struct hda_codec *codec,
5610 const struct hda_fixup *fix, int action)
5611{
5612 if (action == HDA_FIXUP_ACT_PRE_PROBE)
5613 snd_hda_codec_set_pin_target(codec, 0x19, PIN_VREFHIZ);
5614}
5615
Takashi Iwaib317b032014-01-08 11:44:21 +01005616/* for hda_fixup_thinkpad_acpi() */
5617#include "thinkpad_helper.c"
David Henningssonb67ae3f2013-11-05 13:11:37 +01005618
Takashi Iwaid5a6cab2018-06-13 12:43:10 +02005619static void alc_fixup_thinkpad_acpi(struct hda_codec *codec,
5620 const struct hda_fixup *fix, int action)
5621{
5622 alc_fixup_no_shutup(codec, fix, action); /* reduce click noise */
5623 hda_fixup_thinkpad_acpi(codec, fix, action);
5624}
5625
Tom Bridenbbf8ff62018-05-29 17:34:20 +01005626/* for alc295_fixup_hp_top_speakers */
5627#include "hp_x360_helper.c"
5628
Takashi Iwai1d045db2011-07-07 18:23:21 +02005629enum {
5630 ALC269_FIXUP_SONY_VAIO,
5631 ALC275_FIXUP_SONY_VAIO_GPIO2,
5632 ALC269_FIXUP_DELL_M101Z,
5633 ALC269_FIXUP_SKU_IGNORE,
5634 ALC269_FIXUP_ASUS_G73JW,
5635 ALC269_FIXUP_LENOVO_EAPD,
5636 ALC275_FIXUP_SONY_HWEQ,
Takashi Iwaie9bd7d52014-05-21 11:06:49 +02005637 ALC275_FIXUP_SONY_DISABLE_AAMIX,
Takashi Iwai1d045db2011-07-07 18:23:21 +02005638 ALC271_FIXUP_DMIC,
Takashi Iwai017f2a12011-07-09 14:42:25 +02005639 ALC269_FIXUP_PCM_44K,
Takashi Iwaiadabb3e2011-08-03 07:48:37 +02005640 ALC269_FIXUP_STEREO_DMIC,
David Henningsson7c478f02013-10-11 10:18:46 +02005641 ALC269_FIXUP_HEADSET_MIC,
Takashi Iwai24519912011-08-16 15:08:49 +02005642 ALC269_FIXUP_QUANTA_MUTE,
5643 ALC269_FIXUP_LIFEBOOK,
David Henningsson2041d562014-06-13 11:15:44 +02005644 ALC269_FIXUP_LIFEBOOK_EXTMIC,
Takashi Iwaicc7016a2015-04-08 20:47:55 +02005645 ALC269_FIXUP_LIFEBOOK_HP_PIN,
Takashi Iwai4df3fd12015-06-29 08:38:02 +02005646 ALC269_FIXUP_LIFEBOOK_NO_HP_TO_LINEOUT,
Jan-Marek Glogowskifdcc9682018-02-14 11:29:15 +01005647 ALC255_FIXUP_LIFEBOOK_U7x7_HEADSET_MIC,
Takashi Iwaia4297b52011-08-23 18:40:12 +02005648 ALC269_FIXUP_AMIC,
5649 ALC269_FIXUP_DMIC,
5650 ALC269VB_FIXUP_AMIC,
5651 ALC269VB_FIXUP_DMIC,
Takashi Iwai08fb0d02013-01-10 17:33:58 +01005652 ALC269_FIXUP_HP_MUTE_LED,
David Henningssond06ac142013-02-18 11:41:55 +01005653 ALC269_FIXUP_HP_MUTE_LED_MIC1,
Takashi Iwai08fb0d02013-01-10 17:33:58 +01005654 ALC269_FIXUP_HP_MUTE_LED_MIC2,
Tom Briden7f783bd2017-03-25 10:12:01 +00005655 ALC269_FIXUP_HP_MUTE_LED_MIC3,
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01005656 ALC269_FIXUP_HP_GPIO_LED,
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08005657 ALC269_FIXUP_HP_GPIO_MIC1_LED,
5658 ALC269_FIXUP_HP_LINE1_MIC1_LED,
David Henningsson693b6132012-06-22 19:12:10 +02005659 ALC269_FIXUP_INV_DMIC,
David Henningsson108cc102012-07-20 10:37:25 +02005660 ALC269_FIXUP_LENOVO_DOCK,
Takashi Iwai9b745ab2014-03-07 08:37:19 +01005661 ALC269_FIXUP_NO_SHUTUP,
David Henningsson88cfcf82013-10-11 10:18:45 +02005662 ALC286_FIXUP_SONY_MIC_NO_PRESENCE,
David Henningsson108cc102012-07-20 10:37:25 +02005663 ALC269_FIXUP_PINCFG_NO_HP_TO_LINEOUT,
David Henningsson73bdd592013-04-15 15:44:14 +02005664 ALC269_FIXUP_DELL1_MIC_NO_PRESENCE,
5665 ALC269_FIXUP_DELL2_MIC_NO_PRESENCE,
David Henningsson338cae52013-10-07 10:39:59 +02005666 ALC269_FIXUP_DELL3_MIC_NO_PRESENCE,
Kailang Yangfcc6c872017-06-29 15:21:27 +08005667 ALC269_FIXUP_DELL4_MIC_NO_PRESENCE,
David Henningsson73bdd592013-04-15 15:44:14 +02005668 ALC269_FIXUP_HEADSET_MODE,
5669 ALC269_FIXUP_HEADSET_MODE_NO_HP_MIC,
Takashi Iwai78197172015-06-27 10:21:13 +02005670 ALC269_FIXUP_ASPIRE_HEADSET_MIC,
David Henningssond240d1d2013-04-15 12:50:02 +02005671 ALC269_FIXUP_ASUS_X101_FUNC,
5672 ALC269_FIXUP_ASUS_X101_VERB,
5673 ALC269_FIXUP_ASUS_X101,
Dylan Reid08a978d2012-11-18 22:56:40 -08005674 ALC271_FIXUP_AMIC_MIC2,
5675 ALC271_FIXUP_HP_GATE_MIC_JACK,
Oleksij Rempelb1e89722013-12-04 20:50:53 +01005676 ALC271_FIXUP_HP_GATE_MIC_JACK_E1_572,
Dylan Reid42397002013-04-05 14:58:22 -07005677 ALC269_FIXUP_ACER_AC700,
David Henningsson3e0d6112013-04-22 14:30:14 +02005678 ALC269_FIXUP_LIMIT_INT_MIC_BOOST,
Oleksij Rempel2cede302013-11-27 17:12:03 +01005679 ALC269VB_FIXUP_ASUS_ZENBOOK,
Takashi Iwai23870832013-11-29 14:13:12 +01005680 ALC269VB_FIXUP_ASUS_ZENBOOK_UX31A,
David Henningsson8e35cd42013-11-06 11:20:01 +01005681 ALC269_FIXUP_LIMIT_INT_MIC_BOOST_MUTE_LED,
Anisse Astier02b504d2013-06-03 11:53:10 +02005682 ALC269VB_FIXUP_ORDISSIMO_EVE2,
Kailang Yangcd217a62013-08-22 10:15:24 +02005683 ALC283_FIXUP_CHROME_BOOK,
Kailang Yang0202e992013-12-02 15:20:15 +08005684 ALC283_FIXUP_SENSE_COMBO_JACK,
Takashi Iwai7bba2152013-09-06 15:45:38 +02005685 ALC282_FIXUP_ASUS_TX300,
Kailang Yang1bb3e062013-09-27 13:10:25 +02005686 ALC283_FIXUP_INT_MIC,
David Henningsson338cae52013-10-07 10:39:59 +02005687 ALC290_FIXUP_MONO_SPEAKERS,
David Henningsson0f4881d2013-12-20 16:08:13 +01005688 ALC290_FIXUP_MONO_SPEAKERS_HSJACK,
5689 ALC290_FIXUP_SUBWOOFER,
5690 ALC290_FIXUP_SUBWOOFER_HSJACK,
David Henningssonb67ae3f2013-11-05 13:11:37 +01005691 ALC269_FIXUP_THINKPAD_ACPI,
David Henningsson56f27012016-01-11 09:33:14 +01005692 ALC269_FIXUP_DMIC_THINKPAD_ACPI,
Chris Chiu5824ce82017-02-28 14:17:11 -06005693 ALC255_FIXUP_ACER_MIC_NO_PRESENCE,
Chris Chiu615966a2017-02-28 14:17:12 -06005694 ALC255_FIXUP_ASUS_MIC_NO_PRESENCE,
Kailang Yang9a22a8f52013-11-08 15:54:49 +08005695 ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Kailang Yang31278992014-03-03 15:27:22 +08005696 ALC255_FIXUP_DELL2_MIC_NO_PRESENCE,
Kailang Yang9a22a8f52013-11-08 15:54:49 +08005697 ALC255_FIXUP_HEADSET_MODE,
Kailang Yang31278992014-03-03 15:27:22 +08005698 ALC255_FIXUP_HEADSET_MODE_NO_HP_MIC,
Kailang Yanga22aa262014-04-23 17:34:28 +08005699 ALC293_FIXUP_DELL1_MIC_NO_PRESENCE,
Takashi Iwai1c37c222014-05-06 17:34:42 +02005700 ALC292_FIXUP_TPT440_DOCK,
Takashi Iwai9a811232015-12-09 15:17:43 +01005701 ALC292_FIXUP_TPT440,
Anisse Astierabaa22742016-08-24 09:14:13 +02005702 ALC283_FIXUP_HEADSET_MIC,
Takashi Iwaib3802782018-11-26 17:47:46 +01005703 ALC255_FIXUP_MIC_MUTE_LED,
Takashi Iwai1a22e772014-08-27 08:19:05 +02005704 ALC282_FIXUP_ASPIRE_V5_PINS,
David Henningsson7a5255f2014-10-30 08:26:01 +01005705 ALC280_FIXUP_HP_GPIO4,
Kailang Yangeaa8e5e2014-11-21 15:49:11 +08005706 ALC286_FIXUP_HP_GPIO_LED,
David Henningsson33f4acd2015-01-07 15:50:13 +01005707 ALC280_FIXUP_HP_GPIO2_MIC_HOTKEY,
TienFu Chenb4b33f92015-01-20 15:06:21 +01005708 ALC280_FIXUP_HP_DOCK_PINS,
Jaroslav Kysela04d54662017-03-09 13:29:13 +01005709 ALC269_FIXUP_HP_DOCK_GPIO_MIC1_LED,
Keith Packard98973f22015-07-15 12:14:39 -07005710 ALC280_FIXUP_HP_9480M,
Kailang Yange1e62b92015-04-08 16:01:22 +08005711 ALC288_FIXUP_DELL_HEADSET_MODE,
5712 ALC288_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wang831bfdf92015-06-26 12:35:17 +08005713 ALC288_FIXUP_DELL_XPS_13,
5714 ALC288_FIXUP_DISABLE_AAMIX,
Takashi Iwai8b99aba2015-06-15 11:59:32 +02005715 ALC292_FIXUP_DELL_E7X,
5716 ALC292_FIXUP_DISABLE_AAMIX,
David Henningssonc04017e2015-12-15 14:44:03 +01005717 ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK,
Kailang Yang977e6272015-05-18 15:31:20 +08005718 ALC298_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wang2f726ae2017-03-31 10:31:40 +08005719 ALC298_FIXUP_DELL_AIO_MIC_NO_PRESENCE,
Kailang Yang6ed11312015-10-26 15:37:39 +08005720 ALC275_FIXUP_DELL_XPS,
Hui Wang8c697292015-11-24 11:08:18 +08005721 ALC256_FIXUP_DELL_XPS_13_HEADPHONE_NOISE,
Hui Wang23adc192015-12-08 12:27:18 +08005722 ALC293_FIXUP_LENOVO_SPK_NOISE,
Kailang3694cb22015-12-28 11:35:24 +08005723 ALC233_FIXUP_LENOVO_LINE2_MIC_HOTKEY,
Kai-Heng Feng3b43b712016-02-25 15:19:38 +08005724 ALC255_FIXUP_DELL_SPK_NOISE,
Kailang Yangd1dd4212019-01-09 17:05:24 +08005725 ALC225_FIXUP_DISABLE_MIC_VREF,
David Henningsson2ae95572016-02-25 09:37:05 +01005726 ALC225_FIXUP_DELL1_MIC_NO_PRESENCE,
Takashi Iwaie312a862018-03-06 12:14:17 +01005727 ALC295_FIXUP_DISABLE_DAC3,
Takashi Iwaif8839822016-02-25 14:31:59 +01005728 ALC280_FIXUP_HP_HEADSET_MIC,
Hui Wange549d192016-04-01 11:00:15 +08005729 ALC221_FIXUP_HP_FRONT_MIC,
Sven Eckelmannc636b952016-04-11 16:55:26 +02005730 ALC292_FIXUP_TPT460,
Hui Wangdd9aa332016-08-01 10:20:32 +08005731 ALC298_FIXUP_SPK_VOLUME,
Kai-Heng Fengfd06c772016-08-30 15:36:34 +08005732 ALC256_FIXUP_DELL_INSPIRON_7559_SUBWOOFER,
Gabriele Mazzotta823ff162016-12-24 19:50:01 +01005733 ALC269_FIXUP_ATIV_BOOK_8,
Kailang Yang9eb5d0e2017-01-17 15:40:25 +08005734 ALC221_FIXUP_HP_MIC_NO_PRESENCE,
Chris Chiuc1732ed2017-02-28 14:17:13 -06005735 ALC256_FIXUP_ASUS_HEADSET_MODE,
5736 ALC256_FIXUP_ASUS_MIC,
Chris Chiueeed4cd2017-02-28 14:17:15 -06005737 ALC256_FIXUP_ASUS_AIO_GPIO2,
Chris Chiu216d7ae2017-02-28 14:17:14 -06005738 ALC233_FIXUP_ASUS_MIC_NO_PRESENCE,
5739 ALC233_FIXUP_EAPD_COEF_AND_MIC_NO_PRESENCE,
Kailang Yangca169cc2017-04-25 16:17:40 +08005740 ALC233_FIXUP_LENOVO_MULTI_CODECS,
Jian-Hong Panea5c7eb2019-04-01 11:25:05 +08005741 ALC233_FIXUP_ACER_HEADSET_MIC,
Hui Wangf33f79f2017-07-07 12:08:29 +08005742 ALC294_FIXUP_LENOVO_MIC_LOCATION,
Kailang Yang5f364132017-07-25 16:28:16 +08005743 ALC225_FIXUP_DELL_WYSE_MIC_NO_PRESENCE,
PeiSen Houb84e8432017-09-01 15:11:56 +08005744 ALC700_FIXUP_INTEL_REFERENCE,
Kailang Yang92266652017-12-14 15:28:58 +08005745 ALC274_FIXUP_DELL_BIND_DACS,
5746 ALC274_FIXUP_DELL_AIO_LINEOUT_VERB,
Kailang Yang61fcf8e2018-02-02 15:26:46 +08005747 ALC298_FIXUP_TPT470_DOCK,
Kailang Yangae104a22018-02-05 16:07:20 +08005748 ALC255_FIXUP_DUMMY_LINEOUT_VERB,
Kailang Yangf0ba9d62018-03-16 11:46:08 +08005749 ALC255_FIXUP_DELL_HEADSET_MIC,
Ayman Bagabas8ac51bb2018-12-12 18:07:57 -05005750 ALC256_FIXUP_HUAWEI_MBXP_PINS,
Tom Bridenbbf8ff62018-05-29 17:34:20 +01005751 ALC295_FIXUP_HP_X360,
Kailang Yang8a328ac2018-08-21 16:54:41 +08005752 ALC221_FIXUP_HP_HEADSET_MIC,
Hui Wangc4cfcf62018-11-26 14:17:16 +08005753 ALC285_FIXUP_LENOVO_HEADPHONE_NOISE,
Girija Kumar Kasinadhunie8ed64b2018-11-26 13:40:46 -05005754 ALC295_FIXUP_HP_AUTO_MUTE,
Chris Chiu33aaebd2018-12-05 14:48:53 +08005755 ALC286_FIXUP_ACER_AIO_MIC_NO_PRESENCE,
Chris Chiud8ae4582018-12-07 17:17:11 +08005756 ALC294_FIXUP_ASUS_MIC,
Jian-Hong Pan4e051102018-12-07 17:17:12 +08005757 ALC294_FIXUP_ASUS_HEADSET_MIC,
5758 ALC294_FIXUP_ASUS_SPK,
Jeremy Soller89e3a562019-01-30 16:12:31 -07005759 ALC293_FIXUP_SYSTEM76_MIC_NO_PRESENCE,
Hui Wangc8c6ee62019-02-14 11:41:33 +08005760 ALC285_FIXUP_LENOVO_PC_BEEP_IN_NOISE,
Jian-Hong Pancbc05fd2019-03-13 17:33:24 +08005761 ALC255_FIXUP_ACER_HEADSET_MIC,
Kailang Yang10f5b1b2019-02-21 16:10:22 +08005762 ALC295_FIXUP_CHROME_BOOK,
Kailang Yang8983eb62019-04-03 15:31:49 +08005763 ALC225_FIXUP_HEADSET_JACK,
Kailang Yang136824e2019-03-14 16:22:45 +08005764 ALC225_FIXUP_DELL_WYSE_AIO_MIC_NO_PRESENCE,
5765 ALC225_FIXUP_WYSE_AUTO_MUTE,
5766 ALC225_FIXUP_WYSE_DISABLE_MIC_VREF,
Jian-Hong Pan667a8f72019-03-15 17:51:09 +08005767 ALC286_FIXUP_ACER_AIO_HEADSET_MIC,
Jian-Hong Pane1037352019-03-22 11:37:18 +08005768 ALC256_FIXUP_ASUS_MIC_NO_PRESENCE,
Bernhard Rosenkraenzere2a829b2019-03-05 00:38:19 +01005769 ALC299_FIXUP_PREDATOR_SPK,
Takashi Iwai1d045db2011-07-07 18:23:21 +02005770};
5771
Takashi Iwai1727a772013-01-10 09:52:52 +01005772static const struct hda_fixup alc269_fixups[] = {
Takashi Iwai1d045db2011-07-07 18:23:21 +02005773 [ALC269_FIXUP_SONY_VAIO] = {
Takashi Iwaifd108212013-01-10 10:18:14 +01005774 .type = HDA_FIXUP_PINCTLS,
5775 .v.pins = (const struct hda_pintbl[]) {
5776 {0x19, PIN_VREFGRD},
Takashi Iwai1d045db2011-07-07 18:23:21 +02005777 {}
5778 }
5779 },
5780 [ALC275_FIXUP_SONY_VAIO_GPIO2] = {
Takashi Iwaiae065f12018-06-19 23:00:03 +02005781 .type = HDA_FIXUP_FUNC,
5782 .v.func = alc275_fixup_gpio4_off,
Takashi Iwai1d045db2011-07-07 18:23:21 +02005783 .chained = true,
5784 .chain_id = ALC269_FIXUP_SONY_VAIO
5785 },
5786 [ALC269_FIXUP_DELL_M101Z] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005787 .type = HDA_FIXUP_VERBS,
Takashi Iwai1d045db2011-07-07 18:23:21 +02005788 .v.verbs = (const struct hda_verb[]) {
5789 /* Enables internal speaker */
5790 {0x20, AC_VERB_SET_COEF_INDEX, 13},
5791 {0x20, AC_VERB_SET_PROC_COEF, 0x4040},
5792 {}
5793 }
5794 },
5795 [ALC269_FIXUP_SKU_IGNORE] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005796 .type = HDA_FIXUP_FUNC,
Takashi Iwai23d30f22012-05-07 17:17:32 +02005797 .v.func = alc_fixup_sku_ignore,
Takashi Iwai1d045db2011-07-07 18:23:21 +02005798 },
5799 [ALC269_FIXUP_ASUS_G73JW] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005800 .type = HDA_FIXUP_PINS,
5801 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02005802 { 0x17, 0x99130111 }, /* subwoofer */
5803 { }
5804 }
5805 },
5806 [ALC269_FIXUP_LENOVO_EAPD] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005807 .type = HDA_FIXUP_VERBS,
Takashi Iwai1d045db2011-07-07 18:23:21 +02005808 .v.verbs = (const struct hda_verb[]) {
5809 {0x14, AC_VERB_SET_EAPD_BTLENABLE, 0},
5810 {}
5811 }
5812 },
5813 [ALC275_FIXUP_SONY_HWEQ] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005814 .type = HDA_FIXUP_FUNC,
Takashi Iwai1d045db2011-07-07 18:23:21 +02005815 .v.func = alc269_fixup_hweq,
5816 .chained = true,
5817 .chain_id = ALC275_FIXUP_SONY_VAIO_GPIO2
5818 },
Takashi Iwaie9bd7d52014-05-21 11:06:49 +02005819 [ALC275_FIXUP_SONY_DISABLE_AAMIX] = {
5820 .type = HDA_FIXUP_FUNC,
5821 .v.func = alc_fixup_disable_aamix,
5822 .chained = true,
5823 .chain_id = ALC269_FIXUP_SONY_VAIO
5824 },
Takashi Iwai1d045db2011-07-07 18:23:21 +02005825 [ALC271_FIXUP_DMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005826 .type = HDA_FIXUP_FUNC,
Takashi Iwai1d045db2011-07-07 18:23:21 +02005827 .v.func = alc271_fixup_dmic,
5828 },
Takashi Iwai017f2a12011-07-09 14:42:25 +02005829 [ALC269_FIXUP_PCM_44K] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005830 .type = HDA_FIXUP_FUNC,
Takashi Iwai017f2a12011-07-09 14:42:25 +02005831 .v.func = alc269_fixup_pcm_44k,
David Henningsson012e7eb2012-08-08 08:43:37 +02005832 .chained = true,
5833 .chain_id = ALC269_FIXUP_QUANTA_MUTE
Takashi Iwai017f2a12011-07-09 14:42:25 +02005834 },
Takashi Iwaiadabb3e2011-08-03 07:48:37 +02005835 [ALC269_FIXUP_STEREO_DMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005836 .type = HDA_FIXUP_FUNC,
Takashi Iwaiadabb3e2011-08-03 07:48:37 +02005837 .v.func = alc269_fixup_stereo_dmic,
5838 },
David Henningsson7c478f02013-10-11 10:18:46 +02005839 [ALC269_FIXUP_HEADSET_MIC] = {
5840 .type = HDA_FIXUP_FUNC,
5841 .v.func = alc269_fixup_headset_mic,
5842 },
Takashi Iwai24519912011-08-16 15:08:49 +02005843 [ALC269_FIXUP_QUANTA_MUTE] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005844 .type = HDA_FIXUP_FUNC,
Takashi Iwai24519912011-08-16 15:08:49 +02005845 .v.func = alc269_fixup_quanta_mute,
5846 },
5847 [ALC269_FIXUP_LIFEBOOK] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005848 .type = HDA_FIXUP_PINS,
5849 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai24519912011-08-16 15:08:49 +02005850 { 0x1a, 0x2101103f }, /* dock line-out */
5851 { 0x1b, 0x23a11040 }, /* dock mic-in */
5852 { }
5853 },
5854 .chained = true,
5855 .chain_id = ALC269_FIXUP_QUANTA_MUTE
5856 },
David Henningsson2041d562014-06-13 11:15:44 +02005857 [ALC269_FIXUP_LIFEBOOK_EXTMIC] = {
5858 .type = HDA_FIXUP_PINS,
5859 .v.pins = (const struct hda_pintbl[]) {
5860 { 0x19, 0x01a1903c }, /* headset mic, with jack detect */
5861 { }
5862 },
5863 },
Takashi Iwaicc7016a2015-04-08 20:47:55 +02005864 [ALC269_FIXUP_LIFEBOOK_HP_PIN] = {
5865 .type = HDA_FIXUP_PINS,
5866 .v.pins = (const struct hda_pintbl[]) {
5867 { 0x21, 0x0221102f }, /* HP out */
5868 { }
5869 },
5870 },
Takashi Iwai4df3fd12015-06-29 08:38:02 +02005871 [ALC269_FIXUP_LIFEBOOK_NO_HP_TO_LINEOUT] = {
5872 .type = HDA_FIXUP_FUNC,
5873 .v.func = alc269_fixup_pincfg_no_hp_to_lineout,
5874 },
Jan-Marek Glogowskifdcc9682018-02-14 11:29:15 +01005875 [ALC255_FIXUP_LIFEBOOK_U7x7_HEADSET_MIC] = {
5876 .type = HDA_FIXUP_FUNC,
5877 .v.func = alc269_fixup_pincfg_U7x7_headset_mic,
5878 },
Takashi Iwaia4297b52011-08-23 18:40:12 +02005879 [ALC269_FIXUP_AMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005880 .type = HDA_FIXUP_PINS,
5881 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaia4297b52011-08-23 18:40:12 +02005882 { 0x14, 0x99130110 }, /* speaker */
5883 { 0x15, 0x0121401f }, /* HP out */
5884 { 0x18, 0x01a19c20 }, /* mic */
5885 { 0x19, 0x99a3092f }, /* int-mic */
5886 { }
5887 },
5888 },
5889 [ALC269_FIXUP_DMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005890 .type = HDA_FIXUP_PINS,
5891 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaia4297b52011-08-23 18:40:12 +02005892 { 0x12, 0x99a3092f }, /* int-mic */
5893 { 0x14, 0x99130110 }, /* speaker */
5894 { 0x15, 0x0121401f }, /* HP out */
5895 { 0x18, 0x01a19c20 }, /* mic */
5896 { }
5897 },
5898 },
5899 [ALC269VB_FIXUP_AMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005900 .type = HDA_FIXUP_PINS,
5901 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaia4297b52011-08-23 18:40:12 +02005902 { 0x14, 0x99130110 }, /* speaker */
5903 { 0x18, 0x01a19c20 }, /* mic */
5904 { 0x19, 0x99a3092f }, /* int-mic */
5905 { 0x21, 0x0121401f }, /* HP out */
5906 { }
5907 },
5908 },
David Henningsson2267ea92012-01-03 08:45:56 +01005909 [ALC269VB_FIXUP_DMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005910 .type = HDA_FIXUP_PINS,
5911 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaia4297b52011-08-23 18:40:12 +02005912 { 0x12, 0x99a3092f }, /* int-mic */
5913 { 0x14, 0x99130110 }, /* speaker */
5914 { 0x18, 0x01a19c20 }, /* mic */
5915 { 0x21, 0x0121401f }, /* HP out */
5916 { }
5917 },
5918 },
Takashi Iwai08fb0d02013-01-10 17:33:58 +01005919 [ALC269_FIXUP_HP_MUTE_LED] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005920 .type = HDA_FIXUP_FUNC,
Takashi Iwai08fb0d02013-01-10 17:33:58 +01005921 .v.func = alc269_fixup_hp_mute_led,
David Henningsson6d3cd5d2013-01-07 12:03:47 +01005922 },
David Henningssond06ac142013-02-18 11:41:55 +01005923 [ALC269_FIXUP_HP_MUTE_LED_MIC1] = {
5924 .type = HDA_FIXUP_FUNC,
5925 .v.func = alc269_fixup_hp_mute_led_mic1,
5926 },
Takashi Iwai08fb0d02013-01-10 17:33:58 +01005927 [ALC269_FIXUP_HP_MUTE_LED_MIC2] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005928 .type = HDA_FIXUP_FUNC,
Takashi Iwai08fb0d02013-01-10 17:33:58 +01005929 .v.func = alc269_fixup_hp_mute_led_mic2,
Takashi Iwai420b0fe2012-03-12 12:35:27 +01005930 },
Tom Briden7f783bd2017-03-25 10:12:01 +00005931 [ALC269_FIXUP_HP_MUTE_LED_MIC3] = {
5932 .type = HDA_FIXUP_FUNC,
5933 .v.func = alc269_fixup_hp_mute_led_mic3,
Girija Kumar Kasinadhunie8ed64b2018-11-26 13:40:46 -05005934 .chained = true,
5935 .chain_id = ALC295_FIXUP_HP_AUTO_MUTE
Tom Briden7f783bd2017-03-25 10:12:01 +00005936 },
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01005937 [ALC269_FIXUP_HP_GPIO_LED] = {
5938 .type = HDA_FIXUP_FUNC,
5939 .v.func = alc269_fixup_hp_gpio_led,
5940 },
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08005941 [ALC269_FIXUP_HP_GPIO_MIC1_LED] = {
5942 .type = HDA_FIXUP_FUNC,
5943 .v.func = alc269_fixup_hp_gpio_mic1_led,
5944 },
5945 [ALC269_FIXUP_HP_LINE1_MIC1_LED] = {
5946 .type = HDA_FIXUP_FUNC,
5947 .v.func = alc269_fixup_hp_line1_mic1_led,
5948 },
David Henningsson693b6132012-06-22 19:12:10 +02005949 [ALC269_FIXUP_INV_DMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005950 .type = HDA_FIXUP_FUNC,
David Henningsson9d36a7d2014-10-07 10:18:42 +02005951 .v.func = alc_fixup_inv_dmic,
David Henningsson693b6132012-06-22 19:12:10 +02005952 },
Takashi Iwai9b745ab2014-03-07 08:37:19 +01005953 [ALC269_FIXUP_NO_SHUTUP] = {
5954 .type = HDA_FIXUP_FUNC,
5955 .v.func = alc_fixup_no_shutup,
5956 },
David Henningsson108cc102012-07-20 10:37:25 +02005957 [ALC269_FIXUP_LENOVO_DOCK] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005958 .type = HDA_FIXUP_PINS,
5959 .v.pins = (const struct hda_pintbl[]) {
David Henningsson108cc102012-07-20 10:37:25 +02005960 { 0x19, 0x23a11040 }, /* dock mic */
5961 { 0x1b, 0x2121103f }, /* dock headphone */
5962 { }
5963 },
5964 .chained = true,
5965 .chain_id = ALC269_FIXUP_PINCFG_NO_HP_TO_LINEOUT
5966 },
5967 [ALC269_FIXUP_PINCFG_NO_HP_TO_LINEOUT] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01005968 .type = HDA_FIXUP_FUNC,
David Henningsson108cc102012-07-20 10:37:25 +02005969 .v.func = alc269_fixup_pincfg_no_hp_to_lineout,
David Henningsson52129002013-11-19 10:25:53 +01005970 .chained = true,
5971 .chain_id = ALC269_FIXUP_THINKPAD_ACPI,
David Henningsson108cc102012-07-20 10:37:25 +02005972 },
David Henningsson73bdd592013-04-15 15:44:14 +02005973 [ALC269_FIXUP_DELL1_MIC_NO_PRESENCE] = {
5974 .type = HDA_FIXUP_PINS,
5975 .v.pins = (const struct hda_pintbl[]) {
5976 { 0x19, 0x01a1913c }, /* use as headset mic, without its own jack detect */
5977 { 0x1a, 0x01a1913d }, /* use as headphone mic, without its own jack detect */
5978 { }
5979 },
5980 .chained = true,
5981 .chain_id = ALC269_FIXUP_HEADSET_MODE
5982 },
5983 [ALC269_FIXUP_DELL2_MIC_NO_PRESENCE] = {
5984 .type = HDA_FIXUP_PINS,
5985 .v.pins = (const struct hda_pintbl[]) {
5986 { 0x16, 0x21014020 }, /* dock line out */
5987 { 0x19, 0x21a19030 }, /* dock mic */
5988 { 0x1a, 0x01a1913c }, /* use as headset mic, without its own jack detect */
5989 { }
5990 },
5991 .chained = true,
5992 .chain_id = ALC269_FIXUP_HEADSET_MODE_NO_HP_MIC
5993 },
David Henningsson338cae52013-10-07 10:39:59 +02005994 [ALC269_FIXUP_DELL3_MIC_NO_PRESENCE] = {
5995 .type = HDA_FIXUP_PINS,
5996 .v.pins = (const struct hda_pintbl[]) {
5997 { 0x1a, 0x01a1913c }, /* use as headset mic, without its own jack detect */
5998 { }
5999 },
6000 .chained = true,
6001 .chain_id = ALC269_FIXUP_HEADSET_MODE_NO_HP_MIC
6002 },
Kailang Yangfcc6c872017-06-29 15:21:27 +08006003 [ALC269_FIXUP_DELL4_MIC_NO_PRESENCE] = {
6004 .type = HDA_FIXUP_PINS,
6005 .v.pins = (const struct hda_pintbl[]) {
6006 { 0x19, 0x01a1913c }, /* use as headset mic, without its own jack detect */
6007 { 0x1b, 0x01a1913d }, /* use as headphone mic, without its own jack detect */
6008 { }
6009 },
6010 .chained = true,
6011 .chain_id = ALC269_FIXUP_HEADSET_MODE
6012 },
David Henningsson73bdd592013-04-15 15:44:14 +02006013 [ALC269_FIXUP_HEADSET_MODE] = {
6014 .type = HDA_FIXUP_FUNC,
6015 .v.func = alc_fixup_headset_mode,
Hui Wang6676f302014-11-18 17:57:41 +08006016 .chained = true,
Takashi Iwaib3802782018-11-26 17:47:46 +01006017 .chain_id = ALC255_FIXUP_MIC_MUTE_LED
David Henningsson73bdd592013-04-15 15:44:14 +02006018 },
6019 [ALC269_FIXUP_HEADSET_MODE_NO_HP_MIC] = {
6020 .type = HDA_FIXUP_FUNC,
6021 .v.func = alc_fixup_headset_mode_no_hp_mic,
6022 },
Takashi Iwai78197172015-06-27 10:21:13 +02006023 [ALC269_FIXUP_ASPIRE_HEADSET_MIC] = {
6024 .type = HDA_FIXUP_PINS,
6025 .v.pins = (const struct hda_pintbl[]) {
6026 { 0x19, 0x01a1913c }, /* headset mic w/o jack detect */
6027 { }
6028 },
6029 .chained = true,
6030 .chain_id = ALC269_FIXUP_HEADSET_MODE,
6031 },
David Henningsson88cfcf82013-10-11 10:18:45 +02006032 [ALC286_FIXUP_SONY_MIC_NO_PRESENCE] = {
6033 .type = HDA_FIXUP_PINS,
6034 .v.pins = (const struct hda_pintbl[]) {
6035 { 0x18, 0x01a1913c }, /* use as headset mic, without its own jack detect */
6036 { }
6037 },
David Henningssonfbc78ad2013-10-11 13:46:04 +02006038 .chained = true,
6039 .chain_id = ALC269_FIXUP_HEADSET_MIC
David Henningsson88cfcf82013-10-11 10:18:45 +02006040 },
Ayman Bagabas8ac51bb2018-12-12 18:07:57 -05006041 [ALC256_FIXUP_HUAWEI_MBXP_PINS] = {
6042 .type = HDA_FIXUP_PINS,
6043 .v.pins = (const struct hda_pintbl[]) {
6044 {0x12, 0x90a60130},
6045 {0x13, 0x40000000},
6046 {0x14, 0x90170110},
6047 {0x18, 0x411111f0},
6048 {0x19, 0x04a11040},
6049 {0x1a, 0x411111f0},
6050 {0x1b, 0x90170112},
6051 {0x1d, 0x40759a05},
6052 {0x1e, 0x411111f0},
6053 {0x21, 0x04211020},
6054 { }
6055 },
Ayman Bagabase2744fd2018-12-12 18:07:59 -05006056 .chained = true,
6057 .chain_id = ALC255_FIXUP_MIC_MUTE_LED
Ayman Bagabas8ac51bb2018-12-12 18:07:57 -05006058 },
David Henningssond240d1d2013-04-15 12:50:02 +02006059 [ALC269_FIXUP_ASUS_X101_FUNC] = {
6060 .type = HDA_FIXUP_FUNC,
6061 .v.func = alc269_fixup_x101_headset_mic,
6062 },
6063 [ALC269_FIXUP_ASUS_X101_VERB] = {
6064 .type = HDA_FIXUP_VERBS,
6065 .v.verbs = (const struct hda_verb[]) {
6066 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
6067 {0x20, AC_VERB_SET_COEF_INDEX, 0x08},
6068 {0x20, AC_VERB_SET_PROC_COEF, 0x0310},
6069 { }
6070 },
6071 .chained = true,
6072 .chain_id = ALC269_FIXUP_ASUS_X101_FUNC
6073 },
6074 [ALC269_FIXUP_ASUS_X101] = {
6075 .type = HDA_FIXUP_PINS,
6076 .v.pins = (const struct hda_pintbl[]) {
6077 { 0x18, 0x04a1182c }, /* Headset mic */
6078 { }
6079 },
6080 .chained = true,
6081 .chain_id = ALC269_FIXUP_ASUS_X101_VERB
6082 },
Dylan Reid08a978d2012-11-18 22:56:40 -08006083 [ALC271_FIXUP_AMIC_MIC2] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006084 .type = HDA_FIXUP_PINS,
6085 .v.pins = (const struct hda_pintbl[]) {
Dylan Reid08a978d2012-11-18 22:56:40 -08006086 { 0x14, 0x99130110 }, /* speaker */
6087 { 0x19, 0x01a19c20 }, /* mic */
6088 { 0x1b, 0x99a7012f }, /* int-mic */
6089 { 0x21, 0x0121401f }, /* HP out */
6090 { }
6091 },
6092 },
6093 [ALC271_FIXUP_HP_GATE_MIC_JACK] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01006094 .type = HDA_FIXUP_FUNC,
Dylan Reid08a978d2012-11-18 22:56:40 -08006095 .v.func = alc271_hp_gate_mic_jack,
6096 .chained = true,
6097 .chain_id = ALC271_FIXUP_AMIC_MIC2,
6098 },
Oleksij Rempelb1e89722013-12-04 20:50:53 +01006099 [ALC271_FIXUP_HP_GATE_MIC_JACK_E1_572] = {
6100 .type = HDA_FIXUP_FUNC,
6101 .v.func = alc269_fixup_limit_int_mic_boost,
6102 .chained = true,
6103 .chain_id = ALC271_FIXUP_HP_GATE_MIC_JACK,
6104 },
Dylan Reid42397002013-04-05 14:58:22 -07006105 [ALC269_FIXUP_ACER_AC700] = {
6106 .type = HDA_FIXUP_PINS,
6107 .v.pins = (const struct hda_pintbl[]) {
6108 { 0x12, 0x99a3092f }, /* int-mic */
6109 { 0x14, 0x99130110 }, /* speaker */
6110 { 0x18, 0x03a11c20 }, /* mic */
6111 { 0x1e, 0x0346101e }, /* SPDIF1 */
6112 { 0x21, 0x0321101f }, /* HP out */
6113 { }
6114 },
6115 .chained = true,
6116 .chain_id = ALC271_FIXUP_DMIC,
6117 },
David Henningsson3e0d6112013-04-22 14:30:14 +02006118 [ALC269_FIXUP_LIMIT_INT_MIC_BOOST] = {
6119 .type = HDA_FIXUP_FUNC,
6120 .v.func = alc269_fixup_limit_int_mic_boost,
David Henningsson27937692013-11-18 11:51:47 +01006121 .chained = true,
6122 .chain_id = ALC269_FIXUP_THINKPAD_ACPI,
David Henningsson3e0d6112013-04-22 14:30:14 +02006123 },
Oleksij Rempel2cede302013-11-27 17:12:03 +01006124 [ALC269VB_FIXUP_ASUS_ZENBOOK] = {
6125 .type = HDA_FIXUP_FUNC,
6126 .v.func = alc269_fixup_limit_int_mic_boost,
6127 .chained = true,
6128 .chain_id = ALC269VB_FIXUP_DMIC,
6129 },
Takashi Iwai23870832013-11-29 14:13:12 +01006130 [ALC269VB_FIXUP_ASUS_ZENBOOK_UX31A] = {
6131 .type = HDA_FIXUP_VERBS,
6132 .v.verbs = (const struct hda_verb[]) {
6133 /* class-D output amp +5dB */
6134 { 0x20, AC_VERB_SET_COEF_INDEX, 0x12 },
6135 { 0x20, AC_VERB_SET_PROC_COEF, 0x2800 },
6136 {}
6137 },
6138 .chained = true,
6139 .chain_id = ALC269VB_FIXUP_ASUS_ZENBOOK,
6140 },
David Henningsson8e35cd42013-11-06 11:20:01 +01006141 [ALC269_FIXUP_LIMIT_INT_MIC_BOOST_MUTE_LED] = {
6142 .type = HDA_FIXUP_FUNC,
6143 .v.func = alc269_fixup_limit_int_mic_boost,
6144 .chained = true,
6145 .chain_id = ALC269_FIXUP_HP_MUTE_LED_MIC1,
6146 },
Anisse Astier02b504d2013-06-03 11:53:10 +02006147 [ALC269VB_FIXUP_ORDISSIMO_EVE2] = {
6148 .type = HDA_FIXUP_PINS,
6149 .v.pins = (const struct hda_pintbl[]) {
6150 { 0x12, 0x99a3092f }, /* int-mic */
6151 { 0x18, 0x03a11d20 }, /* mic */
6152 { 0x19, 0x411111f0 }, /* Unused bogus pin */
6153 { }
6154 },
6155 },
Kailang Yangcd217a62013-08-22 10:15:24 +02006156 [ALC283_FIXUP_CHROME_BOOK] = {
6157 .type = HDA_FIXUP_FUNC,
6158 .v.func = alc283_fixup_chromebook,
6159 },
Kailang Yang0202e992013-12-02 15:20:15 +08006160 [ALC283_FIXUP_SENSE_COMBO_JACK] = {
6161 .type = HDA_FIXUP_FUNC,
6162 .v.func = alc283_fixup_sense_combo_jack,
6163 .chained = true,
6164 .chain_id = ALC283_FIXUP_CHROME_BOOK,
6165 },
Takashi Iwai7bba2152013-09-06 15:45:38 +02006166 [ALC282_FIXUP_ASUS_TX300] = {
6167 .type = HDA_FIXUP_FUNC,
6168 .v.func = alc282_fixup_asus_tx300,
6169 },
Kailang Yang1bb3e062013-09-27 13:10:25 +02006170 [ALC283_FIXUP_INT_MIC] = {
6171 .type = HDA_FIXUP_VERBS,
6172 .v.verbs = (const struct hda_verb[]) {
6173 {0x20, AC_VERB_SET_COEF_INDEX, 0x1a},
6174 {0x20, AC_VERB_SET_PROC_COEF, 0x0011},
6175 { }
6176 },
6177 .chained = true,
6178 .chain_id = ALC269_FIXUP_LIMIT_INT_MIC_BOOST
6179 },
David Henningsson0f4881d2013-12-20 16:08:13 +01006180 [ALC290_FIXUP_SUBWOOFER_HSJACK] = {
6181 .type = HDA_FIXUP_PINS,
6182 .v.pins = (const struct hda_pintbl[]) {
6183 { 0x17, 0x90170112 }, /* subwoofer */
6184 { }
6185 },
6186 .chained = true,
6187 .chain_id = ALC290_FIXUP_MONO_SPEAKERS_HSJACK,
6188 },
6189 [ALC290_FIXUP_SUBWOOFER] = {
6190 .type = HDA_FIXUP_PINS,
6191 .v.pins = (const struct hda_pintbl[]) {
6192 { 0x17, 0x90170112 }, /* subwoofer */
6193 { }
6194 },
6195 .chained = true,
6196 .chain_id = ALC290_FIXUP_MONO_SPEAKERS,
6197 },
David Henningsson338cae52013-10-07 10:39:59 +02006198 [ALC290_FIXUP_MONO_SPEAKERS] = {
6199 .type = HDA_FIXUP_FUNC,
6200 .v.func = alc290_fixup_mono_speakers,
David Henningsson0f4881d2013-12-20 16:08:13 +01006201 },
6202 [ALC290_FIXUP_MONO_SPEAKERS_HSJACK] = {
6203 .type = HDA_FIXUP_FUNC,
6204 .v.func = alc290_fixup_mono_speakers,
David Henningsson338cae52013-10-07 10:39:59 +02006205 .chained = true,
6206 .chain_id = ALC269_FIXUP_DELL3_MIC_NO_PRESENCE,
6207 },
David Henningssonb67ae3f2013-11-05 13:11:37 +01006208 [ALC269_FIXUP_THINKPAD_ACPI] = {
6209 .type = HDA_FIXUP_FUNC,
Takashi Iwaid5a6cab2018-06-13 12:43:10 +02006210 .v.func = alc_fixup_thinkpad_acpi,
Takashi Iwai09da1112016-09-08 12:15:02 +02006211 .chained = true,
6212 .chain_id = ALC269_FIXUP_SKU_IGNORE,
David Henningssonb67ae3f2013-11-05 13:11:37 +01006213 },
David Henningsson56f27012016-01-11 09:33:14 +01006214 [ALC269_FIXUP_DMIC_THINKPAD_ACPI] = {
6215 .type = HDA_FIXUP_FUNC,
6216 .v.func = alc_fixup_inv_dmic,
6217 .chained = true,
6218 .chain_id = ALC269_FIXUP_THINKPAD_ACPI,
6219 },
Chris Chiu5824ce82017-02-28 14:17:11 -06006220 [ALC255_FIXUP_ACER_MIC_NO_PRESENCE] = {
6221 .type = HDA_FIXUP_PINS,
6222 .v.pins = (const struct hda_pintbl[]) {
6223 { 0x19, 0x01a1913c }, /* use as headset mic, without its own jack detect */
6224 { }
6225 },
6226 .chained = true,
6227 .chain_id = ALC255_FIXUP_HEADSET_MODE
6228 },
Chris Chiu615966a2017-02-28 14:17:12 -06006229 [ALC255_FIXUP_ASUS_MIC_NO_PRESENCE] = {
6230 .type = HDA_FIXUP_PINS,
6231 .v.pins = (const struct hda_pintbl[]) {
6232 { 0x19, 0x01a1913c }, /* use as headset mic, without its own jack detect */
6233 { }
6234 },
6235 .chained = true,
6236 .chain_id = ALC255_FIXUP_HEADSET_MODE
6237 },
Kailang Yang9a22a8f52013-11-08 15:54:49 +08006238 [ALC255_FIXUP_DELL1_MIC_NO_PRESENCE] = {
6239 .type = HDA_FIXUP_PINS,
6240 .v.pins = (const struct hda_pintbl[]) {
6241 { 0x19, 0x01a1913c }, /* use as headset mic, without its own jack detect */
6242 { 0x1a, 0x01a1913d }, /* use as headphone mic, without its own jack detect */
6243 { }
6244 },
6245 .chained = true,
6246 .chain_id = ALC255_FIXUP_HEADSET_MODE
6247 },
Kailang Yang31278992014-03-03 15:27:22 +08006248 [ALC255_FIXUP_DELL2_MIC_NO_PRESENCE] = {
6249 .type = HDA_FIXUP_PINS,
6250 .v.pins = (const struct hda_pintbl[]) {
6251 { 0x19, 0x01a1913c }, /* use as headset mic, without its own jack detect */
6252 { }
6253 },
6254 .chained = true,
6255 .chain_id = ALC255_FIXUP_HEADSET_MODE_NO_HP_MIC
6256 },
Kailang Yang9a22a8f52013-11-08 15:54:49 +08006257 [ALC255_FIXUP_HEADSET_MODE] = {
6258 .type = HDA_FIXUP_FUNC,
6259 .v.func = alc_fixup_headset_mode_alc255,
Hui Wang4a83d422014-11-18 17:57:40 +08006260 .chained = true,
Takashi Iwaib3802782018-11-26 17:47:46 +01006261 .chain_id = ALC255_FIXUP_MIC_MUTE_LED
Kailang Yang9a22a8f52013-11-08 15:54:49 +08006262 },
Kailang Yang31278992014-03-03 15:27:22 +08006263 [ALC255_FIXUP_HEADSET_MODE_NO_HP_MIC] = {
6264 .type = HDA_FIXUP_FUNC,
6265 .v.func = alc_fixup_headset_mode_alc255_no_hp_mic,
6266 },
Kailang Yanga22aa262014-04-23 17:34:28 +08006267 [ALC293_FIXUP_DELL1_MIC_NO_PRESENCE] = {
6268 .type = HDA_FIXUP_PINS,
6269 .v.pins = (const struct hda_pintbl[]) {
6270 { 0x18, 0x01a1913d }, /* use as headphone mic, without its own jack detect */
6271 { 0x1a, 0x01a1913c }, /* use as headset mic, without its own jack detect */
6272 { }
6273 },
6274 .chained = true,
6275 .chain_id = ALC269_FIXUP_HEADSET_MODE
6276 },
Takashi Iwai1c37c222014-05-06 17:34:42 +02006277 [ALC292_FIXUP_TPT440_DOCK] = {
David Henningssonec56af62015-06-24 10:46:33 +02006278 .type = HDA_FIXUP_FUNC,
Takashi Iwai7f57d802015-09-24 17:36:51 +02006279 .v.func = alc_fixup_tpt440_dock,
Takashi Iwai1c37c222014-05-06 17:34:42 +02006280 .chained = true,
6281 .chain_id = ALC269_FIXUP_LIMIT_INT_MIC_BOOST
6282 },
Takashi Iwai9a811232015-12-09 15:17:43 +01006283 [ALC292_FIXUP_TPT440] = {
6284 .type = HDA_FIXUP_FUNC,
Takashi Iwai157f0b72015-12-10 23:30:43 +01006285 .v.func = alc_fixup_disable_aamix,
Takashi Iwai9a811232015-12-09 15:17:43 +01006286 .chained = true,
6287 .chain_id = ALC292_FIXUP_TPT440_DOCK,
6288 },
Anisse Astierabaa22742016-08-24 09:14:13 +02006289 [ALC283_FIXUP_HEADSET_MIC] = {
Daniel Drake9dc12862014-07-22 10:58:29 +01006290 .type = HDA_FIXUP_PINS,
6291 .v.pins = (const struct hda_pintbl[]) {
6292 { 0x19, 0x04a110f0 },
6293 { },
6294 },
6295 },
Takashi Iwaib3802782018-11-26 17:47:46 +01006296 [ALC255_FIXUP_MIC_MUTE_LED] = {
Hui Wang00ef9942014-07-31 11:52:38 +08006297 .type = HDA_FIXUP_FUNC,
Takashi Iwaib3802782018-11-26 17:47:46 +01006298 .v.func = snd_hda_gen_fixup_micmute_led,
Hui Wang00ef9942014-07-31 11:52:38 +08006299 },
Takashi Iwai1a22e772014-08-27 08:19:05 +02006300 [ALC282_FIXUP_ASPIRE_V5_PINS] = {
6301 .type = HDA_FIXUP_PINS,
6302 .v.pins = (const struct hda_pintbl[]) {
6303 { 0x12, 0x90a60130 },
6304 { 0x14, 0x90170110 },
6305 { 0x17, 0x40000008 },
6306 { 0x18, 0x411111f0 },
Mateusz Sylwestrzak04206942015-07-19 17:38:56 +02006307 { 0x19, 0x01a1913c },
Takashi Iwai1a22e772014-08-27 08:19:05 +02006308 { 0x1a, 0x411111f0 },
6309 { 0x1b, 0x411111f0 },
6310 { 0x1d, 0x40f89b2d },
6311 { 0x1e, 0x411111f0 },
6312 { 0x21, 0x0321101f },
6313 { },
6314 },
6315 },
David Henningsson7a5255f2014-10-30 08:26:01 +01006316 [ALC280_FIXUP_HP_GPIO4] = {
6317 .type = HDA_FIXUP_FUNC,
6318 .v.func = alc280_fixup_hp_gpio4,
6319 },
Kailang Yangeaa8e5e2014-11-21 15:49:11 +08006320 [ALC286_FIXUP_HP_GPIO_LED] = {
6321 .type = HDA_FIXUP_FUNC,
6322 .v.func = alc286_fixup_hp_gpio_led,
6323 },
David Henningsson33f4acd2015-01-07 15:50:13 +01006324 [ALC280_FIXUP_HP_GPIO2_MIC_HOTKEY] = {
6325 .type = HDA_FIXUP_FUNC,
6326 .v.func = alc280_fixup_hp_gpio2_mic_hotkey,
6327 },
TienFu Chenb4b33f92015-01-20 15:06:21 +01006328 [ALC280_FIXUP_HP_DOCK_PINS] = {
6329 .type = HDA_FIXUP_PINS,
6330 .v.pins = (const struct hda_pintbl[]) {
6331 { 0x1b, 0x21011020 }, /* line-out */
6332 { 0x1a, 0x01a1903c }, /* headset mic */
6333 { 0x18, 0x2181103f }, /* line-in */
6334 { },
6335 },
6336 .chained = true,
6337 .chain_id = ALC280_FIXUP_HP_GPIO4
6338 },
Jaroslav Kysela04d54662017-03-09 13:29:13 +01006339 [ALC269_FIXUP_HP_DOCK_GPIO_MIC1_LED] = {
6340 .type = HDA_FIXUP_PINS,
6341 .v.pins = (const struct hda_pintbl[]) {
6342 { 0x1b, 0x21011020 }, /* line-out */
6343 { 0x18, 0x2181103f }, /* line-in */
6344 { },
6345 },
6346 .chained = true,
6347 .chain_id = ALC269_FIXUP_HP_GPIO_MIC1_LED
6348 },
Keith Packard98973f22015-07-15 12:14:39 -07006349 [ALC280_FIXUP_HP_9480M] = {
6350 .type = HDA_FIXUP_FUNC,
6351 .v.func = alc280_fixup_hp_9480m,
6352 },
Kailang Yange1e62b92015-04-08 16:01:22 +08006353 [ALC288_FIXUP_DELL_HEADSET_MODE] = {
6354 .type = HDA_FIXUP_FUNC,
6355 .v.func = alc_fixup_headset_mode_dell_alc288,
6356 .chained = true,
Takashi Iwaib3802782018-11-26 17:47:46 +01006357 .chain_id = ALC255_FIXUP_MIC_MUTE_LED
Kailang Yange1e62b92015-04-08 16:01:22 +08006358 },
6359 [ALC288_FIXUP_DELL1_MIC_NO_PRESENCE] = {
6360 .type = HDA_FIXUP_PINS,
6361 .v.pins = (const struct hda_pintbl[]) {
6362 { 0x18, 0x01a1913c }, /* use as headset mic, without its own jack detect */
6363 { 0x1a, 0x01a1913d }, /* use as headphone mic, without its own jack detect */
6364 { }
6365 },
6366 .chained = true,
6367 .chain_id = ALC288_FIXUP_DELL_HEADSET_MODE
6368 },
Hui Wang831bfdf92015-06-26 12:35:17 +08006369 [ALC288_FIXUP_DISABLE_AAMIX] = {
6370 .type = HDA_FIXUP_FUNC,
6371 .v.func = alc_fixup_disable_aamix,
6372 .chained = true,
Takashi Iwaid44a6862018-06-19 23:04:03 +02006373 .chain_id = ALC288_FIXUP_DELL1_MIC_NO_PRESENCE
Hui Wang831bfdf92015-06-26 12:35:17 +08006374 },
6375 [ALC288_FIXUP_DELL_XPS_13] = {
6376 .type = HDA_FIXUP_FUNC,
6377 .v.func = alc_fixup_dell_xps13,
6378 .chained = true,
6379 .chain_id = ALC288_FIXUP_DISABLE_AAMIX
6380 },
Takashi Iwai8b99aba2015-06-15 11:59:32 +02006381 [ALC292_FIXUP_DISABLE_AAMIX] = {
6382 .type = HDA_FIXUP_FUNC,
6383 .v.func = alc_fixup_disable_aamix,
Hui Wang831bfdf92015-06-26 12:35:17 +08006384 .chained = true,
6385 .chain_id = ALC269_FIXUP_DELL2_MIC_NO_PRESENCE
Takashi Iwai8b99aba2015-06-15 11:59:32 +02006386 },
David Henningssonc04017e2015-12-15 14:44:03 +01006387 [ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK] = {
6388 .type = HDA_FIXUP_FUNC,
6389 .v.func = alc_fixup_disable_aamix,
6390 .chained = true,
6391 .chain_id = ALC293_FIXUP_DELL1_MIC_NO_PRESENCE
6392 },
Takashi Iwai8b99aba2015-06-15 11:59:32 +02006393 [ALC292_FIXUP_DELL_E7X] = {
6394 .type = HDA_FIXUP_FUNC,
6395 .v.func = alc_fixup_dell_xps13,
6396 .chained = true,
6397 .chain_id = ALC292_FIXUP_DISABLE_AAMIX
6398 },
Kailang Yang977e6272015-05-18 15:31:20 +08006399 [ALC298_FIXUP_DELL1_MIC_NO_PRESENCE] = {
6400 .type = HDA_FIXUP_PINS,
6401 .v.pins = (const struct hda_pintbl[]) {
6402 { 0x18, 0x01a1913c }, /* use as headset mic, without its own jack detect */
6403 { 0x1a, 0x01a1913d }, /* use as headphone mic, without its own jack detect */
6404 { }
6405 },
6406 .chained = true,
6407 .chain_id = ALC269_FIXUP_HEADSET_MODE
6408 },
Hui Wang2f726ae2017-03-31 10:31:40 +08006409 [ALC298_FIXUP_DELL_AIO_MIC_NO_PRESENCE] = {
6410 .type = HDA_FIXUP_PINS,
6411 .v.pins = (const struct hda_pintbl[]) {
6412 { 0x18, 0x01a1913c }, /* use as headset mic, without its own jack detect */
6413 { }
6414 },
6415 .chained = true,
6416 .chain_id = ALC269_FIXUP_HEADSET_MODE
6417 },
Kailang Yang6ed11312015-10-26 15:37:39 +08006418 [ALC275_FIXUP_DELL_XPS] = {
6419 .type = HDA_FIXUP_VERBS,
6420 .v.verbs = (const struct hda_verb[]) {
6421 /* Enables internal speaker */
6422 {0x20, AC_VERB_SET_COEF_INDEX, 0x1f},
6423 {0x20, AC_VERB_SET_PROC_COEF, 0x00c0},
6424 {0x20, AC_VERB_SET_COEF_INDEX, 0x30},
6425 {0x20, AC_VERB_SET_PROC_COEF, 0x00b1},
6426 {}
6427 }
6428 },
Hui Wang8c697292015-11-24 11:08:18 +08006429 [ALC256_FIXUP_DELL_XPS_13_HEADPHONE_NOISE] = {
6430 .type = HDA_FIXUP_VERBS,
6431 .v.verbs = (const struct hda_verb[]) {
6432 /* Disable pass-through path for FRONT 14h */
6433 {0x20, AC_VERB_SET_COEF_INDEX, 0x36},
6434 {0x20, AC_VERB_SET_PROC_COEF, 0x1737},
6435 {}
6436 },
6437 .chained = true,
6438 .chain_id = ALC255_FIXUP_DELL1_MIC_NO_PRESENCE
6439 },
Hui Wang23adc192015-12-08 12:27:18 +08006440 [ALC293_FIXUP_LENOVO_SPK_NOISE] = {
6441 .type = HDA_FIXUP_FUNC,
6442 .v.func = alc_fixup_disable_aamix,
6443 .chained = true,
6444 .chain_id = ALC269_FIXUP_THINKPAD_ACPI
6445 },
Kailang3694cb22015-12-28 11:35:24 +08006446 [ALC233_FIXUP_LENOVO_LINE2_MIC_HOTKEY] = {
6447 .type = HDA_FIXUP_FUNC,
6448 .v.func = alc233_fixup_lenovo_line2_mic_hotkey,
6449 },
Kai-Heng Feng3b43b712016-02-25 15:19:38 +08006450 [ALC255_FIXUP_DELL_SPK_NOISE] = {
6451 .type = HDA_FIXUP_FUNC,
6452 .v.func = alc_fixup_disable_aamix,
6453 .chained = true,
6454 .chain_id = ALC255_FIXUP_DELL1_MIC_NO_PRESENCE
6455 },
Kailang Yangd1dd4212019-01-09 17:05:24 +08006456 [ALC225_FIXUP_DISABLE_MIC_VREF] = {
6457 .type = HDA_FIXUP_FUNC,
6458 .v.func = alc_fixup_disable_mic_vref,
6459 .chained = true,
6460 .chain_id = ALC269_FIXUP_DELL1_MIC_NO_PRESENCE
6461 },
David Henningsson2ae95572016-02-25 09:37:05 +01006462 [ALC225_FIXUP_DELL1_MIC_NO_PRESENCE] = {
6463 .type = HDA_FIXUP_VERBS,
6464 .v.verbs = (const struct hda_verb[]) {
6465 /* Disable pass-through path for FRONT 14h */
6466 { 0x20, AC_VERB_SET_COEF_INDEX, 0x36 },
6467 { 0x20, AC_VERB_SET_PROC_COEF, 0x57d7 },
6468 {}
6469 },
6470 .chained = true,
Kailang Yangd1dd4212019-01-09 17:05:24 +08006471 .chain_id = ALC225_FIXUP_DISABLE_MIC_VREF
David Henningsson2ae95572016-02-25 09:37:05 +01006472 },
Takashi Iwaif8839822016-02-25 14:31:59 +01006473 [ALC280_FIXUP_HP_HEADSET_MIC] = {
6474 .type = HDA_FIXUP_FUNC,
6475 .v.func = alc_fixup_disable_aamix,
6476 .chained = true,
6477 .chain_id = ALC269_FIXUP_HEADSET_MIC,
6478 },
Hui Wange549d192016-04-01 11:00:15 +08006479 [ALC221_FIXUP_HP_FRONT_MIC] = {
6480 .type = HDA_FIXUP_PINS,
6481 .v.pins = (const struct hda_pintbl[]) {
6482 { 0x19, 0x02a19020 }, /* Front Mic */
6483 { }
6484 },
6485 },
Sven Eckelmannc636b952016-04-11 16:55:26 +02006486 [ALC292_FIXUP_TPT460] = {
6487 .type = HDA_FIXUP_FUNC,
6488 .v.func = alc_fixup_tpt440_dock,
6489 .chained = true,
6490 .chain_id = ALC293_FIXUP_LENOVO_SPK_NOISE,
6491 },
Hui Wangdd9aa332016-08-01 10:20:32 +08006492 [ALC298_FIXUP_SPK_VOLUME] = {
6493 .type = HDA_FIXUP_FUNC,
6494 .v.func = alc298_fixup_speaker_volume,
Hui Wang59ec4b52016-08-04 15:28:04 +08006495 .chained = true,
Hui Wang2f726ae2017-03-31 10:31:40 +08006496 .chain_id = ALC298_FIXUP_DELL_AIO_MIC_NO_PRESENCE,
Hui Wangdd9aa332016-08-01 10:20:32 +08006497 },
Takashi Iwaie312a862018-03-06 12:14:17 +01006498 [ALC295_FIXUP_DISABLE_DAC3] = {
6499 .type = HDA_FIXUP_FUNC,
6500 .v.func = alc295_fixup_disable_dac3,
6501 },
Kai-Heng Fengfd06c772016-08-30 15:36:34 +08006502 [ALC256_FIXUP_DELL_INSPIRON_7559_SUBWOOFER] = {
6503 .type = HDA_FIXUP_PINS,
6504 .v.pins = (const struct hda_pintbl[]) {
6505 { 0x1b, 0x90170151 },
6506 { }
6507 },
6508 .chained = true,
6509 .chain_id = ALC255_FIXUP_DELL1_MIC_NO_PRESENCE
6510 },
Gabriele Mazzotta823ff162016-12-24 19:50:01 +01006511 [ALC269_FIXUP_ATIV_BOOK_8] = {
6512 .type = HDA_FIXUP_FUNC,
6513 .v.func = alc_fixup_auto_mute_via_amp,
6514 .chained = true,
6515 .chain_id = ALC269_FIXUP_NO_SHUTUP
6516 },
Kailang Yang9eb5d0e2017-01-17 15:40:25 +08006517 [ALC221_FIXUP_HP_MIC_NO_PRESENCE] = {
6518 .type = HDA_FIXUP_PINS,
6519 .v.pins = (const struct hda_pintbl[]) {
6520 { 0x18, 0x01a1913c }, /* use as headset mic, without its own jack detect */
6521 { 0x1a, 0x01a1913d }, /* use as headphone mic, without its own jack detect */
6522 { }
6523 },
6524 .chained = true,
6525 .chain_id = ALC269_FIXUP_HEADSET_MODE
6526 },
Chris Chiuc1732ed2017-02-28 14:17:13 -06006527 [ALC256_FIXUP_ASUS_HEADSET_MODE] = {
6528 .type = HDA_FIXUP_FUNC,
6529 .v.func = alc_fixup_headset_mode,
6530 },
6531 [ALC256_FIXUP_ASUS_MIC] = {
6532 .type = HDA_FIXUP_PINS,
6533 .v.pins = (const struct hda_pintbl[]) {
6534 { 0x13, 0x90a60160 }, /* use as internal mic */
6535 { 0x19, 0x04a11120 }, /* use as headset mic, without its own jack detect */
6536 { }
6537 },
6538 .chained = true,
6539 .chain_id = ALC256_FIXUP_ASUS_HEADSET_MODE
6540 },
Chris Chiueeed4cd2017-02-28 14:17:15 -06006541 [ALC256_FIXUP_ASUS_AIO_GPIO2] = {
Takashi Iwaiae065f12018-06-19 23:00:03 +02006542 .type = HDA_FIXUP_FUNC,
6543 /* Set up GPIO2 for the speaker amp */
6544 .v.func = alc_fixup_gpio4,
Chris Chiueeed4cd2017-02-28 14:17:15 -06006545 },
Chris Chiu216d7ae2017-02-28 14:17:14 -06006546 [ALC233_FIXUP_ASUS_MIC_NO_PRESENCE] = {
6547 .type = HDA_FIXUP_PINS,
6548 .v.pins = (const struct hda_pintbl[]) {
6549 { 0x19, 0x01a1913c }, /* use as headset mic, without its own jack detect */
6550 { }
6551 },
6552 .chained = true,
6553 .chain_id = ALC269_FIXUP_HEADSET_MIC
6554 },
6555 [ALC233_FIXUP_EAPD_COEF_AND_MIC_NO_PRESENCE] = {
6556 .type = HDA_FIXUP_VERBS,
6557 .v.verbs = (const struct hda_verb[]) {
6558 /* Enables internal speaker */
6559 {0x20, AC_VERB_SET_COEF_INDEX, 0x40},
6560 {0x20, AC_VERB_SET_PROC_COEF, 0x8800},
6561 {}
6562 },
6563 .chained = true,
6564 .chain_id = ALC233_FIXUP_ASUS_MIC_NO_PRESENCE
6565 },
Kailang Yangca169cc2017-04-25 16:17:40 +08006566 [ALC233_FIXUP_LENOVO_MULTI_CODECS] = {
6567 .type = HDA_FIXUP_FUNC,
6568 .v.func = alc233_alc662_fixup_lenovo_dual_codecs,
6569 },
Jian-Hong Panea5c7eb2019-04-01 11:25:05 +08006570 [ALC233_FIXUP_ACER_HEADSET_MIC] = {
6571 .type = HDA_FIXUP_VERBS,
6572 .v.verbs = (const struct hda_verb[]) {
6573 { 0x20, AC_VERB_SET_COEF_INDEX, 0x45 },
6574 { 0x20, AC_VERB_SET_PROC_COEF, 0x5089 },
6575 { }
6576 },
6577 .chained = true,
6578 .chain_id = ALC233_FIXUP_ASUS_MIC_NO_PRESENCE
6579 },
Hui Wangf33f79f2017-07-07 12:08:29 +08006580 [ALC294_FIXUP_LENOVO_MIC_LOCATION] = {
6581 .type = HDA_FIXUP_PINS,
6582 .v.pins = (const struct hda_pintbl[]) {
6583 /* Change the mic location from front to right, otherwise there are
6584 two front mics with the same name, pulseaudio can't handle them.
6585 This is just a temporary workaround, after applying this fixup,
6586 there will be one "Front Mic" and one "Mic" in this machine.
6587 */
6588 { 0x1a, 0x04a19040 },
6589 { }
6590 },
6591 },
Kailang Yang5f364132017-07-25 16:28:16 +08006592 [ALC225_FIXUP_DELL_WYSE_MIC_NO_PRESENCE] = {
6593 .type = HDA_FIXUP_PINS,
6594 .v.pins = (const struct hda_pintbl[]) {
6595 { 0x16, 0x0101102f }, /* Rear Headset HP */
6596 { 0x19, 0x02a1913c }, /* use as Front headset mic, without its own jack detect */
6597 { 0x1a, 0x01a19030 }, /* Rear Headset MIC */
6598 { 0x1b, 0x02011020 },
6599 { }
6600 },
6601 .chained = true,
6602 .chain_id = ALC269_FIXUP_HEADSET_MODE_NO_HP_MIC
6603 },
PeiSen Houb84e8432017-09-01 15:11:56 +08006604 [ALC700_FIXUP_INTEL_REFERENCE] = {
6605 .type = HDA_FIXUP_VERBS,
6606 .v.verbs = (const struct hda_verb[]) {
6607 /* Enables internal speaker */
6608 {0x20, AC_VERB_SET_COEF_INDEX, 0x45},
6609 {0x20, AC_VERB_SET_PROC_COEF, 0x5289},
6610 {0x20, AC_VERB_SET_COEF_INDEX, 0x4A},
6611 {0x20, AC_VERB_SET_PROC_COEF, 0x001b},
6612 {0x58, AC_VERB_SET_COEF_INDEX, 0x00},
6613 {0x58, AC_VERB_SET_PROC_COEF, 0x3888},
6614 {0x20, AC_VERB_SET_COEF_INDEX, 0x6f},
6615 {0x20, AC_VERB_SET_PROC_COEF, 0x2c0b},
6616 {}
6617 }
6618 },
Kailang Yang92266652017-12-14 15:28:58 +08006619 [ALC274_FIXUP_DELL_BIND_DACS] = {
6620 .type = HDA_FIXUP_FUNC,
6621 .v.func = alc274_fixup_bind_dacs,
6622 .chained = true,
6623 .chain_id = ALC269_FIXUP_DELL1_MIC_NO_PRESENCE
6624 },
6625 [ALC274_FIXUP_DELL_AIO_LINEOUT_VERB] = {
6626 .type = HDA_FIXUP_PINS,
6627 .v.pins = (const struct hda_pintbl[]) {
6628 { 0x1b, 0x0401102f },
6629 { }
6630 },
6631 .chained = true,
6632 .chain_id = ALC274_FIXUP_DELL_BIND_DACS
6633 },
Kailang Yang61fcf8e2018-02-02 15:26:46 +08006634 [ALC298_FIXUP_TPT470_DOCK] = {
6635 .type = HDA_FIXUP_FUNC,
6636 .v.func = alc_fixup_tpt470_dock,
6637 .chained = true,
6638 .chain_id = ALC293_FIXUP_LENOVO_SPK_NOISE
6639 },
Kailang Yangae104a22018-02-05 16:07:20 +08006640 [ALC255_FIXUP_DUMMY_LINEOUT_VERB] = {
6641 .type = HDA_FIXUP_PINS,
6642 .v.pins = (const struct hda_pintbl[]) {
6643 { 0x14, 0x0201101f },
6644 { }
6645 },
6646 .chained = true,
6647 .chain_id = ALC255_FIXUP_DELL1_MIC_NO_PRESENCE
6648 },
Kailang Yangf0ba9d62018-03-16 11:46:08 +08006649 [ALC255_FIXUP_DELL_HEADSET_MIC] = {
6650 .type = HDA_FIXUP_PINS,
6651 .v.pins = (const struct hda_pintbl[]) {
6652 { 0x19, 0x01a1913c }, /* use as headset mic, without its own jack detect */
6653 { }
6654 },
Hui Wang3ce0d5a2018-04-19 13:29:04 +08006655 .chained = true,
6656 .chain_id = ALC269_FIXUP_HEADSET_MIC
Kailang Yangf0ba9d62018-03-16 11:46:08 +08006657 },
Tom Bridenbbf8ff62018-05-29 17:34:20 +01006658 [ALC295_FIXUP_HP_X360] = {
6659 .type = HDA_FIXUP_FUNC,
6660 .v.func = alc295_fixup_hp_top_speakers,
6661 .chained = true,
6662 .chain_id = ALC269_FIXUP_HP_MUTE_LED_MIC3
Kailang Yang8a328ac2018-08-21 16:54:41 +08006663 },
6664 [ALC221_FIXUP_HP_HEADSET_MIC] = {
6665 .type = HDA_FIXUP_PINS,
6666 .v.pins = (const struct hda_pintbl[]) {
6667 { 0x19, 0x0181313f},
6668 { }
6669 },
6670 .chained = true,
6671 .chain_id = ALC269_FIXUP_HEADSET_MIC
6672 },
Hui Wangc4cfcf62018-11-26 14:17:16 +08006673 [ALC285_FIXUP_LENOVO_HEADPHONE_NOISE] = {
6674 .type = HDA_FIXUP_FUNC,
6675 .v.func = alc285_fixup_invalidate_dacs,
Hui Wang6ba189c2018-12-09 09:16:43 +08006676 .chained = true,
6677 .chain_id = ALC269_FIXUP_THINKPAD_ACPI
Hui Wangc4cfcf62018-11-26 14:17:16 +08006678 },
Girija Kumar Kasinadhunie8ed64b2018-11-26 13:40:46 -05006679 [ALC295_FIXUP_HP_AUTO_MUTE] = {
6680 .type = HDA_FIXUP_FUNC,
6681 .v.func = alc_fixup_auto_mute_via_amp,
6682 },
Chris Chiu33aaebd2018-12-05 14:48:53 +08006683 [ALC286_FIXUP_ACER_AIO_MIC_NO_PRESENCE] = {
6684 .type = HDA_FIXUP_PINS,
6685 .v.pins = (const struct hda_pintbl[]) {
6686 { 0x18, 0x01a1913c }, /* use as headset mic, without its own jack detect */
6687 { }
6688 },
6689 .chained = true,
6690 .chain_id = ALC269_FIXUP_HEADSET_MIC
6691 },
Chris Chiud8ae4582018-12-07 17:17:11 +08006692 [ALC294_FIXUP_ASUS_MIC] = {
6693 .type = HDA_FIXUP_PINS,
6694 .v.pins = (const struct hda_pintbl[]) {
6695 { 0x13, 0x90a60160 }, /* use as internal mic */
6696 { 0x19, 0x04a11120 }, /* use as headset mic, without its own jack detect */
6697 { }
6698 },
6699 .chained = true,
6700 .chain_id = ALC269_FIXUP_HEADSET_MODE_NO_HP_MIC
6701 },
Jian-Hong Pan4e051102018-12-07 17:17:12 +08006702 [ALC294_FIXUP_ASUS_HEADSET_MIC] = {
6703 .type = HDA_FIXUP_PINS,
6704 .v.pins = (const struct hda_pintbl[]) {
Jian-Hong Pan82b01142018-12-27 16:46:31 +08006705 { 0x19, 0x01a1103c }, /* use as headset mic */
Jian-Hong Pan4e051102018-12-07 17:17:12 +08006706 { }
6707 },
6708 .chained = true,
6709 .chain_id = ALC269_FIXUP_HEADSET_MODE_NO_HP_MIC
6710 },
6711 [ALC294_FIXUP_ASUS_SPK] = {
6712 .type = HDA_FIXUP_VERBS,
6713 .v.verbs = (const struct hda_verb[]) {
6714 /* Set EAPD high */
6715 { 0x20, AC_VERB_SET_COEF_INDEX, 0x40 },
6716 { 0x20, AC_VERB_SET_PROC_COEF, 0x8800 },
6717 { }
6718 },
6719 .chained = true,
6720 .chain_id = ALC294_FIXUP_ASUS_HEADSET_MIC
6721 },
Jaroslav Kyselac8a9afa2019-03-14 09:21:08 +01006722 [ALC295_FIXUP_CHROME_BOOK] = {
Kailang Yange8547472018-11-28 15:32:45 +08006723 .type = HDA_FIXUP_FUNC,
Jaroslav Kyselac8a9afa2019-03-14 09:21:08 +01006724 .v.func = alc295_fixup_chromebook,
Kailang Yang8983eb62019-04-03 15:31:49 +08006725 .chained = true,
6726 .chain_id = ALC225_FIXUP_HEADSET_JACK
6727 },
6728 [ALC225_FIXUP_HEADSET_JACK] = {
6729 .type = HDA_FIXUP_FUNC,
6730 .v.func = alc_fixup_headset_jack,
Kailang Yange8547472018-11-28 15:32:45 +08006731 },
Jeremy Soller89e3a562019-01-30 16:12:31 -07006732 [ALC293_FIXUP_SYSTEM76_MIC_NO_PRESENCE] = {
6733 .type = HDA_FIXUP_PINS,
6734 .v.pins = (const struct hda_pintbl[]) {
6735 { 0x1a, 0x01a1913c }, /* use as headset mic, without its own jack detect */
6736 { }
6737 },
6738 .chained = true,
6739 .chain_id = ALC269_FIXUP_HEADSET_MODE_NO_HP_MIC
6740 },
Hui Wangc8c6ee62019-02-14 11:41:33 +08006741 [ALC285_FIXUP_LENOVO_PC_BEEP_IN_NOISE] = {
6742 .type = HDA_FIXUP_VERBS,
6743 .v.verbs = (const struct hda_verb[]) {
6744 /* Disable PCBEEP-IN passthrough */
6745 { 0x20, AC_VERB_SET_COEF_INDEX, 0x36 },
6746 { 0x20, AC_VERB_SET_PROC_COEF, 0x57d7 },
6747 { }
6748 },
6749 .chained = true,
6750 .chain_id = ALC285_FIXUP_LENOVO_HEADPHONE_NOISE
6751 },
Jian-Hong Pancbc05fd2019-03-13 17:33:24 +08006752 [ALC255_FIXUP_ACER_HEADSET_MIC] = {
6753 .type = HDA_FIXUP_PINS,
6754 .v.pins = (const struct hda_pintbl[]) {
6755 { 0x19, 0x03a11130 },
6756 { 0x1a, 0x90a60140 }, /* use as internal mic */
6757 { }
6758 },
6759 .chained = true,
6760 .chain_id = ALC255_FIXUP_HEADSET_MODE_NO_HP_MIC
6761 },
Kailang Yang136824e2019-03-14 16:22:45 +08006762 [ALC225_FIXUP_DELL_WYSE_AIO_MIC_NO_PRESENCE] = {
6763 .type = HDA_FIXUP_PINS,
6764 .v.pins = (const struct hda_pintbl[]) {
6765 { 0x16, 0x01011020 }, /* Rear Line out */
6766 { 0x19, 0x01a1913c }, /* use as Front headset mic, without its own jack detect */
6767 { }
6768 },
6769 .chained = true,
6770 .chain_id = ALC225_FIXUP_WYSE_AUTO_MUTE
6771 },
6772 [ALC225_FIXUP_WYSE_AUTO_MUTE] = {
6773 .type = HDA_FIXUP_FUNC,
6774 .v.func = alc_fixup_auto_mute_via_amp,
6775 .chained = true,
6776 .chain_id = ALC225_FIXUP_WYSE_DISABLE_MIC_VREF
6777 },
6778 [ALC225_FIXUP_WYSE_DISABLE_MIC_VREF] = {
6779 .type = HDA_FIXUP_FUNC,
6780 .v.func = alc_fixup_disable_mic_vref,
6781 .chained = true,
6782 .chain_id = ALC269_FIXUP_HEADSET_MODE_NO_HP_MIC
6783 },
Jian-Hong Pan667a8f72019-03-15 17:51:09 +08006784 [ALC286_FIXUP_ACER_AIO_HEADSET_MIC] = {
6785 .type = HDA_FIXUP_VERBS,
6786 .v.verbs = (const struct hda_verb[]) {
6787 { 0x20, AC_VERB_SET_COEF_INDEX, 0x4f },
6788 { 0x20, AC_VERB_SET_PROC_COEF, 0x5029 },
6789 { }
6790 },
6791 .chained = true,
6792 .chain_id = ALC286_FIXUP_ACER_AIO_MIC_NO_PRESENCE
6793 },
Jian-Hong Pane1037352019-03-22 11:37:18 +08006794 [ALC256_FIXUP_ASUS_MIC_NO_PRESENCE] = {
6795 .type = HDA_FIXUP_PINS,
6796 .v.pins = (const struct hda_pintbl[]) {
6797 { 0x19, 0x04a11120 }, /* use as headset mic, without its own jack detect */
6798 { }
6799 },
6800 .chained = true,
6801 .chain_id = ALC256_FIXUP_ASUS_HEADSET_MODE
6802 },
Bernhard Rosenkraenzere2a829b2019-03-05 00:38:19 +01006803 [ALC299_FIXUP_PREDATOR_SPK] = {
6804 .type = HDA_FIXUP_PINS,
6805 .v.pins = (const struct hda_pintbl[]) {
6806 { 0x21, 0x90170150 }, /* use as headset mic, without its own jack detect */
6807 { }
6808 }
6809 },
Takashi Iwai1d045db2011-07-07 18:23:21 +02006810};
6811
6812static const struct snd_pci_quirk alc269_fixup_tbl[] = {
Marius Knausta6b92b62014-03-03 01:48:58 +01006813 SND_PCI_QUIRK(0x1025, 0x0283, "Acer TravelMate 8371", ALC269_FIXUP_INV_DMIC),
David Henningsson693b6132012-06-22 19:12:10 +02006814 SND_PCI_QUIRK(0x1025, 0x029b, "Acer 1810TZ", ALC269_FIXUP_INV_DMIC),
6815 SND_PCI_QUIRK(0x1025, 0x0349, "Acer AOD260", ALC269_FIXUP_INV_DMIC),
David Henningssonaaedfb42013-08-16 14:09:02 +02006816 SND_PCI_QUIRK(0x1025, 0x047c, "Acer AC700", ALC269_FIXUP_ACER_AC700),
Takashi Iwai78197172015-06-27 10:21:13 +02006817 SND_PCI_QUIRK(0x1025, 0x072d, "Acer Aspire V5-571G", ALC269_FIXUP_ASPIRE_HEADSET_MIC),
6818 SND_PCI_QUIRK(0x1025, 0x080d, "Acer Aspire V5-122P", ALC269_FIXUP_ASPIRE_HEADSET_MIC),
David Henningssonaaedfb42013-08-16 14:09:02 +02006819 SND_PCI_QUIRK(0x1025, 0x0740, "Acer AO725", ALC271_FIXUP_HP_GATE_MIC_JACK),
6820 SND_PCI_QUIRK(0x1025, 0x0742, "Acer AO756", ALC271_FIXUP_HP_GATE_MIC_JACK),
Simon South02322ac2016-03-02 23:10:44 -05006821 SND_PCI_QUIRK(0x1025, 0x0762, "Acer Aspire E1-472", ALC271_FIXUP_HP_GATE_MIC_JACK_E1_572),
Oleksij Rempelb1e89722013-12-04 20:50:53 +01006822 SND_PCI_QUIRK(0x1025, 0x0775, "Acer Aspire E1-572", ALC271_FIXUP_HP_GATE_MIC_JACK_E1_572),
Takashi Iwai1a22e772014-08-27 08:19:05 +02006823 SND_PCI_QUIRK(0x1025, 0x079b, "Acer Aspire V5-573G", ALC282_FIXUP_ASPIRE_V5_PINS),
Chris Chiu705b65f2018-12-05 14:48:54 +08006824 SND_PCI_QUIRK(0x1025, 0x102b, "Acer Aspire C24-860", ALC286_FIXUP_ACER_AIO_MIC_NO_PRESENCE),
Takashi Iwaib9c2fa52015-11-19 16:39:50 +01006825 SND_PCI_QUIRK(0x1025, 0x106d, "Acer Cloudbook 14", ALC283_FIXUP_CHROME_BOOK),
Chris Chiuc7531e32019-03-21 17:17:31 +08006826 SND_PCI_QUIRK(0x1025, 0x1099, "Acer Aspire E5-523G", ALC255_FIXUP_ACER_MIC_NO_PRESENCE),
6827 SND_PCI_QUIRK(0x1025, 0x110e, "Acer Aspire ES1-432", ALC255_FIXUP_ACER_MIC_NO_PRESENCE),
Bernhard Rosenkraenzere2a829b2019-03-05 00:38:19 +01006828 SND_PCI_QUIRK(0x1025, 0x1246, "Acer Predator Helios 500", ALC299_FIXUP_PREDATOR_SPK),
Jian-Hong Pan667a8f72019-03-15 17:51:09 +08006829 SND_PCI_QUIRK(0x1025, 0x128f, "Acer Veriton Z6860G", ALC286_FIXUP_ACER_AIO_HEADSET_MIC),
6830 SND_PCI_QUIRK(0x1025, 0x1290, "Acer Veriton Z4860G", ALC286_FIXUP_ACER_AIO_HEADSET_MIC),
6831 SND_PCI_QUIRK(0x1025, 0x1291, "Acer Veriton Z4660G", ALC286_FIXUP_ACER_AIO_HEADSET_MIC),
Jian-Hong Pan2733cce2019-03-21 16:39:04 +08006832 SND_PCI_QUIRK(0x1025, 0x1308, "Acer Aspire Z24-890", ALC286_FIXUP_ACER_AIO_HEADSET_MIC),
Jian-Hong Panea5c7eb2019-04-01 11:25:05 +08006833 SND_PCI_QUIRK(0x1025, 0x132a, "Acer TravelMate B114-21", ALC233_FIXUP_ACER_HEADSET_MIC),
Jian-Hong Pancbc05fd2019-03-13 17:33:24 +08006834 SND_PCI_QUIRK(0x1025, 0x1330, "Acer TravelMate X514-51T", ALC255_FIXUP_ACER_HEADSET_MIC),
David Henningssonaaedfb42013-08-16 14:09:02 +02006835 SND_PCI_QUIRK(0x1028, 0x0470, "Dell M101z", ALC269_FIXUP_DELL_M101Z),
Kailang Yang6ed11312015-10-26 15:37:39 +08006836 SND_PCI_QUIRK(0x1028, 0x054b, "Dell XPS one 2710", ALC275_FIXUP_DELL_XPS),
Takashi Iwai86f799b2015-11-14 17:46:31 +01006837 SND_PCI_QUIRK(0x1028, 0x05bd, "Dell Latitude E6440", ALC292_FIXUP_DELL_E7X),
Takashi Iwaicf521032016-01-15 12:59:25 +01006838 SND_PCI_QUIRK(0x1028, 0x05be, "Dell Latitude E6540", ALC292_FIXUP_DELL_E7X),
Takashi Iwai8b99aba2015-06-15 11:59:32 +02006839 SND_PCI_QUIRK(0x1028, 0x05ca, "Dell Latitude E7240", ALC292_FIXUP_DELL_E7X),
6840 SND_PCI_QUIRK(0x1028, 0x05cb, "Dell Latitude E7440", ALC292_FIXUP_DELL_E7X),
David Henningsson0f4881d2013-12-20 16:08:13 +01006841 SND_PCI_QUIRK(0x1028, 0x05da, "Dell Vostro 5460", ALC290_FIXUP_SUBWOOFER),
David Henningsson73bdd592013-04-15 15:44:14 +02006842 SND_PCI_QUIRK(0x1028, 0x05f4, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE),
6843 SND_PCI_QUIRK(0x1028, 0x05f5, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE),
6844 SND_PCI_QUIRK(0x1028, 0x05f6, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE),
David Henningsson0f4881d2013-12-20 16:08:13 +01006845 SND_PCI_QUIRK(0x1028, 0x0615, "Dell Vostro 5470", ALC290_FIXUP_SUBWOOFER_HSJACK),
6846 SND_PCI_QUIRK(0x1028, 0x0616, "Dell Vostro 5470", ALC290_FIXUP_SUBWOOFER_HSJACK),
Takashi Iwai98070572016-01-12 21:06:39 +01006847 SND_PCI_QUIRK(0x1028, 0x062c, "Dell Latitude E5550", ALC292_FIXUP_DELL_E7X),
Takashi Iwai42755542015-06-29 10:56:53 +02006848 SND_PCI_QUIRK(0x1028, 0x062e, "Dell Latitude E7450", ALC292_FIXUP_DELL_E7X),
David Henningsson0f4881d2013-12-20 16:08:13 +01006849 SND_PCI_QUIRK(0x1028, 0x0638, "Dell Inspiron 5439", ALC290_FIXUP_MONO_SPEAKERS_HSJACK),
Kailang Yanga22aa262014-04-23 17:34:28 +08006850 SND_PCI_QUIRK(0x1028, 0x064a, "Dell", ALC293_FIXUP_DELL1_MIC_NO_PRESENCE),
6851 SND_PCI_QUIRK(0x1028, 0x064b, "Dell", ALC293_FIXUP_DELL1_MIC_NO_PRESENCE),
Hui Wang831bfdf92015-06-26 12:35:17 +08006852 SND_PCI_QUIRK(0x1028, 0x0665, "Dell XPS 13", ALC288_FIXUP_DELL_XPS_13),
Bastien Noceraafecb142016-04-18 11:10:42 +02006853 SND_PCI_QUIRK(0x1028, 0x0669, "Dell Optiplex 9020m", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE),
Takashi Iwai3a05d122015-07-29 09:04:52 +02006854 SND_PCI_QUIRK(0x1028, 0x069a, "Dell Vostro 5480", ALC290_FIXUP_SUBWOOFER_HSJACK),
Kailang Yang8b724152014-12-17 17:08:59 +08006855 SND_PCI_QUIRK(0x1028, 0x06c7, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE),
Kailang Yangb7343042014-12-02 14:50:45 +08006856 SND_PCI_QUIRK(0x1028, 0x06d9, "Dell", ALC293_FIXUP_DELL1_MIC_NO_PRESENCE),
6857 SND_PCI_QUIRK(0x1028, 0x06da, "Dell", ALC293_FIXUP_DELL1_MIC_NO_PRESENCE),
David Henningssonc04017e2015-12-15 14:44:03 +01006858 SND_PCI_QUIRK(0x1028, 0x06db, "Dell", ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK),
6859 SND_PCI_QUIRK(0x1028, 0x06dd, "Dell", ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK),
6860 SND_PCI_QUIRK(0x1028, 0x06de, "Dell", ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK),
6861 SND_PCI_QUIRK(0x1028, 0x06df, "Dell", ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK),
6862 SND_PCI_QUIRK(0x1028, 0x06e0, "Dell", ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK),
Kai-Heng Feng423cd782016-05-20 15:47:23 +08006863 SND_PCI_QUIRK(0x1028, 0x0704, "Dell XPS 13 9350", ALC256_FIXUP_DELL_XPS_13_HEADPHONE_NOISE),
Kai-Heng Fengfd06c772016-08-30 15:36:34 +08006864 SND_PCI_QUIRK(0x1028, 0x0706, "Dell Inspiron 7559", ALC256_FIXUP_DELL_INSPIRON_7559_SUBWOOFER),
Kai-Heng Feng3b43b712016-02-25 15:19:38 +08006865 SND_PCI_QUIRK(0x1028, 0x0725, "Dell Inspiron 3162", ALC255_FIXUP_DELL_SPK_NOISE),
Takashi Iwaic0ca5ec2019-02-20 16:15:45 +01006866 SND_PCI_QUIRK(0x1028, 0x0738, "Dell Precision 5820", ALC269_FIXUP_NO_SHUTUP),
Kai-Heng Feng423cd782016-05-20 15:47:23 +08006867 SND_PCI_QUIRK(0x1028, 0x075b, "Dell XPS 13 9360", ALC256_FIXUP_DELL_XPS_13_HEADPHONE_NOISE),
Kai-Heng Feng709ae622018-10-04 11:39:42 +08006868 SND_PCI_QUIRK(0x1028, 0x075c, "Dell XPS 27 7760", ALC298_FIXUP_SPK_VOLUME),
Hui Wangdd9aa332016-08-01 10:20:32 +08006869 SND_PCI_QUIRK(0x1028, 0x075d, "Dell AIO", ALC298_FIXUP_SPK_VOLUME),
Takashi Iwaie312a862018-03-06 12:14:17 +01006870 SND_PCI_QUIRK(0x1028, 0x07b0, "Dell Precision 7520", ALC295_FIXUP_DISABLE_DAC3),
Takashi Iwai493de342017-02-28 17:27:57 +01006871 SND_PCI_QUIRK(0x1028, 0x0798, "Dell Inspiron 17 7000 Gaming", ALC256_FIXUP_DELL_INSPIRON_7559_SUBWOOFER),
Kailang Yang5f364132017-07-25 16:28:16 +08006872 SND_PCI_QUIRK(0x1028, 0x080c, "Dell WYSE", ALC225_FIXUP_DELL_WYSE_MIC_NO_PRESENCE),
Takashi Iwaie4c9fd12018-01-10 08:34:28 +01006873 SND_PCI_QUIRK(0x1028, 0x082a, "Dell XPS 13 9360", ALC256_FIXUP_DELL_XPS_13_HEADPHONE_NOISE),
Kailang Yang40e2c4e2018-02-02 15:13:09 +08006874 SND_PCI_QUIRK(0x1028, 0x084b, "Dell", ALC274_FIXUP_DELL_AIO_LINEOUT_VERB),
6875 SND_PCI_QUIRK(0x1028, 0x084e, "Dell", ALC274_FIXUP_DELL_AIO_LINEOUT_VERB),
Kailang Yangf0ba9d62018-03-16 11:46:08 +08006876 SND_PCI_QUIRK(0x1028, 0x0871, "Dell Precision 3630", ALC255_FIXUP_DELL_HEADSET_MIC),
6877 SND_PCI_QUIRK(0x1028, 0x0872, "Dell Precision 3630", ALC255_FIXUP_DELL_HEADSET_MIC),
Kailang Yangae104a22018-02-05 16:07:20 +08006878 SND_PCI_QUIRK(0x1028, 0x0873, "Dell Precision 3930", ALC255_FIXUP_DUMMY_LINEOUT_VERB),
Kailang Yang136824e2019-03-14 16:22:45 +08006879 SND_PCI_QUIRK(0x1028, 0x08ad, "Dell WYSE AIO", ALC225_FIXUP_DELL_WYSE_AIO_MIC_NO_PRESENCE),
Kailang Yangda484d02019-03-14 15:50:59 +08006880 SND_PCI_QUIRK(0x1028, 0x08ae, "Dell WYSE NB", ALC225_FIXUP_DELL1_MIC_NO_PRESENCE),
Kailang Yangc2a7c55a2019-01-03 15:53:39 +08006881 SND_PCI_QUIRK(0x1028, 0x0935, "Dell", ALC274_FIXUP_DELL_AIO_LINEOUT_VERB),
Kailang Yanga22aa262014-04-23 17:34:28 +08006882 SND_PCI_QUIRK(0x1028, 0x164a, "Dell", ALC293_FIXUP_DELL1_MIC_NO_PRESENCE),
6883 SND_PCI_QUIRK(0x1028, 0x164b, "Dell", ALC293_FIXUP_DELL1_MIC_NO_PRESENCE),
Takashi Iwai08fb0d02013-01-10 17:33:58 +01006884 SND_PCI_QUIRK(0x103c, 0x1586, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC2),
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01006885 SND_PCI_QUIRK(0x103c, 0x18e6, "HP", ALC269_FIXUP_HP_GPIO_LED),
David Henningsson8e35cd42013-11-06 11:20:01 +01006886 SND_PCI_QUIRK(0x103c, 0x218b, "HP", ALC269_FIXUP_LIMIT_INT_MIC_BOOST_MUTE_LED),
David Henningsson33f4acd2015-01-07 15:50:13 +01006887 SND_PCI_QUIRK(0x103c, 0x225f, "HP", ALC280_FIXUP_HP_GPIO2_MIC_HOTKEY),
Kailang Yangc60666b2014-02-21 16:23:35 +08006888 /* ALC282 */
Hui Wang7976eb42015-02-13 11:14:41 +08006889 SND_PCI_QUIRK(0x103c, 0x21f9, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yang8a02b162014-06-13 17:16:31 +08006890 SND_PCI_QUIRK(0x103c, 0x2210, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yang8a02b162014-06-13 17:16:31 +08006891 SND_PCI_QUIRK(0x103c, 0x2214, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08006892 SND_PCI_QUIRK(0x103c, 0x2236, "HP", ALC269_FIXUP_HP_LINE1_MIC1_LED),
6893 SND_PCI_QUIRK(0x103c, 0x2237, "HP", ALC269_FIXUP_HP_LINE1_MIC1_LED),
6894 SND_PCI_QUIRK(0x103c, 0x2238, "HP", ALC269_FIXUP_HP_LINE1_MIC1_LED),
6895 SND_PCI_QUIRK(0x103c, 0x2239, "HP", ALC269_FIXUP_HP_LINE1_MIC1_LED),
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08006896 SND_PCI_QUIRK(0x103c, 0x224b, "HP", ALC269_FIXUP_HP_LINE1_MIC1_LED),
Kailang Yangc60666b2014-02-21 16:23:35 +08006897 SND_PCI_QUIRK(0x103c, 0x2268, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangc60666b2014-02-21 16:23:35 +08006898 SND_PCI_QUIRK(0x103c, 0x226a, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
6899 SND_PCI_QUIRK(0x103c, 0x226b, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yang8a02b162014-06-13 17:16:31 +08006900 SND_PCI_QUIRK(0x103c, 0x226e, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangeaa8e5e2014-11-21 15:49:11 +08006901 SND_PCI_QUIRK(0x103c, 0x2271, "HP", ALC286_FIXUP_HP_GPIO_LED),
TienFu Chenb4b33f92015-01-20 15:06:21 +01006902 SND_PCI_QUIRK(0x103c, 0x2272, "HP", ALC280_FIXUP_HP_DOCK_PINS),
TienFu Chen3271cb22015-02-10 09:09:41 +01006903 SND_PCI_QUIRK(0x103c, 0x2273, "HP", ALC280_FIXUP_HP_DOCK_PINS),
Kailang Yangc60666b2014-02-21 16:23:35 +08006904 SND_PCI_QUIRK(0x103c, 0x229e, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangc60666b2014-02-21 16:23:35 +08006905 SND_PCI_QUIRK(0x103c, 0x22b2, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
6906 SND_PCI_QUIRK(0x103c, 0x22b7, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
6907 SND_PCI_QUIRK(0x103c, 0x22bf, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangc60666b2014-02-21 16:23:35 +08006908 SND_PCI_QUIRK(0x103c, 0x22cf, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Keith Packard98973f22015-07-15 12:14:39 -07006909 SND_PCI_QUIRK(0x103c, 0x22db, "HP", ALC280_FIXUP_HP_9480M),
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08006910 SND_PCI_QUIRK(0x103c, 0x22dc, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
6911 SND_PCI_QUIRK(0x103c, 0x22fb, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
Kailang Yangc60666b2014-02-21 16:23:35 +08006912 /* ALC290 */
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08006913 SND_PCI_QUIRK(0x103c, 0x221b, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08006914 SND_PCI_QUIRK(0x103c, 0x2221, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08006915 SND_PCI_QUIRK(0x103c, 0x2225, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08006916 SND_PCI_QUIRK(0x103c, 0x2253, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
6917 SND_PCI_QUIRK(0x103c, 0x2254, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
6918 SND_PCI_QUIRK(0x103c, 0x2255, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
6919 SND_PCI_QUIRK(0x103c, 0x2256, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
6920 SND_PCI_QUIRK(0x103c, 0x2257, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08006921 SND_PCI_QUIRK(0x103c, 0x2259, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
Jaroslav Kysela04d54662017-03-09 13:29:13 +01006922 SND_PCI_QUIRK(0x103c, 0x225a, "HP", ALC269_FIXUP_HP_DOCK_GPIO_MIC1_LED),
Kailang Yangc60666b2014-02-21 16:23:35 +08006923 SND_PCI_QUIRK(0x103c, 0x2260, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangc60666b2014-02-21 16:23:35 +08006924 SND_PCI_QUIRK(0x103c, 0x2263, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
6925 SND_PCI_QUIRK(0x103c, 0x2264, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
6926 SND_PCI_QUIRK(0x103c, 0x2265, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08006927 SND_PCI_QUIRK(0x103c, 0x2272, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
6928 SND_PCI_QUIRK(0x103c, 0x2273, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
Kailang Yang9c5dc3b2014-06-19 16:11:36 +08006929 SND_PCI_QUIRK(0x103c, 0x2278, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED),
Kailang Yangc60666b2014-02-21 16:23:35 +08006930 SND_PCI_QUIRK(0x103c, 0x227f, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangc60666b2014-02-21 16:23:35 +08006931 SND_PCI_QUIRK(0x103c, 0x2282, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangc60666b2014-02-21 16:23:35 +08006932 SND_PCI_QUIRK(0x103c, 0x228b, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangc60666b2014-02-21 16:23:35 +08006933 SND_PCI_QUIRK(0x103c, 0x228e, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
6934 SND_PCI_QUIRK(0x103c, 0x22c5, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangc60666b2014-02-21 16:23:35 +08006935 SND_PCI_QUIRK(0x103c, 0x22c7, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
6936 SND_PCI_QUIRK(0x103c, 0x22c8, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yangc60666b2014-02-21 16:23:35 +08006937 SND_PCI_QUIRK(0x103c, 0x22c4, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Kailang Yang8a02b162014-06-13 17:16:31 +08006938 SND_PCI_QUIRK(0x103c, 0x2334, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
6939 SND_PCI_QUIRK(0x103c, 0x2335, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
6940 SND_PCI_QUIRK(0x103c, 0x2336, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
6941 SND_PCI_QUIRK(0x103c, 0x2337, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1),
Takashi Iwaif8839822016-02-25 14:31:59 +01006942 SND_PCI_QUIRK(0x103c, 0x221c, "HP EliteBook 755 G2", ALC280_FIXUP_HP_HEADSET_MIC),
Jaroslav Kysela167897f2019-03-13 13:40:15 +01006943 SND_PCI_QUIRK(0x103c, 0x802e, "HP Z240 SFF", ALC221_FIXUP_HP_MIC_NO_PRESENCE),
6944 SND_PCI_QUIRK(0x103c, 0x802f, "HP Z240", ALC221_FIXUP_HP_MIC_NO_PRESENCE),
Takashi Iwai563785e2018-11-12 09:43:12 +01006945 SND_PCI_QUIRK(0x103c, 0x820d, "HP Pavilion 15", ALC269_FIXUP_HP_MUTE_LED_MIC3),
Hui Wange549d192016-04-01 11:00:15 +08006946 SND_PCI_QUIRK(0x103c, 0x8256, "HP", ALC221_FIXUP_HP_FRONT_MIC),
Tom Bridenbbf8ff62018-05-29 17:34:20 +01006947 SND_PCI_QUIRK(0x103c, 0x827e, "HP x360", ALC295_FIXUP_HP_X360),
Jaroslav Kysela167897f2019-03-13 13:40:15 +01006948 SND_PCI_QUIRK(0x103c, 0x82bf, "HP G3 mini", ALC221_FIXUP_HP_MIC_NO_PRESENCE),
6949 SND_PCI_QUIRK(0x103c, 0x82c0, "HP G3 mini premium", ALC221_FIXUP_HP_MIC_NO_PRESENCE),
Alexandru Gagniuc56e40eb2018-08-04 11:44:44 -05006950 SND_PCI_QUIRK(0x103c, 0x83b9, "HP Spectre x360", ALC269_FIXUP_HP_MUTE_LED_MIC3),
Chris Chiuc1732ed2017-02-28 14:17:13 -06006951 SND_PCI_QUIRK(0x1043, 0x103e, "ASUS X540SA", ALC256_FIXUP_ASUS_MIC),
Takashi Iwai7bba2152013-09-06 15:45:38 +02006952 SND_PCI_QUIRK(0x1043, 0x103f, "ASUS TX300", ALC282_FIXUP_ASUS_TX300),
David Henningsson3e0d6112013-04-22 14:30:14 +02006953 SND_PCI_QUIRK(0x1043, 0x106d, "Asus K53BE", ALC269_FIXUP_LIMIT_INT_MIC_BOOST),
Wandrille RONCE9cf65332018-12-19 14:52:44 +01006954 SND_PCI_QUIRK(0x1043, 0x10a1, "ASUS UX391UA", ALC294_FIXUP_ASUS_SPK),
Chris Chiuc1732ed2017-02-28 14:17:13 -06006955 SND_PCI_QUIRK(0x1043, 0x10c0, "ASUS X540SA", ALC256_FIXUP_ASUS_MIC),
Takashi Iwai4eab0ea2017-06-06 12:33:17 +02006956 SND_PCI_QUIRK(0x1043, 0x10d0, "ASUS X540LA/X540LJ", ALC255_FIXUP_ASUS_MIC_NO_PRESENCE),
David Henningsson3e0d6112013-04-22 14:30:14 +02006957 SND_PCI_QUIRK(0x1043, 0x115d, "Asus 1015E", ALC269_FIXUP_LIMIT_INT_MIC_BOOST),
Takashi Iwai4eab0ea2017-06-06 12:33:17 +02006958 SND_PCI_QUIRK(0x1043, 0x11c0, "ASUS X556UR", ALC255_FIXUP_ASUS_MIC_NO_PRESENCE),
6959 SND_PCI_QUIRK(0x1043, 0x1290, "ASUS X441SA", ALC233_FIXUP_EAPD_COEF_AND_MIC_NO_PRESENCE),
6960 SND_PCI_QUIRK(0x1043, 0x12a0, "ASUS X441UV", ALC233_FIXUP_EAPD_COEF_AND_MIC_NO_PRESENCE),
Chris Chiuc1732ed2017-02-28 14:17:13 -06006961 SND_PCI_QUIRK(0x1043, 0x12f0, "ASUS X541UV", ALC256_FIXUP_ASUS_MIC),
6962 SND_PCI_QUIRK(0x1043, 0x12e0, "ASUS X541SA", ALC256_FIXUP_ASUS_MIC),
6963 SND_PCI_QUIRK(0x1043, 0x13b0, "ASUS Z550SA", ALC256_FIXUP_ASUS_MIC),
Oleksij Rempel2cede302013-11-27 17:12:03 +01006964 SND_PCI_QUIRK(0x1043, 0x1427, "Asus Zenbook UX31E", ALC269VB_FIXUP_ASUS_ZENBOOK),
Takashi Iwai23870832013-11-29 14:13:12 +01006965 SND_PCI_QUIRK(0x1043, 0x1517, "Asus Zenbook UX31A", ALC269VB_FIXUP_ASUS_ZENBOOK_UX31A),
David Henningsson3e0d6112013-04-22 14:30:14 +02006966 SND_PCI_QUIRK(0x1043, 0x16e3, "ASUS UX50", ALC269_FIXUP_STEREO_DMIC),
Takashi Iwai017f2a12011-07-09 14:42:25 +02006967 SND_PCI_QUIRK(0x1043, 0x1a13, "Asus G73Jw", ALC269_FIXUP_ASUS_G73JW),
Chris Chiu28e8af82017-06-05 15:05:30 -06006968 SND_PCI_QUIRK(0x1043, 0x1a30, "ASUS X705UD", ALC256_FIXUP_ASUS_MIC),
David Henningsson693b6132012-06-22 19:12:10 +02006969 SND_PCI_QUIRK(0x1043, 0x1b13, "Asus U41SV", ALC269_FIXUP_INV_DMIC),
Chris Chiu615966a2017-02-28 14:17:12 -06006970 SND_PCI_QUIRK(0x1043, 0x1bbd, "ASUS Z550MA", ALC255_FIXUP_ASUS_MIC_NO_PRESENCE),
Takashi Iwai4eab0ea2017-06-06 12:33:17 +02006971 SND_PCI_QUIRK(0x1043, 0x1c23, "Asus X55U", ALC269_FIXUP_LIMIT_INT_MIC_BOOST),
Chris Chiuc1732ed2017-02-28 14:17:13 -06006972 SND_PCI_QUIRK(0x1043, 0x1ccd, "ASUS X555UB", ALC256_FIXUP_ASUS_MIC),
Chris Chiueeed4cd2017-02-28 14:17:15 -06006973 SND_PCI_QUIRK(0x1043, 0x3030, "ASUS ZN270IE", ALC256_FIXUP_ASUS_AIO_GPIO2),
Takashi Iwaiadabb3e2011-08-03 07:48:37 +02006974 SND_PCI_QUIRK(0x1043, 0x831a, "ASUS P901", ALC269_FIXUP_STEREO_DMIC),
6975 SND_PCI_QUIRK(0x1043, 0x834a, "ASUS S101", ALC269_FIXUP_STEREO_DMIC),
6976 SND_PCI_QUIRK(0x1043, 0x8398, "ASUS P1005", ALC269_FIXUP_STEREO_DMIC),
6977 SND_PCI_QUIRK(0x1043, 0x83ce, "ASUS P1005", ALC269_FIXUP_STEREO_DMIC),
David Henningssond240d1d2013-04-15 12:50:02 +02006978 SND_PCI_QUIRK(0x1043, 0x8516, "ASUS X101CH", ALC269_FIXUP_ASUS_X101),
Takashi Iwaif88abaa2014-02-07 12:07:59 +01006979 SND_PCI_QUIRK(0x104d, 0x90b5, "Sony VAIO Pro 11", ALC286_FIXUP_SONY_MIC_NO_PRESENCE),
David Henningsson88cfcf82013-10-11 10:18:45 +02006980 SND_PCI_QUIRK(0x104d, 0x90b6, "Sony VAIO Pro 13", ALC286_FIXUP_SONY_MIC_NO_PRESENCE),
Takashi Iwai1d045db2011-07-07 18:23:21 +02006981 SND_PCI_QUIRK(0x104d, 0x9073, "Sony VAIO", ALC275_FIXUP_SONY_VAIO_GPIO2),
6982 SND_PCI_QUIRK(0x104d, 0x907b, "Sony VAIO", ALC275_FIXUP_SONY_HWEQ),
6983 SND_PCI_QUIRK(0x104d, 0x9084, "Sony VAIO", ALC275_FIXUP_SONY_HWEQ),
Takashi Iwaie9bd7d52014-05-21 11:06:49 +02006984 SND_PCI_QUIRK(0x104d, 0x9099, "Sony VAIO S13", ALC275_FIXUP_SONY_DISABLE_AAMIX),
Takashi Iwai24519912011-08-16 15:08:49 +02006985 SND_PCI_QUIRK(0x10cf, 0x1475, "Lifebook", ALC269_FIXUP_LIFEBOOK),
Takashi Iwai4df3fd12015-06-29 08:38:02 +02006986 SND_PCI_QUIRK(0x10cf, 0x159f, "Lifebook E780", ALC269_FIXUP_LIFEBOOK_NO_HP_TO_LINEOUT),
Takashi Iwaicc7016a2015-04-08 20:47:55 +02006987 SND_PCI_QUIRK(0x10cf, 0x15dc, "Lifebook T731", ALC269_FIXUP_LIFEBOOK_HP_PIN),
Takashi Iwai88776f32015-05-01 09:20:34 +02006988 SND_PCI_QUIRK(0x10cf, 0x1757, "Lifebook E752", ALC269_FIXUP_LIFEBOOK_HP_PIN),
Jan-Marek Glogowskifdcc9682018-02-14 11:29:15 +01006989 SND_PCI_QUIRK(0x10cf, 0x1629, "Lifebook U7x7", ALC255_FIXUP_LIFEBOOK_U7x7_HEADSET_MIC),
David Henningsson2041d562014-06-13 11:15:44 +02006990 SND_PCI_QUIRK(0x10cf, 0x1845, "Lifebook U904", ALC269_FIXUP_LIFEBOOK_EXTMIC),
PeiSen Houb84e8432017-09-01 15:11:56 +08006991 SND_PCI_QUIRK(0x10ec, 0x10f2, "Intel Reference board", ALC700_FIXUP_INTEL_REFERENCE),
YOKOTA Hiroshi0fca97a2018-07-01 18:30:01 +09006992 SND_PCI_QUIRK(0x10f7, 0x8338, "Panasonic CF-SZ6", ALC269_FIXUP_HEADSET_MODE),
David Henningssona33cc482014-10-07 10:18:41 +02006993 SND_PCI_QUIRK(0x144d, 0xc109, "Samsung Ativ book 9 (NP900X3G)", ALC269_FIXUP_INV_DMIC),
Gabriele Mazzotta823ff162016-12-24 19:50:01 +01006994 SND_PCI_QUIRK(0x144d, 0xc740, "Samsung Ativ book 8 (NP870Z5G)", ALC269_FIXUP_ATIV_BOOK_8),
Anisse Astierabaa22742016-08-24 09:14:13 +02006995 SND_PCI_QUIRK(0x1458, 0xfa53, "Gigabyte BXBT-2807", ALC283_FIXUP_HEADSET_MIC),
6996 SND_PCI_QUIRK(0x1462, 0xb120, "MSI Cubi MS-B120", ALC283_FIXUP_HEADSET_MIC),
Anisse Astier8cd65272018-11-23 17:59:11 +01006997 SND_PCI_QUIRK(0x1462, 0xb171, "Cubi N 8GL (MS-B171)", ALC283_FIXUP_HEADSET_MIC),
Jeremy Soller89e3a562019-01-30 16:12:31 -07006998 SND_PCI_QUIRK(0x1558, 0x1325, "System76 Darter Pro (darp5)", ALC293_FIXUP_SYSTEM76_MIC_NO_PRESENCE),
Jeremy Soller80a50522019-05-07 17:11:08 -04006999 SND_PCI_QUIRK(0x1558, 0x8550, "System76 Gazelle (gaze14)", ALC293_FIXUP_SYSTEM76_MIC_NO_PRESENCE),
7000 SND_PCI_QUIRK(0x1558, 0x8560, "System76 Gazelle (gaze14)", ALC293_FIXUP_SYSTEM76_MIC_NO_PRESENCE),
Kailang Yangca169cc2017-04-25 16:17:40 +08007001 SND_PCI_QUIRK(0x17aa, 0x1036, "Lenovo P520", ALC233_FIXUP_LENOVO_MULTI_CODECS),
Takashi Iwai1d045db2011-07-07 18:23:21 +02007002 SND_PCI_QUIRK(0x17aa, 0x20f2, "Thinkpad SL410/510", ALC269_FIXUP_SKU_IGNORE),
7003 SND_PCI_QUIRK(0x17aa, 0x215e, "Thinkpad L512", ALC269_FIXUP_SKU_IGNORE),
7004 SND_PCI_QUIRK(0x17aa, 0x21b8, "Thinkpad Edge 14", ALC269_FIXUP_SKU_IGNORE),
7005 SND_PCI_QUIRK(0x17aa, 0x21ca, "Thinkpad L412", ALC269_FIXUP_SKU_IGNORE),
7006 SND_PCI_QUIRK(0x17aa, 0x21e9, "Thinkpad Edge 15", ALC269_FIXUP_SKU_IGNORE),
Takashi Iwai707fba32012-08-02 09:04:39 +02007007 SND_PCI_QUIRK(0x17aa, 0x21f6, "Thinkpad T530", ALC269_FIXUP_LENOVO_DOCK),
Felix Kaechelec8415a42012-08-06 23:02:01 +02007008 SND_PCI_QUIRK(0x17aa, 0x21fa, "Thinkpad X230", ALC269_FIXUP_LENOVO_DOCK),
Stefán Freyr84f98fd2012-10-19 22:46:00 +02007009 SND_PCI_QUIRK(0x17aa, 0x21f3, "Thinkpad T430", ALC269_FIXUP_LENOVO_DOCK),
Philipp A. Mohrenweiser4407be62012-08-06 13:14:18 +02007010 SND_PCI_QUIRK(0x17aa, 0x21fb, "Thinkpad T430s", ALC269_FIXUP_LENOVO_DOCK),
David Henningsson108cc102012-07-20 10:37:25 +02007011 SND_PCI_QUIRK(0x17aa, 0x2203, "Thinkpad X230 Tablet", ALC269_FIXUP_LENOVO_DOCK),
David Henningssonaaedfb42013-08-16 14:09:02 +02007012 SND_PCI_QUIRK(0x17aa, 0x2208, "Thinkpad T431s", ALC269_FIXUP_LENOVO_DOCK),
Takashi Iwai9a811232015-12-09 15:17:43 +01007013 SND_PCI_QUIRK(0x17aa, 0x220c, "Thinkpad T440s", ALC292_FIXUP_TPT440),
Takashi Iwai1c37c222014-05-06 17:34:42 +02007014 SND_PCI_QUIRK(0x17aa, 0x220e, "Thinkpad T440p", ALC292_FIXUP_TPT440_DOCK),
Takashi Iwaia12137e2014-06-27 12:14:35 +02007015 SND_PCI_QUIRK(0x17aa, 0x2210, "Thinkpad T540p", ALC292_FIXUP_TPT440_DOCK),
Rick Sherman59a51a62015-08-18 21:04:30 -05007016 SND_PCI_QUIRK(0x17aa, 0x2211, "Thinkpad W541", ALC292_FIXUP_TPT440_DOCK),
Takashi Iwai6d169412014-10-07 17:27:02 +02007017 SND_PCI_QUIRK(0x17aa, 0x2212, "Thinkpad T440", ALC292_FIXUP_TPT440_DOCK),
Lukas Bossard7c215392014-10-29 18:31:07 +01007018 SND_PCI_QUIRK(0x17aa, 0x2214, "Thinkpad X240", ALC292_FIXUP_TPT440_DOCK),
David Henningssona4a9e082013-08-16 14:09:01 +02007019 SND_PCI_QUIRK(0x17aa, 0x2215, "Thinkpad", ALC269_FIXUP_LIMIT_INT_MIC_BOOST),
Takashi Iwaib6903c02015-12-10 12:20:20 +01007020 SND_PCI_QUIRK(0x17aa, 0x2218, "Thinkpad X1 Carbon 2nd", ALC292_FIXUP_TPT440_DOCK),
Laura Abbottd05ea7d2015-10-02 11:09:54 -07007021 SND_PCI_QUIRK(0x17aa, 0x2223, "ThinkPad T550", ALC292_FIXUP_TPT440_DOCK),
Yves-Alexis Perezc0278662015-04-11 09:31:35 +02007022 SND_PCI_QUIRK(0x17aa, 0x2226, "ThinkPad X250", ALC292_FIXUP_TPT440_DOCK),
Kailang Yang61fcf8e2018-02-02 15:26:46 +08007023 SND_PCI_QUIRK(0x17aa, 0x222d, "Thinkpad", ALC298_FIXUP_TPT470_DOCK),
7024 SND_PCI_QUIRK(0x17aa, 0x222e, "Thinkpad", ALC298_FIXUP_TPT470_DOCK),
Torsten Hilbrichdab38e42016-06-07 13:14:21 +02007025 SND_PCI_QUIRK(0x17aa, 0x2231, "Thinkpad T560", ALC292_FIXUP_TPT460),
Sven Eckelmannc636b952016-04-11 16:55:26 +02007026 SND_PCI_QUIRK(0x17aa, 0x2233, "Thinkpad", ALC292_FIXUP_TPT460),
Kailang Yang61fcf8e2018-02-02 15:26:46 +08007027 SND_PCI_QUIRK(0x17aa, 0x2245, "Thinkpad T470", ALC298_FIXUP_TPT470_DOCK),
7028 SND_PCI_QUIRK(0x17aa, 0x2246, "Thinkpad", ALC298_FIXUP_TPT470_DOCK),
7029 SND_PCI_QUIRK(0x17aa, 0x2247, "Thinkpad", ALC298_FIXUP_TPT470_DOCK),
Dennis Wassenberge4c07b32018-03-08 13:17:54 +01007030 SND_PCI_QUIRK(0x17aa, 0x2249, "Thinkpad", ALC292_FIXUP_TPT460),
Kailang Yang61fcf8e2018-02-02 15:26:46 +08007031 SND_PCI_QUIRK(0x17aa, 0x224b, "Thinkpad", ALC298_FIXUP_TPT470_DOCK),
7032 SND_PCI_QUIRK(0x17aa, 0x224c, "Thinkpad", ALC298_FIXUP_TPT470_DOCK),
7033 SND_PCI_QUIRK(0x17aa, 0x224d, "Thinkpad", ALC298_FIXUP_TPT470_DOCK),
Benjamin Berg85981df2018-02-14 13:29:39 +01007034 SND_PCI_QUIRK(0x17aa, 0x225d, "Thinkpad T480", ALC269_FIXUP_LIMIT_INT_MIC_BOOST),
Kailang3694cb22015-12-28 11:35:24 +08007035 SND_PCI_QUIRK(0x17aa, 0x30bb, "ThinkCentre AIO", ALC233_FIXUP_LENOVO_LINE2_MIC_HOTKEY),
Hui Wang6ef2f682016-03-11 12:04:02 +08007036 SND_PCI_QUIRK(0x17aa, 0x30e2, "ThinkCentre AIO", ALC233_FIXUP_LENOVO_LINE2_MIC_HOTKEY),
Hui Wangf33f79f2017-07-07 12:08:29 +08007037 SND_PCI_QUIRK(0x17aa, 0x310c, "ThinkCentre Station", ALC294_FIXUP_LENOVO_MIC_LOCATION),
Hui Wange41fc8c2018-06-25 14:40:56 +08007038 SND_PCI_QUIRK(0x17aa, 0x312a, "ThinkCentre Station", ALC294_FIXUP_LENOVO_MIC_LOCATION),
Kailang Yang65811832018-04-25 17:07:27 +08007039 SND_PCI_QUIRK(0x17aa, 0x312f, "ThinkCentre Station", ALC294_FIXUP_LENOVO_MIC_LOCATION),
Hui Wang8da5bbf2017-12-22 11:17:46 +08007040 SND_PCI_QUIRK(0x17aa, 0x313c, "ThinkCentre Station", ALC294_FIXUP_LENOVO_MIC_LOCATION),
David Henningsson56f27012016-01-11 09:33:14 +01007041 SND_PCI_QUIRK(0x17aa, 0x3902, "Lenovo E50-80", ALC269_FIXUP_DMIC_THINKPAD_ACPI),
Takashi Iwaifedb2242014-11-13 07:11:38 +01007042 SND_PCI_QUIRK(0x17aa, 0x3977, "IdeaPad S210", ALC283_FIXUP_INT_MIC),
Takashi Iwai9b745ab2014-03-07 08:37:19 +01007043 SND_PCI_QUIRK(0x17aa, 0x3978, "IdeaPad Y410P", ALC269_FIXUP_NO_SHUTUP),
David Henningssona4a9e082013-08-16 14:09:01 +02007044 SND_PCI_QUIRK(0x17aa, 0x5013, "Thinkpad", ALC269_FIXUP_LIMIT_INT_MIC_BOOST),
Kailang Yang1bb3e062013-09-27 13:10:25 +02007045 SND_PCI_QUIRK(0x17aa, 0x501a, "Thinkpad", ALC283_FIXUP_INT_MIC),
Takashi Iwaic497d9f2014-10-08 12:14:40 +02007046 SND_PCI_QUIRK(0x17aa, 0x501e, "Thinkpad L440", ALC292_FIXUP_TPT440_DOCK),
David Henningssoncd5302c2013-08-19 12:22:33 +02007047 SND_PCI_QUIRK(0x17aa, 0x5026, "Thinkpad", ALC269_FIXUP_LIMIT_INT_MIC_BOOST),
Jo-Philipp Wichf2aa1112015-04-13 12:47:26 +02007048 SND_PCI_QUIRK(0x17aa, 0x5034, "Thinkpad T450", ALC292_FIXUP_TPT440_DOCK),
Sebastian Wicki80b311d2015-03-23 17:23:11 +01007049 SND_PCI_QUIRK(0x17aa, 0x5036, "Thinkpad T450s", ALC292_FIXUP_TPT440_DOCK),
Ansgar Hegerfeld09ea9972015-05-14 12:31:32 +02007050 SND_PCI_QUIRK(0x17aa, 0x503c, "Thinkpad L450", ALC292_FIXUP_TPT440_DOCK),
Conrad Kostecki037e1192016-04-26 10:08:10 +02007051 SND_PCI_QUIRK(0x17aa, 0x504a, "ThinkPad X260", ALC292_FIXUP_TPT440_DOCK),
Hui Wang23adc192015-12-08 12:27:18 +08007052 SND_PCI_QUIRK(0x17aa, 0x504b, "Thinkpad", ALC293_FIXUP_LENOVO_SPK_NOISE),
Jaroslav Kysela0f087ee2016-06-24 15:13:16 +02007053 SND_PCI_QUIRK(0x17aa, 0x5050, "Thinkpad T560p", ALC292_FIXUP_TPT460),
Torsten Hilbrich9cd25742016-07-05 10:40:22 +02007054 SND_PCI_QUIRK(0x17aa, 0x5051, "Thinkpad L460", ALC292_FIXUP_TPT460),
Jaroslav Kysela0f087ee2016-06-24 15:13:16 +02007055 SND_PCI_QUIRK(0x17aa, 0x5053, "Thinkpad T460", ALC292_FIXUP_TPT460),
Kailang Yang61fcf8e2018-02-02 15:26:46 +08007056 SND_PCI_QUIRK(0x17aa, 0x505d, "Thinkpad", ALC298_FIXUP_TPT470_DOCK),
7057 SND_PCI_QUIRK(0x17aa, 0x505f, "Thinkpad", ALC298_FIXUP_TPT470_DOCK),
7058 SND_PCI_QUIRK(0x17aa, 0x5062, "Thinkpad", ALC298_FIXUP_TPT470_DOCK),
David Henningssoncd5302c2013-08-19 12:22:33 +02007059 SND_PCI_QUIRK(0x17aa, 0x5109, "Thinkpad", ALC269_FIXUP_LIMIT_INT_MIC_BOOST),
Kailang Yang61fcf8e2018-02-02 15:26:46 +08007060 SND_PCI_QUIRK(0x17aa, 0x511e, "Thinkpad", ALC298_FIXUP_TPT470_DOCK),
7061 SND_PCI_QUIRK(0x17aa, 0x511f, "Thinkpad", ALC298_FIXUP_TPT470_DOCK),
David Henningsson012e7eb2012-08-08 08:43:37 +02007062 SND_PCI_QUIRK(0x17aa, 0x3bf8, "Quanta FL1", ALC269_FIXUP_PCM_44K),
Takashi Iwai1d045db2011-07-07 18:23:21 +02007063 SND_PCI_QUIRK(0x17aa, 0x9e54, "LENOVO NB", ALC269_FIXUP_LENOVO_EAPD),
Ayman Bagabase2744fd2018-12-12 18:07:59 -05007064 SND_PCI_QUIRK(0x19e5, 0x3200, "Huawei MBX", ALC255_FIXUP_MIC_MUTE_LED),
7065 SND_PCI_QUIRK(0x19e5, 0x3201, "Huawei MBX", ALC255_FIXUP_MIC_MUTE_LED),
Ayman Bagabas8ac51bb2018-12-12 18:07:57 -05007066 SND_PCI_QUIRK(0x19e5, 0x3204, "Huawei MBXP", ALC256_FIXUP_HUAWEI_MBXP_PINS),
Anisse Astier02b504d2013-06-03 11:53:10 +02007067 SND_PCI_QUIRK(0x1b7d, 0xa831, "Ordissimo EVE2 ", ALC269VB_FIXUP_ORDISSIMO_EVE2), /* Also known as Malata PC-B1303 */
Takashi Iwaia4297b52011-08-23 18:40:12 +02007068
Takashi Iwaia7f3eed2012-02-16 13:03:18 +01007069#if 0
Takashi Iwaia4297b52011-08-23 18:40:12 +02007070 /* Below is a quirk table taken from the old code.
7071 * Basically the device should work as is without the fixup table.
7072 * If BIOS doesn't give a proper info, enable the corresponding
7073 * fixup entry.
Jesper Juhl7d7eb9e2012-04-12 22:11:25 +02007074 */
Takashi Iwaia4297b52011-08-23 18:40:12 +02007075 SND_PCI_QUIRK(0x1043, 0x8330, "ASUS Eeepc P703 P900A",
7076 ALC269_FIXUP_AMIC),
7077 SND_PCI_QUIRK(0x1043, 0x1013, "ASUS N61Da", ALC269_FIXUP_AMIC),
Takashi Iwaia4297b52011-08-23 18:40:12 +02007078 SND_PCI_QUIRK(0x1043, 0x1143, "ASUS B53f", ALC269_FIXUP_AMIC),
7079 SND_PCI_QUIRK(0x1043, 0x1133, "ASUS UJ20ft", ALC269_FIXUP_AMIC),
7080 SND_PCI_QUIRK(0x1043, 0x1183, "ASUS K72DR", ALC269_FIXUP_AMIC),
7081 SND_PCI_QUIRK(0x1043, 0x11b3, "ASUS K52DR", ALC269_FIXUP_AMIC),
7082 SND_PCI_QUIRK(0x1043, 0x11e3, "ASUS U33Jc", ALC269_FIXUP_AMIC),
7083 SND_PCI_QUIRK(0x1043, 0x1273, "ASUS UL80Jt", ALC269_FIXUP_AMIC),
7084 SND_PCI_QUIRK(0x1043, 0x1283, "ASUS U53Jc", ALC269_FIXUP_AMIC),
7085 SND_PCI_QUIRK(0x1043, 0x12b3, "ASUS N82JV", ALC269_FIXUP_AMIC),
7086 SND_PCI_QUIRK(0x1043, 0x12d3, "ASUS N61Jv", ALC269_FIXUP_AMIC),
7087 SND_PCI_QUIRK(0x1043, 0x13a3, "ASUS UL30Vt", ALC269_FIXUP_AMIC),
7088 SND_PCI_QUIRK(0x1043, 0x1373, "ASUS G73JX", ALC269_FIXUP_AMIC),
7089 SND_PCI_QUIRK(0x1043, 0x1383, "ASUS UJ30Jc", ALC269_FIXUP_AMIC),
7090 SND_PCI_QUIRK(0x1043, 0x13d3, "ASUS N61JA", ALC269_FIXUP_AMIC),
7091 SND_PCI_QUIRK(0x1043, 0x1413, "ASUS UL50", ALC269_FIXUP_AMIC),
7092 SND_PCI_QUIRK(0x1043, 0x1443, "ASUS UL30", ALC269_FIXUP_AMIC),
7093 SND_PCI_QUIRK(0x1043, 0x1453, "ASUS M60Jv", ALC269_FIXUP_AMIC),
7094 SND_PCI_QUIRK(0x1043, 0x1483, "ASUS UL80", ALC269_FIXUP_AMIC),
7095 SND_PCI_QUIRK(0x1043, 0x14f3, "ASUS F83Vf", ALC269_FIXUP_AMIC),
7096 SND_PCI_QUIRK(0x1043, 0x14e3, "ASUS UL20", ALC269_FIXUP_AMIC),
7097 SND_PCI_QUIRK(0x1043, 0x1513, "ASUS UX30", ALC269_FIXUP_AMIC),
7098 SND_PCI_QUIRK(0x1043, 0x1593, "ASUS N51Vn", ALC269_FIXUP_AMIC),
7099 SND_PCI_QUIRK(0x1043, 0x15a3, "ASUS N60Jv", ALC269_FIXUP_AMIC),
7100 SND_PCI_QUIRK(0x1043, 0x15b3, "ASUS N60Dp", ALC269_FIXUP_AMIC),
7101 SND_PCI_QUIRK(0x1043, 0x15c3, "ASUS N70De", ALC269_FIXUP_AMIC),
7102 SND_PCI_QUIRK(0x1043, 0x15e3, "ASUS F83T", ALC269_FIXUP_AMIC),
7103 SND_PCI_QUIRK(0x1043, 0x1643, "ASUS M60J", ALC269_FIXUP_AMIC),
7104 SND_PCI_QUIRK(0x1043, 0x1653, "ASUS U50", ALC269_FIXUP_AMIC),
7105 SND_PCI_QUIRK(0x1043, 0x1693, "ASUS F50N", ALC269_FIXUP_AMIC),
7106 SND_PCI_QUIRK(0x1043, 0x16a3, "ASUS F5Q", ALC269_FIXUP_AMIC),
7107 SND_PCI_QUIRK(0x1043, 0x1723, "ASUS P80", ALC269_FIXUP_AMIC),
7108 SND_PCI_QUIRK(0x1043, 0x1743, "ASUS U80", ALC269_FIXUP_AMIC),
7109 SND_PCI_QUIRK(0x1043, 0x1773, "ASUS U20A", ALC269_FIXUP_AMIC),
7110 SND_PCI_QUIRK(0x1043, 0x1883, "ASUS F81Se", ALC269_FIXUP_AMIC),
7111 SND_PCI_QUIRK(0x152d, 0x1778, "Quanta ON1", ALC269_FIXUP_DMIC),
7112 SND_PCI_QUIRK(0x17aa, 0x3be9, "Quanta Wistron", ALC269_FIXUP_AMIC),
7113 SND_PCI_QUIRK(0x17aa, 0x3bf8, "Quanta FL1", ALC269_FIXUP_AMIC),
7114 SND_PCI_QUIRK(0x17ff, 0x059a, "Quanta EL3", ALC269_FIXUP_DMIC),
7115 SND_PCI_QUIRK(0x17ff, 0x059b, "Quanta JR1", ALC269_FIXUP_DMIC),
7116#endif
7117 {}
7118};
7119
David Henningsson214eef72014-07-22 14:09:35 +02007120static const struct snd_pci_quirk alc269_fixup_vendor_tbl[] = {
7121 SND_PCI_QUIRK_VENDOR(0x1025, "Acer Aspire", ALC271_FIXUP_DMIC),
7122 SND_PCI_QUIRK_VENDOR(0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED),
7123 SND_PCI_QUIRK_VENDOR(0x104d, "Sony VAIO", ALC269_FIXUP_SONY_VAIO),
7124 SND_PCI_QUIRK_VENDOR(0x17aa, "Thinkpad", ALC269_FIXUP_THINKPAD_ACPI),
7125 {}
7126};
7127
Takashi Iwai1727a772013-01-10 09:52:52 +01007128static const struct hda_model_fixup alc269_fixup_models[] = {
Takashi Iwaia4297b52011-08-23 18:40:12 +02007129 {.id = ALC269_FIXUP_AMIC, .name = "laptop-amic"},
7130 {.id = ALC269_FIXUP_DMIC, .name = "laptop-dmic"},
Takashi Iwai6e72aa52012-06-25 10:52:25 +02007131 {.id = ALC269_FIXUP_STEREO_DMIC, .name = "alc269-dmic"},
7132 {.id = ALC271_FIXUP_DMIC, .name = "alc271-dmic"},
7133 {.id = ALC269_FIXUP_INV_DMIC, .name = "inv-dmic"},
David Henningsson7c478f02013-10-11 10:18:46 +02007134 {.id = ALC269_FIXUP_HEADSET_MIC, .name = "headset-mic"},
Takashi Iwaib0169512015-05-19 10:20:13 +02007135 {.id = ALC269_FIXUP_HEADSET_MODE, .name = "headset-mode"},
7136 {.id = ALC269_FIXUP_HEADSET_MODE_NO_HP_MIC, .name = "headset-mode-no-hp-mic"},
David Henningsson108cc102012-07-20 10:37:25 +02007137 {.id = ALC269_FIXUP_LENOVO_DOCK, .name = "lenovo-dock"},
Takashi Iwai9f5c6fa2013-03-18 14:15:58 +01007138 {.id = ALC269_FIXUP_HP_GPIO_LED, .name = "hp-gpio-led"},
Jaroslav Kysela04d54662017-03-09 13:29:13 +01007139 {.id = ALC269_FIXUP_HP_DOCK_GPIO_MIC1_LED, .name = "hp-dock-gpio-mic1-led"},
David Henningssone32aa852013-06-17 11:04:02 +02007140 {.id = ALC269_FIXUP_DELL1_MIC_NO_PRESENCE, .name = "dell-headset-multi"},
7141 {.id = ALC269_FIXUP_DELL2_MIC_NO_PRESENCE, .name = "dell-headset-dock"},
Takashi Iwaia26d96c2018-06-26 15:09:25 +02007142 {.id = ALC269_FIXUP_DELL3_MIC_NO_PRESENCE, .name = "dell-headset3"},
7143 {.id = ALC269_FIXUP_DELL4_MIC_NO_PRESENCE, .name = "dell-headset4"},
Kailang Yangbe8ef162014-04-08 15:19:49 +08007144 {.id = ALC283_FIXUP_CHROME_BOOK, .name = "alc283-dac-wcaps"},
Kailang Yang0202e992013-12-02 15:20:15 +08007145 {.id = ALC283_FIXUP_SENSE_COMBO_JACK, .name = "alc283-sense-combo"},
Takashi Iwai1c37c222014-05-06 17:34:42 +02007146 {.id = ALC292_FIXUP_TPT440_DOCK, .name = "tpt440-dock"},
Takashi Iwai9a811232015-12-09 15:17:43 +01007147 {.id = ALC292_FIXUP_TPT440, .name = "tpt440"},
Sven Eckelmannc636b952016-04-11 16:55:26 +02007148 {.id = ALC292_FIXUP_TPT460, .name = "tpt460"},
Takashi Iwaia26d96c2018-06-26 15:09:25 +02007149 {.id = ALC298_FIXUP_TPT470_DOCK, .name = "tpt470-dock"},
Takashi Iwaiba90d6a2017-05-22 16:38:47 +02007150 {.id = ALC233_FIXUP_LENOVO_MULTI_CODECS, .name = "dual-codecs"},
Takashi Iwai28d1d6d2017-10-18 14:01:58 +02007151 {.id = ALC700_FIXUP_INTEL_REFERENCE, .name = "alc700-ref"},
Takashi Iwaia26d96c2018-06-26 15:09:25 +02007152 {.id = ALC269_FIXUP_SONY_VAIO, .name = "vaio"},
7153 {.id = ALC269_FIXUP_DELL_M101Z, .name = "dell-m101z"},
7154 {.id = ALC269_FIXUP_ASUS_G73JW, .name = "asus-g73jw"},
7155 {.id = ALC269_FIXUP_LENOVO_EAPD, .name = "lenovo-eapd"},
7156 {.id = ALC275_FIXUP_SONY_HWEQ, .name = "sony-hweq"},
7157 {.id = ALC269_FIXUP_PCM_44K, .name = "pcm44k"},
7158 {.id = ALC269_FIXUP_LIFEBOOK, .name = "lifebook"},
7159 {.id = ALC269_FIXUP_LIFEBOOK_EXTMIC, .name = "lifebook-extmic"},
7160 {.id = ALC269_FIXUP_LIFEBOOK_HP_PIN, .name = "lifebook-hp-pin"},
7161 {.id = ALC255_FIXUP_LIFEBOOK_U7x7_HEADSET_MIC, .name = "lifebook-u7x7"},
7162 {.id = ALC269VB_FIXUP_AMIC, .name = "alc269vb-amic"},
7163 {.id = ALC269VB_FIXUP_DMIC, .name = "alc269vb-dmic"},
7164 {.id = ALC269_FIXUP_HP_MUTE_LED_MIC1, .name = "hp-mute-led-mic1"},
7165 {.id = ALC269_FIXUP_HP_MUTE_LED_MIC2, .name = "hp-mute-led-mic2"},
7166 {.id = ALC269_FIXUP_HP_MUTE_LED_MIC3, .name = "hp-mute-led-mic3"},
7167 {.id = ALC269_FIXUP_HP_GPIO_MIC1_LED, .name = "hp-gpio-mic1"},
7168 {.id = ALC269_FIXUP_HP_LINE1_MIC1_LED, .name = "hp-line1-mic1"},
7169 {.id = ALC269_FIXUP_NO_SHUTUP, .name = "noshutup"},
7170 {.id = ALC286_FIXUP_SONY_MIC_NO_PRESENCE, .name = "sony-nomic"},
7171 {.id = ALC269_FIXUP_ASPIRE_HEADSET_MIC, .name = "aspire-headset-mic"},
7172 {.id = ALC269_FIXUP_ASUS_X101, .name = "asus-x101"},
7173 {.id = ALC271_FIXUP_HP_GATE_MIC_JACK, .name = "acer-ao7xx"},
7174 {.id = ALC271_FIXUP_HP_GATE_MIC_JACK_E1_572, .name = "acer-aspire-e1"},
7175 {.id = ALC269_FIXUP_ACER_AC700, .name = "acer-ac700"},
7176 {.id = ALC269_FIXUP_LIMIT_INT_MIC_BOOST, .name = "limit-mic-boost"},
7177 {.id = ALC269VB_FIXUP_ASUS_ZENBOOK, .name = "asus-zenbook"},
7178 {.id = ALC269VB_FIXUP_ASUS_ZENBOOK_UX31A, .name = "asus-zenbook-ux31a"},
7179 {.id = ALC269VB_FIXUP_ORDISSIMO_EVE2, .name = "ordissimo"},
7180 {.id = ALC282_FIXUP_ASUS_TX300, .name = "asus-tx300"},
7181 {.id = ALC283_FIXUP_INT_MIC, .name = "alc283-int-mic"},
7182 {.id = ALC290_FIXUP_MONO_SPEAKERS_HSJACK, .name = "mono-speakers"},
7183 {.id = ALC290_FIXUP_SUBWOOFER_HSJACK, .name = "alc290-subwoofer"},
7184 {.id = ALC269_FIXUP_THINKPAD_ACPI, .name = "thinkpad"},
7185 {.id = ALC269_FIXUP_DMIC_THINKPAD_ACPI, .name = "dmic-thinkpad"},
7186 {.id = ALC255_FIXUP_ACER_MIC_NO_PRESENCE, .name = "alc255-acer"},
7187 {.id = ALC255_FIXUP_ASUS_MIC_NO_PRESENCE, .name = "alc255-asus"},
7188 {.id = ALC255_FIXUP_DELL1_MIC_NO_PRESENCE, .name = "alc255-dell1"},
7189 {.id = ALC255_FIXUP_DELL2_MIC_NO_PRESENCE, .name = "alc255-dell2"},
7190 {.id = ALC293_FIXUP_DELL1_MIC_NO_PRESENCE, .name = "alc293-dell1"},
7191 {.id = ALC283_FIXUP_HEADSET_MIC, .name = "alc283-headset"},
Takashi Iwaib3802782018-11-26 17:47:46 +01007192 {.id = ALC255_FIXUP_MIC_MUTE_LED, .name = "alc255-dell-mute"},
Takashi Iwaia26d96c2018-06-26 15:09:25 +02007193 {.id = ALC282_FIXUP_ASPIRE_V5_PINS, .name = "aspire-v5"},
7194 {.id = ALC280_FIXUP_HP_GPIO4, .name = "hp-gpio4"},
7195 {.id = ALC286_FIXUP_HP_GPIO_LED, .name = "hp-gpio-led"},
7196 {.id = ALC280_FIXUP_HP_GPIO2_MIC_HOTKEY, .name = "hp-gpio2-hotkey"},
7197 {.id = ALC280_FIXUP_HP_DOCK_PINS, .name = "hp-dock-pins"},
7198 {.id = ALC269_FIXUP_HP_DOCK_GPIO_MIC1_LED, .name = "hp-dock-gpio-mic"},
7199 {.id = ALC280_FIXUP_HP_9480M, .name = "hp-9480m"},
7200 {.id = ALC288_FIXUP_DELL_HEADSET_MODE, .name = "alc288-dell-headset"},
7201 {.id = ALC288_FIXUP_DELL1_MIC_NO_PRESENCE, .name = "alc288-dell1"},
7202 {.id = ALC288_FIXUP_DELL_XPS_13, .name = "alc288-dell-xps13"},
7203 {.id = ALC292_FIXUP_DELL_E7X, .name = "dell-e7x"},
7204 {.id = ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK, .name = "alc293-dell"},
7205 {.id = ALC298_FIXUP_DELL1_MIC_NO_PRESENCE, .name = "alc298-dell1"},
7206 {.id = ALC298_FIXUP_DELL_AIO_MIC_NO_PRESENCE, .name = "alc298-dell-aio"},
7207 {.id = ALC275_FIXUP_DELL_XPS, .name = "alc275-dell-xps"},
7208 {.id = ALC256_FIXUP_DELL_XPS_13_HEADPHONE_NOISE, .name = "alc256-dell-xps13"},
7209 {.id = ALC293_FIXUP_LENOVO_SPK_NOISE, .name = "lenovo-spk-noise"},
7210 {.id = ALC233_FIXUP_LENOVO_LINE2_MIC_HOTKEY, .name = "lenovo-hotkey"},
7211 {.id = ALC255_FIXUP_DELL_SPK_NOISE, .name = "dell-spk-noise"},
Kailang Yang82aa0d72019-01-11 17:15:53 +08007212 {.id = ALC225_FIXUP_DELL1_MIC_NO_PRESENCE, .name = "alc225-dell1"},
Takashi Iwaia26d96c2018-06-26 15:09:25 +02007213 {.id = ALC295_FIXUP_DISABLE_DAC3, .name = "alc295-disable-dac3"},
7214 {.id = ALC280_FIXUP_HP_HEADSET_MIC, .name = "alc280-hp-headset"},
7215 {.id = ALC221_FIXUP_HP_FRONT_MIC, .name = "alc221-hp-mic"},
7216 {.id = ALC298_FIXUP_SPK_VOLUME, .name = "alc298-spk-volume"},
7217 {.id = ALC256_FIXUP_DELL_INSPIRON_7559_SUBWOOFER, .name = "dell-inspiron-7559"},
7218 {.id = ALC269_FIXUP_ATIV_BOOK_8, .name = "ativ-book"},
7219 {.id = ALC221_FIXUP_HP_MIC_NO_PRESENCE, .name = "alc221-hp-mic"},
7220 {.id = ALC256_FIXUP_ASUS_HEADSET_MODE, .name = "alc256-asus-headset"},
7221 {.id = ALC256_FIXUP_ASUS_MIC, .name = "alc256-asus-mic"},
7222 {.id = ALC256_FIXUP_ASUS_AIO_GPIO2, .name = "alc256-asus-aio"},
7223 {.id = ALC233_FIXUP_ASUS_MIC_NO_PRESENCE, .name = "alc233-asus"},
7224 {.id = ALC233_FIXUP_EAPD_COEF_AND_MIC_NO_PRESENCE, .name = "alc233-eapd"},
7225 {.id = ALC294_FIXUP_LENOVO_MIC_LOCATION, .name = "alc294-lenovo-mic"},
7226 {.id = ALC225_FIXUP_DELL_WYSE_MIC_NO_PRESENCE, .name = "alc225-wyse"},
7227 {.id = ALC274_FIXUP_DELL_AIO_LINEOUT_VERB, .name = "alc274-dell-aio"},
7228 {.id = ALC255_FIXUP_DUMMY_LINEOUT_VERB, .name = "alc255-dummy-lineout"},
7229 {.id = ALC255_FIXUP_DELL_HEADSET_MIC, .name = "alc255-dell-headset"},
7230 {.id = ALC295_FIXUP_HP_X360, .name = "alc295-hp-x360"},
Kailang Yang8983eb62019-04-03 15:31:49 +08007231 {.id = ALC225_FIXUP_HEADSET_JACK, .name = "alc-headset-jack"},
7232 {.id = ALC295_FIXUP_CHROME_BOOK, .name = "alc-chrome-book"},
Bernhard Rosenkraenzere2a829b2019-03-05 00:38:19 +01007233 {.id = ALC299_FIXUP_PREDATOR_SPK, .name = "predator-spk"},
Takashi Iwai1d045db2011-07-07 18:23:21 +02007234 {}
7235};
Kailang Yangcfc5a842016-02-03 15:20:39 +08007236#define ALC225_STANDARD_PINS \
Kailang Yangcfc5a842016-02-03 15:20:39 +08007237 {0x21, 0x04211020}
Takashi Iwai1d045db2011-07-07 18:23:21 +02007238
Hui Wange8191a82015-04-24 13:39:59 +08007239#define ALC256_STANDARD_PINS \
7240 {0x12, 0x90a60140}, \
7241 {0x14, 0x90170110}, \
Hui Wange8191a82015-04-24 13:39:59 +08007242 {0x21, 0x02211020}
7243
David Henningssonfea185e2014-09-03 10:23:04 +02007244#define ALC282_STANDARD_PINS \
Hui Wang11580292015-08-03 11:03:49 +08007245 {0x14, 0x90170110}
Kailang Yange1e62b92015-04-08 16:01:22 +08007246
David Henningssonfea185e2014-09-03 10:23:04 +02007247#define ALC290_STANDARD_PINS \
Hui Wang11580292015-08-03 11:03:49 +08007248 {0x12, 0x99a30130}
David Henningssonfea185e2014-09-03 10:23:04 +02007249
7250#define ALC292_STANDARD_PINS \
7251 {0x14, 0x90170110}, \
Hui Wang11580292015-08-03 11:03:49 +08007252 {0x15, 0x0221401f}
Kailang Yang977e6272015-05-18 15:31:20 +08007253
Hui Wang3f6409702016-09-11 11:26:16 +08007254#define ALC295_STANDARD_PINS \
7255 {0x12, 0xb7a60130}, \
7256 {0x14, 0x90170110}, \
Hui Wang3f6409702016-09-11 11:26:16 +08007257 {0x21, 0x04211020}
7258
Woodrow Shen703867e2015-08-05 12:34:12 +08007259#define ALC298_STANDARD_PINS \
7260 {0x12, 0x90a60130}, \
7261 {0x21, 0x03211020}
7262
Hui Wange1918932014-05-26 16:22:44 +08007263static const struct snd_hda_pin_quirk alc269_pin_fixup_tbl[] = {
Kailang Yang8a328ac2018-08-21 16:54:41 +08007264 SND_HDA_PIN_QUIRK(0x10ec0221, 0x103c, "HP Workstation", ALC221_FIXUP_HP_HEADSET_MIC,
7265 {0x14, 0x01014020},
7266 {0x17, 0x90170110},
7267 {0x18, 0x02a11030},
7268 {0x19, 0x0181303F},
7269 {0x21, 0x0221102f}),
Chris Chiu5824ce82017-02-28 14:17:11 -06007270 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1025, "Acer", ALC255_FIXUP_ACER_MIC_NO_PRESENCE,
7271 {0x12, 0x90a601c0},
7272 {0x14, 0x90171120},
7273 {0x21, 0x02211030}),
Chris Chiu615966a2017-02-28 14:17:12 -06007274 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1043, "ASUS", ALC255_FIXUP_ASUS_MIC_NO_PRESENCE,
7275 {0x14, 0x90170110},
7276 {0x1b, 0x90a70130},
7277 {0x21, 0x03211020}),
7278 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1043, "ASUS", ALC255_FIXUP_ASUS_MIC_NO_PRESENCE,
7279 {0x1a, 0x90a70130},
7280 {0x1b, 0x90170110},
7281 {0x21, 0x03211020}),
David Henningsson2ae95572016-02-25 09:37:05 +01007282 SND_HDA_PIN_QUIRK(0x10ec0225, 0x1028, "Dell", ALC225_FIXUP_DELL1_MIC_NO_PRESENCE,
Kailang Yangcfc5a842016-02-03 15:20:39 +08007283 ALC225_STANDARD_PINS,
Hui Wang8a132092016-07-08 14:26:57 +08007284 {0x12, 0xb7a60130},
Kailang Yangcfc5a842016-02-03 15:20:39 +08007285 {0x14, 0x901701a0}),
David Henningsson2ae95572016-02-25 09:37:05 +01007286 SND_HDA_PIN_QUIRK(0x10ec0225, 0x1028, "Dell", ALC225_FIXUP_DELL1_MIC_NO_PRESENCE,
Kailang Yangcfc5a842016-02-03 15:20:39 +08007287 ALC225_STANDARD_PINS,
Hui Wang8a132092016-07-08 14:26:57 +08007288 {0x12, 0xb7a60130},
Kailang Yangcfc5a842016-02-03 15:20:39 +08007289 {0x14, 0x901701b0}),
Hui Wang8a132092016-07-08 14:26:57 +08007290 SND_HDA_PIN_QUIRK(0x10ec0225, 0x1028, "Dell", ALC225_FIXUP_DELL1_MIC_NO_PRESENCE,
7291 ALC225_STANDARD_PINS,
7292 {0x12, 0xb7a60150},
7293 {0x14, 0x901701a0}),
7294 SND_HDA_PIN_QUIRK(0x10ec0225, 0x1028, "Dell", ALC225_FIXUP_DELL1_MIC_NO_PRESENCE,
7295 ALC225_STANDARD_PINS,
7296 {0x12, 0xb7a60150},
7297 {0x14, 0x901701b0}),
7298 SND_HDA_PIN_QUIRK(0x10ec0225, 0x1028, "Dell", ALC225_FIXUP_DELL1_MIC_NO_PRESENCE,
7299 ALC225_STANDARD_PINS,
7300 {0x12, 0xb7a60130},
7301 {0x1b, 0x90170110}),
Kai-Heng Feng0ce48e12017-11-20 02:29:13 -05007302 SND_HDA_PIN_QUIRK(0x10ec0233, 0x8086, "Intel NUC Skull Canyon", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE,
7303 {0x1b, 0x01111010},
7304 {0x1e, 0x01451130},
7305 {0x21, 0x02211020}),
Hui Wang986376b2018-05-30 12:33:07 +08007306 SND_HDA_PIN_QUIRK(0x10ec0235, 0x17aa, "Lenovo", ALC233_FIXUP_LENOVO_LINE2_MIC_HOTKEY,
7307 {0x12, 0x90a60140},
7308 {0x14, 0x90170110},
7309 {0x19, 0x02a11030},
7310 {0x21, 0x02211020}),
Hui Wange41fc8c2018-06-25 14:40:56 +08007311 SND_HDA_PIN_QUIRK(0x10ec0235, 0x17aa, "Lenovo", ALC294_FIXUP_LENOVO_MIC_LOCATION,
7312 {0x14, 0x90170110},
7313 {0x19, 0x02a11030},
7314 {0x1a, 0x02a11040},
7315 {0x1b, 0x01014020},
7316 {0x21, 0x0221101f}),
Hui Wangc6b17f12018-07-06 15:14:11 +08007317 SND_HDA_PIN_QUIRK(0x10ec0235, 0x17aa, "Lenovo", ALC294_FIXUP_LENOVO_MIC_LOCATION,
7318 {0x14, 0x90170110},
Hui Wangd06fb562018-10-10 11:57:25 +08007319 {0x19, 0x02a11030},
7320 {0x1a, 0x02a11040},
7321 {0x1b, 0x01011020},
7322 {0x21, 0x0221101f}),
7323 SND_HDA_PIN_QUIRK(0x10ec0235, 0x17aa, "Lenovo", ALC294_FIXUP_LENOVO_MIC_LOCATION,
7324 {0x14, 0x90170110},
Hui Wangc6b17f12018-07-06 15:14:11 +08007325 {0x19, 0x02a11020},
7326 {0x1a, 0x02a11030},
7327 {0x21, 0x0221101f}),
Hui Wangf2657882017-10-24 16:53:34 +08007328 SND_HDA_PIN_QUIRK(0x10ec0236, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
7329 {0x12, 0x90a60140},
7330 {0x14, 0x90170110},
7331 {0x21, 0x02211020}),
7332 SND_HDA_PIN_QUIRK(0x10ec0236, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
7333 {0x12, 0x90a60140},
7334 {0x14, 0x90170150},
7335 {0x21, 0x02211020}),
Hui Wangb26e36b2019-04-17 16:10:32 +08007336 SND_HDA_PIN_QUIRK(0x10ec0236, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
7337 {0x21, 0x02211020}),
Kailang Yang0a29c572019-04-24 16:34:25 +08007338 SND_HDA_PIN_QUIRK(0x10ec0236, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
7339 {0x12, 0x40000000},
7340 {0x14, 0x90170110},
7341 {0x21, 0x02211020}),
Hui Wangc77900e2014-09-03 11:31:07 +08007342 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL2_MIC_NO_PRESENCE,
Hui Wangc77900e2014-09-03 11:31:07 +08007343 {0x14, 0x90170110},
Hui Wangc77900e2014-09-03 11:31:07 +08007344 {0x21, 0x02211020}),
David Henningsson76c21322014-06-24 14:46:54 +02007345 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wang86c72d12016-05-25 12:12:32 +08007346 {0x14, 0x90170130},
7347 {0x21, 0x02211040}),
7348 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
David Henningsson76c21322014-06-24 14:46:54 +02007349 {0x12, 0x90a60140},
7350 {0x14, 0x90170110},
David Henningsson76c21322014-06-24 14:46:54 +02007351 {0x21, 0x02211020}),
7352 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
7353 {0x12, 0x90a60160},
7354 {0x14, 0x90170120},
David Henningsson76c21322014-06-24 14:46:54 +02007355 {0x21, 0x02211030}),
7356 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wang392c9da2016-09-26 10:59:38 +08007357 {0x14, 0x90170110},
7358 {0x1b, 0x02011020},
7359 {0x21, 0x0221101f}),
7360 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wang6aecd872016-10-20 14:03:33 +08007361 {0x14, 0x90170110},
7362 {0x1b, 0x01011020},
7363 {0x21, 0x0221101f}),
7364 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
David Henningssoncba59972015-07-22 10:00:25 +02007365 {0x14, 0x90170130},
David Henningssoncba59972015-07-22 10:00:25 +02007366 {0x1b, 0x01014020},
David Henningssoncba59972015-07-22 10:00:25 +02007367 {0x21, 0x0221103f}),
7368 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wang59ec4b52016-08-04 15:28:04 +08007369 {0x14, 0x90170130},
Hui Wang6aecd872016-10-20 14:03:33 +08007370 {0x1b, 0x01011020},
7371 {0x21, 0x0221103f}),
7372 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
7373 {0x14, 0x90170130},
Hui Wang59ec4b52016-08-04 15:28:04 +08007374 {0x1b, 0x02011020},
7375 {0x21, 0x0221103f}),
7376 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Woodrow Shene9c28e12015-07-25 10:36:16 +08007377 {0x14, 0x90170150},
Woodrow Shene9c28e12015-07-25 10:36:16 +08007378 {0x1b, 0x02011020},
Woodrow Shene9c28e12015-07-25 10:36:16 +08007379 {0x21, 0x0221105f}),
7380 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Woodrow Shene9c28e12015-07-25 10:36:16 +08007381 {0x14, 0x90170110},
Woodrow Shene9c28e12015-07-25 10:36:16 +08007382 {0x1b, 0x01014020},
Woodrow Shene9c28e12015-07-25 10:36:16 +08007383 {0x21, 0x0221101f}),
7384 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
David Henningsson76c21322014-06-24 14:46:54 +02007385 {0x12, 0x90a60160},
7386 {0x14, 0x90170120},
7387 {0x17, 0x90170140},
David Henningsson76c21322014-06-24 14:46:54 +02007388 {0x21, 0x0321102f}),
7389 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
7390 {0x12, 0x90a60160},
7391 {0x14, 0x90170130},
David Henningsson76c21322014-06-24 14:46:54 +02007392 {0x21, 0x02211040}),
7393 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
7394 {0x12, 0x90a60160},
7395 {0x14, 0x90170140},
David Henningsson76c21322014-06-24 14:46:54 +02007396 {0x21, 0x02211050}),
7397 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
7398 {0x12, 0x90a60170},
7399 {0x14, 0x90170120},
David Henningsson76c21322014-06-24 14:46:54 +02007400 {0x21, 0x02211030}),
7401 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
7402 {0x12, 0x90a60170},
7403 {0x14, 0x90170130},
David Henningsson76c21322014-06-24 14:46:54 +02007404 {0x21, 0x02211040}),
Hui Wang70658b92015-03-06 14:03:57 +08007405 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wang70658b92015-03-06 14:03:57 +08007406 {0x12, 0x90a60170},
Hui Wang0a1f90a2016-01-13 11:51:38 +08007407 {0x14, 0x90171130},
7408 {0x21, 0x02211040}),
7409 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
7410 {0x12, 0x90a60170},
Hui Wang70658b92015-03-06 14:03:57 +08007411 {0x14, 0x90170140},
Hui Wang70658b92015-03-06 14:03:57 +08007412 {0x21, 0x02211050}),
David Henningsson9b5a4e32015-05-11 14:04:14 +02007413 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell Inspiron 5548", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
David Henningsson9b5a4e32015-05-11 14:04:14 +02007414 {0x12, 0x90a60180},
7415 {0x14, 0x90170130},
David Henningsson9b5a4e32015-05-11 14:04:14 +02007416 {0x21, 0x02211040}),
AceLan Kaof90d83b2016-06-03 14:45:25 +08007417 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell Inspiron 5565", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
7418 {0x12, 0x90a60180},
7419 {0x14, 0x90170120},
7420 {0x21, 0x02211030}),
Hui Wang989dbe42016-11-23 16:05:38 +08007421 SND_HDA_PIN_QUIRK(0x10ec0255, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
7422 {0x1b, 0x01011020},
7423 {0x21, 0x02211010}),
Kailang Yang7081adf2015-03-30 17:05:37 +08007424 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wang285d5dd2017-12-22 11:17:45 +08007425 {0x12, 0x90a60130},
7426 {0x14, 0x90170110},
7427 {0x1b, 0x01011020},
7428 {0x21, 0x0221101f}),
7429 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Kailang Yang81a12312015-05-28 16:48:22 +08007430 {0x12, 0x90a60160},
7431 {0x14, 0x90170120},
Kailang Yang81a12312015-05-28 16:48:22 +08007432 {0x21, 0x02211030}),
7433 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Woodrow Shenf83c3292016-06-24 15:58:34 +08007434 {0x12, 0x90a60170},
7435 {0x14, 0x90170120},
7436 {0x21, 0x02211030}),
Shrirang Bagul311042d2016-08-29 15:19:27 +08007437 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1028, "Dell Inspiron 5468", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
7438 {0x12, 0x90a60180},
7439 {0x14, 0x90170120},
7440 {0x21, 0x02211030}),
Woodrow Shenf83c3292016-06-24 15:58:34 +08007441 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wang3f6409702016-09-11 11:26:16 +08007442 {0x12, 0xb7a60130},
7443 {0x14, 0x90170110},
7444 {0x21, 0x02211020}),
David Henningssoncf51eb92014-10-30 08:26:02 +01007445 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wang3f2f7c52018-01-29 14:23:15 +08007446 {0x12, 0x90a60130},
7447 {0x14, 0x90170110},
7448 {0x14, 0x01011020},
7449 {0x21, 0x0221101f}),
7450 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
David Henningssoncf51eb92014-10-30 08:26:02 +01007451 ALC256_STANDARD_PINS),
Hui Wangb26e36b2019-04-17 16:10:32 +08007452 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
7453 {0x14, 0x90170110},
7454 {0x1b, 0x01011020},
7455 {0x21, 0x0221101f}),
Chris Chiuc1732ed2017-02-28 14:17:13 -06007456 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1043, "ASUS", ALC256_FIXUP_ASUS_MIC,
7457 {0x14, 0x90170110},
7458 {0x1b, 0x90a70130},
7459 {0x21, 0x04211020}),
7460 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1043, "ASUS", ALC256_FIXUP_ASUS_MIC,
7461 {0x14, 0x90170110},
7462 {0x1b, 0x90a70130},
7463 {0x21, 0x03211020}),
Jian-Hong Pane1037352019-03-22 11:37:18 +08007464 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1043, "ASUS", ALC256_FIXUP_ASUS_MIC_NO_PRESENCE,
Chris Chiua806ef12019-03-22 11:37:20 +08007465 {0x12, 0x90a60130},
7466 {0x14, 0x90170110},
7467 {0x21, 0x03211020}),
7468 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1043, "ASUS", ALC256_FIXUP_ASUS_MIC_NO_PRESENCE,
Jian-Hong Pan6ac371a2019-03-22 11:37:22 +08007469 {0x12, 0x90a60130},
7470 {0x14, 0x90170110},
7471 {0x21, 0x04211020}),
7472 SND_HDA_PIN_QUIRK(0x10ec0256, 0x1043, "ASUS", ALC256_FIXUP_ASUS_MIC_NO_PRESENCE,
Jian-Hong Pane1037352019-03-22 11:37:18 +08007473 {0x1a, 0x90a70130},
7474 {0x1b, 0x90170110},
7475 {0x21, 0x03211020}),
Kailang Yang92266652017-12-14 15:28:58 +08007476 SND_HDA_PIN_QUIRK(0x10ec0274, 0x1028, "Dell", ALC274_FIXUP_DELL_AIO_LINEOUT_VERB,
Hui Wang75ee94b2017-11-09 08:48:08 +08007477 {0x12, 0xb7a60130},
7478 {0x13, 0xb8a61140},
7479 {0x16, 0x90170110},
7480 {0x21, 0x04211020}),
David Henningssoncf51eb92014-10-30 08:26:02 +01007481 SND_HDA_PIN_QUIRK(0x10ec0280, 0x103c, "HP", ALC280_FIXUP_HP_GPIO4,
7482 {0x12, 0x90a60130},
7483 {0x14, 0x90170110},
7484 {0x15, 0x0421101f},
Hui Wang11580292015-08-03 11:03:49 +08007485 {0x1a, 0x04a11020}),
Hui Wang02796612014-09-03 11:31:11 +08007486 SND_HDA_PIN_QUIRK(0x10ec0280, 0x103c, "HP", ALC269_FIXUP_HP_GPIO_MIC1_LED,
7487 {0x12, 0x90a60140},
Hui Wang02796612014-09-03 11:31:11 +08007488 {0x14, 0x90170110},
7489 {0x15, 0x0421101f},
Hui Wang02796612014-09-03 11:31:11 +08007490 {0x18, 0x02811030},
Hui Wang02796612014-09-03 11:31:11 +08007491 {0x1a, 0x04a1103f},
Hui Wang11580292015-08-03 11:03:49 +08007492 {0x1b, 0x02011020}),
David Henningsson42304472014-07-22 11:42:17 +02007493 SND_HDA_PIN_QUIRK(0x10ec0282, 0x103c, "HP 15 Touchsmart", ALC269_FIXUP_HP_MUTE_LED_MIC1,
David Henningssonaec856d2014-09-03 10:23:05 +02007494 ALC282_STANDARD_PINS,
David Henningsson42304472014-07-22 11:42:17 +02007495 {0x12, 0x99a30130},
David Henningsson42304472014-07-22 11:42:17 +02007496 {0x19, 0x03a11020},
David Henningsson42304472014-07-22 11:42:17 +02007497 {0x21, 0x0321101f}),
Hui Wang2c609992014-09-03 11:31:08 +08007498 SND_HDA_PIN_QUIRK(0x10ec0282, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
David Henningssonaec856d2014-09-03 10:23:05 +02007499 ALC282_STANDARD_PINS,
Hui Wang2c609992014-09-03 11:31:08 +08007500 {0x12, 0x99a30130},
Hui Wang2c609992014-09-03 11:31:08 +08007501 {0x19, 0x03a11020},
Hui Wang2c609992014-09-03 11:31:08 +08007502 {0x21, 0x03211040}),
7503 SND_HDA_PIN_QUIRK(0x10ec0282, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
David Henningssonaec856d2014-09-03 10:23:05 +02007504 ALC282_STANDARD_PINS,
Hui Wang2c609992014-09-03 11:31:08 +08007505 {0x12, 0x99a30130},
Hui Wang2c609992014-09-03 11:31:08 +08007506 {0x19, 0x03a11030},
Hui Wang2c609992014-09-03 11:31:08 +08007507 {0x21, 0x03211020}),
7508 SND_HDA_PIN_QUIRK(0x10ec0282, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
David Henningssonaec856d2014-09-03 10:23:05 +02007509 ALC282_STANDARD_PINS,
Hui Wang2c609992014-09-03 11:31:08 +08007510 {0x12, 0x99a30130},
Hui Wang2c609992014-09-03 11:31:08 +08007511 {0x19, 0x04a11020},
Hui Wang2c609992014-09-03 11:31:08 +08007512 {0x21, 0x0421101f}),
Hui Wang200afc02014-09-03 11:31:10 +08007513 SND_HDA_PIN_QUIRK(0x10ec0282, 0x103c, "HP", ALC269_FIXUP_HP_LINE1_MIC1_LED,
David Henningssonaec856d2014-09-03 10:23:05 +02007514 ALC282_STANDARD_PINS,
Hui Wang200afc02014-09-03 11:31:10 +08007515 {0x12, 0x90a60140},
Hui Wang200afc02014-09-03 11:31:10 +08007516 {0x19, 0x04a11030},
Hui Wang200afc02014-09-03 11:31:10 +08007517 {0x21, 0x04211020}),
David Henningsson76c21322014-06-24 14:46:54 +02007518 SND_HDA_PIN_QUIRK(0x10ec0283, 0x1028, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE,
David Henningssonaec856d2014-09-03 10:23:05 +02007519 ALC282_STANDARD_PINS,
David Henningsson76c21322014-06-24 14:46:54 +02007520 {0x12, 0x90a60130},
David Henningsson76c21322014-06-24 14:46:54 +02007521 {0x21, 0x0321101f}),
7522 SND_HDA_PIN_QUIRK(0x10ec0283, 0x1028, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE,
7523 {0x12, 0x90a60160},
7524 {0x14, 0x90170120},
David Henningsson76c21322014-06-24 14:46:54 +02007525 {0x21, 0x02211030}),
Hui Wangbc262172014-09-03 11:31:05 +08007526 SND_HDA_PIN_QUIRK(0x10ec0283, 0x1028, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE,
David Henningssonaec856d2014-09-03 10:23:05 +02007527 ALC282_STANDARD_PINS,
Hui Wangbc262172014-09-03 11:31:05 +08007528 {0x12, 0x90a60130},
Hui Wangbc262172014-09-03 11:31:05 +08007529 {0x19, 0x03a11020},
Hui Wangbc262172014-09-03 11:31:05 +08007530 {0x21, 0x0321101f}),
Hui Wangc8c6ee62019-02-14 11:41:33 +08007531 SND_HDA_PIN_QUIRK(0x10ec0285, 0x17aa, "Lenovo", ALC285_FIXUP_LENOVO_PC_BEEP_IN_NOISE,
Hui Wangc4cfcf62018-11-26 14:17:16 +08007532 {0x12, 0x90a60130},
7533 {0x14, 0x90170110},
7534 {0x19, 0x04a11040},
7535 {0x21, 0x04211020}),
Chris Chiu33aaebd2018-12-05 14:48:53 +08007536 SND_HDA_PIN_QUIRK(0x10ec0286, 0x1025, "Acer", ALC286_FIXUP_ACER_AIO_MIC_NO_PRESENCE,
7537 {0x12, 0x90a60130},
7538 {0x17, 0x90170110},
7539 {0x21, 0x02211020}),
Takashi Iwaid44a6862018-06-19 23:04:03 +02007540 SND_HDA_PIN_QUIRK(0x10ec0288, 0x1028, "Dell", ALC288_FIXUP_DELL1_MIC_NO_PRESENCE,
Kailang Yange1e62b92015-04-08 16:01:22 +08007541 {0x12, 0x90a60120},
Kailang Yange1e62b92015-04-08 16:01:22 +08007542 {0x14, 0x90170110},
Kailang Yange1e62b92015-04-08 16:01:22 +08007543 {0x21, 0x0321101f}),
Hui Wangd5078192018-03-02 13:05:36 +08007544 SND_HDA_PIN_QUIRK(0x10ec0289, 0x1028, "Dell", ALC269_FIXUP_DELL4_MIC_NO_PRESENCE,
Hui Wang3f2f7c52018-01-29 14:23:15 +08007545 {0x12, 0xb7a60130},
7546 {0x14, 0x90170110},
7547 {0x21, 0x04211020}),
Hui Wange4442bc2014-09-03 11:31:09 +08007548 SND_HDA_PIN_QUIRK(0x10ec0290, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
David Henningssonaec856d2014-09-03 10:23:05 +02007549 ALC290_STANDARD_PINS,
Hui Wange4442bc2014-09-03 11:31:09 +08007550 {0x15, 0x04211040},
Hui Wange4442bc2014-09-03 11:31:09 +08007551 {0x18, 0x90170112},
Hui Wang11580292015-08-03 11:03:49 +08007552 {0x1a, 0x04a11020}),
Hui Wange4442bc2014-09-03 11:31:09 +08007553 SND_HDA_PIN_QUIRK(0x10ec0290, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
David Henningssonaec856d2014-09-03 10:23:05 +02007554 ALC290_STANDARD_PINS,
Hui Wange4442bc2014-09-03 11:31:09 +08007555 {0x15, 0x04211040},
Hui Wange4442bc2014-09-03 11:31:09 +08007556 {0x18, 0x90170110},
Hui Wang11580292015-08-03 11:03:49 +08007557 {0x1a, 0x04a11020}),
Hui Wange4442bc2014-09-03 11:31:09 +08007558 SND_HDA_PIN_QUIRK(0x10ec0290, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
David Henningssonaec856d2014-09-03 10:23:05 +02007559 ALC290_STANDARD_PINS,
Hui Wange4442bc2014-09-03 11:31:09 +08007560 {0x15, 0x0421101f},
Hui Wang11580292015-08-03 11:03:49 +08007561 {0x1a, 0x04a11020}),
Hui Wange4442bc2014-09-03 11:31:09 +08007562 SND_HDA_PIN_QUIRK(0x10ec0290, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
David Henningssonaec856d2014-09-03 10:23:05 +02007563 ALC290_STANDARD_PINS,
Hui Wange4442bc2014-09-03 11:31:09 +08007564 {0x15, 0x04211020},
Hui Wang11580292015-08-03 11:03:49 +08007565 {0x1a, 0x04a11040}),
Hui Wange4442bc2014-09-03 11:31:09 +08007566 SND_HDA_PIN_QUIRK(0x10ec0290, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
David Henningssonaec856d2014-09-03 10:23:05 +02007567 ALC290_STANDARD_PINS,
Hui Wange4442bc2014-09-03 11:31:09 +08007568 {0x14, 0x90170110},
7569 {0x15, 0x04211020},
Hui Wang11580292015-08-03 11:03:49 +08007570 {0x1a, 0x04a11040}),
Hui Wange4442bc2014-09-03 11:31:09 +08007571 SND_HDA_PIN_QUIRK(0x10ec0290, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
David Henningssonaec856d2014-09-03 10:23:05 +02007572 ALC290_STANDARD_PINS,
Hui Wange4442bc2014-09-03 11:31:09 +08007573 {0x14, 0x90170110},
7574 {0x15, 0x04211020},
Hui Wang11580292015-08-03 11:03:49 +08007575 {0x1a, 0x04a11020}),
Hui Wange4442bc2014-09-03 11:31:09 +08007576 SND_HDA_PIN_QUIRK(0x10ec0290, 0x103c, "HP", ALC269_FIXUP_HP_MUTE_LED_MIC1,
David Henningssonaec856d2014-09-03 10:23:05 +02007577 ALC290_STANDARD_PINS,
Hui Wange4442bc2014-09-03 11:31:09 +08007578 {0x14, 0x90170110},
7579 {0x15, 0x0421101f},
Hui Wang11580292015-08-03 11:03:49 +08007580 {0x1a, 0x04a11020}),
Hui Wange8818fa2014-09-03 11:31:04 +08007581 SND_HDA_PIN_QUIRK(0x10ec0292, 0x1028, "Dell", ALC269_FIXUP_DELL2_MIC_NO_PRESENCE,
David Henningssonaec856d2014-09-03 10:23:05 +02007582 ALC292_STANDARD_PINS,
Hui Wange8818fa2014-09-03 11:31:04 +08007583 {0x12, 0x90a60140},
Hui Wange8818fa2014-09-03 11:31:04 +08007584 {0x16, 0x01014020},
Hui Wang11580292015-08-03 11:03:49 +08007585 {0x19, 0x01a19030}),
Hui Wange8818fa2014-09-03 11:31:04 +08007586 SND_HDA_PIN_QUIRK(0x10ec0292, 0x1028, "Dell", ALC269_FIXUP_DELL2_MIC_NO_PRESENCE,
David Henningssonaec856d2014-09-03 10:23:05 +02007587 ALC292_STANDARD_PINS,
Hui Wange8818fa2014-09-03 11:31:04 +08007588 {0x12, 0x90a60140},
Hui Wange8818fa2014-09-03 11:31:04 +08007589 {0x16, 0x01014020},
7590 {0x18, 0x02a19031},
Hui Wang11580292015-08-03 11:03:49 +08007591 {0x19, 0x01a1903e}),
David Henningsson76c21322014-06-24 14:46:54 +02007592 SND_HDA_PIN_QUIRK(0x10ec0292, 0x1028, "Dell", ALC269_FIXUP_DELL3_MIC_NO_PRESENCE,
David Henningssonaec856d2014-09-03 10:23:05 +02007593 ALC292_STANDARD_PINS,
Hui Wang11580292015-08-03 11:03:49 +08007594 {0x12, 0x90a60140}),
David Henningsson76c21322014-06-24 14:46:54 +02007595 SND_HDA_PIN_QUIRK(0x10ec0293, 0x1028, "Dell", ALC293_FIXUP_DELL1_MIC_NO_PRESENCE,
David Henningssonaec856d2014-09-03 10:23:05 +02007596 ALC292_STANDARD_PINS,
David Henningsson76c21322014-06-24 14:46:54 +02007597 {0x13, 0x90a60140},
David Henningsson76c21322014-06-24 14:46:54 +02007598 {0x16, 0x21014020},
Hui Wang11580292015-08-03 11:03:49 +08007599 {0x19, 0x21a19030}),
David Henningssone03fdbd2014-06-27 08:41:59 +02007600 SND_HDA_PIN_QUIRK(0x10ec0293, 0x1028, "Dell", ALC293_FIXUP_DELL1_MIC_NO_PRESENCE,
David Henningssonaec856d2014-09-03 10:23:05 +02007601 ALC292_STANDARD_PINS,
Hui Wang11580292015-08-03 11:03:49 +08007602 {0x13, 0x90a60140}),
Chris Chiud8ae4582018-12-07 17:17:11 +08007603 SND_HDA_PIN_QUIRK(0x10ec0294, 0x1043, "ASUS", ALC294_FIXUP_ASUS_MIC,
7604 {0x14, 0x90170110},
7605 {0x1b, 0x90a70130},
7606 {0x21, 0x04211020}),
Jian-Hong Pan0bea4cc2018-12-07 17:17:13 +08007607 SND_HDA_PIN_QUIRK(0x10ec0294, 0x1043, "ASUS", ALC294_FIXUP_ASUS_SPK,
7608 {0x12, 0x90a60130},
7609 {0x17, 0x90170110},
Jian-Hong Pan8bb37a22019-02-21 17:00:18 +08007610 {0x21, 0x03211020}),
7611 SND_HDA_PIN_QUIRK(0x10ec0294, 0x1043, "ASUS", ALC294_FIXUP_ASUS_SPK,
7612 {0x12, 0x90a60130},
7613 {0x17, 0x90170110},
Jian-Hong Pan0bea4cc2018-12-07 17:17:13 +08007614 {0x21, 0x04211020}),
Takashi Iwai3887c262019-04-30 15:10:01 +02007615 SND_HDA_PIN_QUIRK(0x10ec0295, 0x1043, "ASUS", ALC294_FIXUP_ASUS_SPK,
7616 {0x12, 0x90a60130},
7617 {0x17, 0x90170110},
7618 {0x21, 0x03211020}),
Hui Wang3f6409702016-09-11 11:26:16 +08007619 SND_HDA_PIN_QUIRK(0x10ec0295, 0x1028, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE,
Kailang Yang0a29c572019-04-24 16:34:25 +08007620 {0x14, 0x90170110},
7621 {0x21, 0x04211020}),
Hui Wang3f6409702016-09-11 11:26:16 +08007622 SND_HDA_PIN_QUIRK(0x10ec0295, 0x1028, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE,
Hui Wangf771d5b2016-10-18 10:59:09 +08007623 ALC295_STANDARD_PINS,
7624 {0x17, 0x21014020},
7625 {0x18, 0x21a19030}),
7626 SND_HDA_PIN_QUIRK(0x10ec0295, 0x1028, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE,
7627 ALC295_STANDARD_PINS,
7628 {0x17, 0x21014040},
7629 {0x18, 0x21a19050}),
Hui Wang3f307832017-03-23 10:00:25 +08007630 SND_HDA_PIN_QUIRK(0x10ec0295, 0x1028, "Dell", ALC269_FIXUP_DELL1_MIC_NO_PRESENCE,
7631 ALC295_STANDARD_PINS),
Kailang Yang977e6272015-05-18 15:31:20 +08007632 SND_HDA_PIN_QUIRK(0x10ec0298, 0x1028, "Dell", ALC298_FIXUP_DELL1_MIC_NO_PRESENCE,
Woodrow Shen703867e2015-08-05 12:34:12 +08007633 ALC298_STANDARD_PINS,
Takashi Iwai9f502ff2015-08-05 12:09:45 +02007634 {0x17, 0x90170110}),
7635 SND_HDA_PIN_QUIRK(0x10ec0298, 0x1028, "Dell", ALC298_FIXUP_DELL1_MIC_NO_PRESENCE,
7636 ALC298_STANDARD_PINS,
Woodrow Shen703867e2015-08-05 12:34:12 +08007637 {0x17, 0x90170140}),
7638 SND_HDA_PIN_QUIRK(0x10ec0298, 0x1028, "Dell", ALC298_FIXUP_DELL1_MIC_NO_PRESENCE,
7639 ALC298_STANDARD_PINS,
Takashi Iwai9f502ff2015-08-05 12:09:45 +02007640 {0x17, 0x90170150}),
Kai-Heng Feng9f1bc2c42017-02-16 15:26:54 +08007641 SND_HDA_PIN_QUIRK(0x10ec0298, 0x1028, "Dell", ALC298_FIXUP_SPK_VOLUME,
7642 {0x12, 0xb7a60140},
7643 {0x13, 0xb7a60150},
7644 {0x17, 0x90170110},
7645 {0x1a, 0x03011020},
7646 {0x21, 0x03211030}),
Kailang Yangfcc6c872017-06-29 15:21:27 +08007647 SND_HDA_PIN_QUIRK(0x10ec0299, 0x1028, "Dell", ALC269_FIXUP_DELL4_MIC_NO_PRESENCE,
7648 ALC225_STANDARD_PINS,
7649 {0x12, 0xb7a60130},
Kailang Yangfcc6c872017-06-29 15:21:27 +08007650 {0x17, 0x90170110}),
Hui Wange1918932014-05-26 16:22:44 +08007651 {}
7652};
Takashi Iwai1d045db2011-07-07 18:23:21 +02007653
Takashi Iwai546bb672012-03-07 08:37:19 +01007654static void alc269_fill_coef(struct hda_codec *codec)
Takashi Iwai1d045db2011-07-07 18:23:21 +02007655{
Kailang Yang526af6e2012-03-07 08:25:20 +01007656 struct alc_spec *spec = codec->spec;
Takashi Iwai1d045db2011-07-07 18:23:21 +02007657 int val;
7658
Kailang Yang526af6e2012-03-07 08:25:20 +01007659 if (spec->codec_variant != ALC269_TYPE_ALC269VB)
Takashi Iwai546bb672012-03-07 08:37:19 +01007660 return;
Kailang Yang526af6e2012-03-07 08:25:20 +01007661
Takashi Iwai1bb7e432011-10-17 16:50:59 +02007662 if ((alc_get_coef0(codec) & 0x00ff) < 0x015) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02007663 alc_write_coef_idx(codec, 0xf, 0x960b);
7664 alc_write_coef_idx(codec, 0xe, 0x8817);
7665 }
7666
Takashi Iwai1bb7e432011-10-17 16:50:59 +02007667 if ((alc_get_coef0(codec) & 0x00ff) == 0x016) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02007668 alc_write_coef_idx(codec, 0xf, 0x960b);
7669 alc_write_coef_idx(codec, 0xe, 0x8814);
7670 }
7671
Takashi Iwai1bb7e432011-10-17 16:50:59 +02007672 if ((alc_get_coef0(codec) & 0x00ff) == 0x017) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02007673 /* Power up output pin */
Takashi Iwai98b24882014-08-18 13:47:50 +02007674 alc_update_coef_idx(codec, 0x04, 0, 1<<11);
Takashi Iwai1d045db2011-07-07 18:23:21 +02007675 }
7676
Takashi Iwai1bb7e432011-10-17 16:50:59 +02007677 if ((alc_get_coef0(codec) & 0x00ff) == 0x018) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02007678 val = alc_read_coef_idx(codec, 0xd);
Takashi Iwaif3ee07d2014-08-15 17:35:00 +02007679 if (val != -1 && (val & 0x0c00) >> 10 != 0x1) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02007680 /* Capless ramp up clock control */
7681 alc_write_coef_idx(codec, 0xd, val | (1<<10));
7682 }
7683 val = alc_read_coef_idx(codec, 0x17);
Takashi Iwaif3ee07d2014-08-15 17:35:00 +02007684 if (val != -1 && (val & 0x01c0) >> 6 != 0x4) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02007685 /* Class D power on reset */
7686 alc_write_coef_idx(codec, 0x17, val | (1<<7));
7687 }
7688 }
7689
Takashi Iwai98b24882014-08-18 13:47:50 +02007690 /* HP */
7691 alc_update_coef_idx(codec, 0x4, 0, 1<<11);
Takashi Iwai1d045db2011-07-07 18:23:21 +02007692}
7693
7694/*
7695 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02007696static int patch_alc269(struct hda_codec *codec)
7697{
7698 struct alc_spec *spec;
Takashi Iwai3de95172012-05-07 18:03:15 +02007699 int err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02007700
Takashi Iwai3de95172012-05-07 18:03:15 +02007701 err = alc_alloc_spec(codec, 0x0b);
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02007702 if (err < 0)
Takashi Iwai3de95172012-05-07 18:03:15 +02007703 return err;
7704
7705 spec = codec->spec;
Takashi Iwai08c189f2012-12-19 15:22:24 +01007706 spec->gen.shared_mic_vref_pin = 0x18;
Takashi Iwai8b99aba2015-06-15 11:59:32 +02007707 codec->power_save_node = 1;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02007708
Takashi Iwai225068a2015-05-29 10:42:14 +02007709#ifdef CONFIG_PM
7710 codec->patch_ops.suspend = alc269_suspend;
7711 codec->patch_ops.resume = alc269_resume;
7712#endif
Kailang Yangc2d6af52017-06-21 14:50:54 +08007713 spec->shutup = alc_default_shutup;
7714 spec->init_hook = alc_default_init;
Takashi Iwai225068a2015-05-29 10:42:14 +02007715
Takashi Iwai7639a062015-03-03 10:07:24 +01007716 switch (codec->core.vendor_id) {
Kailang Yang065380f2013-01-10 10:25:48 +01007717 case 0x10ec0269:
Takashi Iwai1d045db2011-07-07 18:23:21 +02007718 spec->codec_variant = ALC269_TYPE_ALC269VA;
Takashi Iwai1bb7e432011-10-17 16:50:59 +02007719 switch (alc_get_coef0(codec) & 0x00f0) {
7720 case 0x0010:
Takashi Iwai5100cd02014-02-15 10:03:19 +01007721 if (codec->bus->pci &&
7722 codec->bus->pci->subsystem_vendor == 0x1025 &&
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02007723 spec->cdefine.platform_type == 1)
Takashi Iwai20ca0c32011-10-17 16:00:35 +02007724 err = alc_codec_rename(codec, "ALC271X");
Takashi Iwai1d045db2011-07-07 18:23:21 +02007725 spec->codec_variant = ALC269_TYPE_ALC269VB;
Takashi Iwai1bb7e432011-10-17 16:50:59 +02007726 break;
7727 case 0x0020:
Takashi Iwai5100cd02014-02-15 10:03:19 +01007728 if (codec->bus->pci &&
7729 codec->bus->pci->subsystem_vendor == 0x17aa &&
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02007730 codec->bus->pci->subsystem_device == 0x21f3)
Takashi Iwai20ca0c32011-10-17 16:00:35 +02007731 err = alc_codec_rename(codec, "ALC3202");
Takashi Iwai1d045db2011-07-07 18:23:21 +02007732 spec->codec_variant = ALC269_TYPE_ALC269VC;
Takashi Iwai1bb7e432011-10-17 16:50:59 +02007733 break;
Kailang Yangadcc70b2012-05-25 08:08:38 +02007734 case 0x0030:
7735 spec->codec_variant = ALC269_TYPE_ALC269VD;
7736 break;
Takashi Iwai1bb7e432011-10-17 16:50:59 +02007737 default:
Takashi Iwai1d045db2011-07-07 18:23:21 +02007738 alc_fix_pll_init(codec, 0x20, 0x04, 15);
Takashi Iwai1bb7e432011-10-17 16:50:59 +02007739 }
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02007740 if (err < 0)
7741 goto error;
Kailang Yangc2d6af52017-06-21 14:50:54 +08007742 spec->shutup = alc269_shutup;
Takashi Iwai546bb672012-03-07 08:37:19 +01007743 spec->init_hook = alc269_fill_coef;
Takashi Iwai1d045db2011-07-07 18:23:21 +02007744 alc269_fill_coef(codec);
Kailang Yang065380f2013-01-10 10:25:48 +01007745 break;
7746
7747 case 0x10ec0280:
7748 case 0x10ec0290:
7749 spec->codec_variant = ALC269_TYPE_ALC280;
7750 break;
7751 case 0x10ec0282:
Kailang Yang065380f2013-01-10 10:25:48 +01007752 spec->codec_variant = ALC269_TYPE_ALC282;
Kailang Yang7b5c7a02014-03-18 16:15:54 +08007753 spec->shutup = alc282_shutup;
7754 spec->init_hook = alc282_init;
Kailang Yang065380f2013-01-10 10:25:48 +01007755 break;
Kailang Yang2af02be2013-08-22 10:03:50 +02007756 case 0x10ec0233:
7757 case 0x10ec0283:
7758 spec->codec_variant = ALC269_TYPE_ALC283;
7759 spec->shutup = alc283_shutup;
7760 spec->init_hook = alc283_init;
7761 break;
Kailang Yang065380f2013-01-10 10:25:48 +01007762 case 0x10ec0284:
7763 case 0x10ec0292:
7764 spec->codec_variant = ALC269_TYPE_ALC284;
7765 break;
Kailang Yang161ebf22013-10-24 11:36:33 +02007766 case 0x10ec0293:
Kailang Yang4731d5d2017-06-30 15:22:57 +08007767 spec->codec_variant = ALC269_TYPE_ALC293;
Kailang Yang161ebf22013-10-24 11:36:33 +02007768 break;
Kailang Yang7fc7d042013-04-25 11:04:43 +02007769 case 0x10ec0286:
Kailang Yang7c665932014-04-14 15:09:44 +08007770 case 0x10ec0288:
Kailang Yang7fc7d042013-04-25 11:04:43 +02007771 spec->codec_variant = ALC269_TYPE_ALC286;
7772 break;
Kailang Yang506b62c2014-12-18 17:07:44 +08007773 case 0x10ec0298:
7774 spec->codec_variant = ALC269_TYPE_ALC298;
7775 break;
Kailang Yangea04a1d2018-04-25 15:31:52 +08007776 case 0x10ec0235:
Kailang Yang1d04c9d2013-10-24 11:35:18 +02007777 case 0x10ec0255:
7778 spec->codec_variant = ALC269_TYPE_ALC255;
Kailang Yangab3b8e52018-04-25 16:05:27 +08007779 spec->shutup = alc256_shutup;
7780 spec->init_hook = alc256_init;
Kailang Yang1d04c9d2013-10-24 11:35:18 +02007781 break;
Kailang Yang736f20a2017-10-20 15:06:34 +08007782 case 0x10ec0236:
Kailang Yang4344aec2014-12-17 17:39:05 +08007783 case 0x10ec0256:
7784 spec->codec_variant = ALC269_TYPE_ALC256;
Kailang Yang4a219ef2017-06-16 16:54:35 +08007785 spec->shutup = alc256_shutup;
7786 spec->init_hook = alc256_init;
David Henningsson7d1b6e22015-04-21 10:48:46 +02007787 spec->gen.mixer_nid = 0; /* ALC256 does not have any loopback mixer path */
Kailang Yangd32b6662015-04-23 15:10:53 +08007788 alc_update_coef_idx(codec, 0x36, 1 << 13, 1 << 5); /* Switch pcbeep path to Line in path*/
Kailang Yang4344aec2014-12-17 17:39:05 +08007789 break;
Kailang Yangf429e7e2017-12-05 15:38:24 +08007790 case 0x10ec0257:
7791 spec->codec_variant = ALC269_TYPE_ALC257;
Kailang Yang88d42b22018-03-14 16:08:57 +08007792 spec->shutup = alc256_shutup;
7793 spec->init_hook = alc256_init;
Kailang Yangf429e7e2017-12-05 15:38:24 +08007794 spec->gen.mixer_nid = 0;
7795 break;
Kailang Yang0a6f0602017-06-30 16:00:48 +08007796 case 0x10ec0215:
7797 case 0x10ec0285:
7798 case 0x10ec0289:
7799 spec->codec_variant = ALC269_TYPE_ALC215;
Kailang Yang1b6832be2018-01-17 15:32:03 +08007800 spec->shutup = alc225_shutup;
7801 spec->init_hook = alc225_init;
Kailang Yang0a6f0602017-06-30 16:00:48 +08007802 spec->gen.mixer_nid = 0;
7803 break;
Kailang Yang42314302016-02-03 15:03:50 +08007804 case 0x10ec0225:
Kailang Yang7d727862016-05-24 16:46:07 +08007805 case 0x10ec0295:
Kailang Yang28f1f9b2017-01-04 14:49:07 +08007806 case 0x10ec0299:
Kailang Yang42314302016-02-03 15:03:50 +08007807 spec->codec_variant = ALC269_TYPE_ALC225;
Kailang Yangda911b12018-01-05 16:50:08 +08007808 spec->shutup = alc225_shutup;
7809 spec->init_hook = alc225_init;
Takashi Iwaic1350bf2017-12-27 09:03:45 +01007810 spec->gen.mixer_nid = 0; /* no loopback on ALC225, ALC295 and ALC299 */
Kailang Yang42314302016-02-03 15:03:50 +08007811 break;
Kailang Yangdcd4f0d2016-05-04 15:50:18 +08007812 case 0x10ec0234:
7813 case 0x10ec0274:
7814 case 0x10ec0294:
7815 spec->codec_variant = ALC269_TYPE_ALC294;
Hui Wang532a7782017-06-26 12:30:32 +08007816 spec->gen.mixer_nid = 0; /* ALC2x4 does not have any loopback mixer path */
Kailang Yang71683c32017-06-20 16:33:50 +08007817 alc_update_coef_idx(codec, 0x6b, 0x0018, (1<<4) | (1<<3)); /* UAJ MIC Vref control by verb */
Kailang Yang693abe12019-01-29 15:38:21 +08007818 spec->init_hook = alc294_init;
Kailang Yangdcd4f0d2016-05-04 15:50:18 +08007819 break;
Kailang Yang1078bef2018-11-08 16:36:15 +08007820 case 0x10ec0300:
7821 spec->codec_variant = ALC269_TYPE_ALC300;
7822 spec->gen.mixer_nid = 0; /* no loopback on ALC300 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02007823 break;
Kailang Yang6fbae352016-05-30 16:44:20 +08007824 case 0x10ec0700:
7825 case 0x10ec0701:
7826 case 0x10ec0703:
7827 spec->codec_variant = ALC269_TYPE_ALC700;
7828 spec->gen.mixer_nid = 0; /* ALC700 does not have any loopback mixer path */
Kailang Yang2d7fe612017-11-22 15:21:32 +08007829 alc_update_coef_idx(codec, 0x4a, 1 << 15, 0); /* Combo jack auto trigger control */
Kailang Yang693abe12019-01-29 15:38:21 +08007830 spec->init_hook = alc294_init;
Kailang Yang6fbae352016-05-30 16:44:20 +08007831 break;
7832
Takashi Iwai1d045db2011-07-07 18:23:21 +02007833 }
7834
Kailang Yangad60d502013-06-28 12:03:01 +02007835 if (snd_hda_codec_read(codec, 0x51, 0, AC_VERB_PARAMETERS, 0) == 0x10ec5505) {
Kailang Yang97a26572013-11-29 00:35:26 -05007836 spec->has_alc5505_dsp = 1;
Kailang Yangad60d502013-06-28 12:03:01 +02007837 spec->init_hook = alc5505_dsp_init;
7838 }
7839
Takashi Iwaic9af7532019-05-10 11:01:43 +02007840 alc_pre_init(codec);
7841
Takashi Iwaiefe55732018-06-15 11:55:02 +02007842 snd_hda_pick_fixup(codec, alc269_fixup_models,
7843 alc269_fixup_tbl, alc269_fixups);
7844 snd_hda_pick_pin_fixup(codec, alc269_pin_fixup_tbl, alc269_fixups);
7845 snd_hda_pick_fixup(codec, NULL, alc269_fixup_vendor_tbl,
7846 alc269_fixups);
7847 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
7848
7849 alc_auto_parse_customize_define(codec);
7850
7851 if (has_cdefine_beep(codec))
7852 spec->gen.beep_nid = 0x01;
7853
Takashi Iwaia4297b52011-08-23 18:40:12 +02007854 /* automatic parse from the BIOS config */
7855 err = alc269_parse_auto_config(codec);
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02007856 if (err < 0)
7857 goto error;
Takashi Iwai1d045db2011-07-07 18:23:21 +02007858
Takashi Iwaifea80fa2018-06-20 12:52:46 +02007859 if (!spec->gen.no_analog && spec->gen.beep_nid && spec->gen.mixer_nid) {
7860 err = set_beep_amp(spec, spec->gen.mixer_nid, 0x04, HDA_INPUT);
7861 if (err < 0)
7862 goto error;
7863 }
Takashi Iwai1d045db2011-07-07 18:23:21 +02007864
Takashi Iwai1727a772013-01-10 09:52:52 +01007865 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
Takashi Iwai589876e2012-02-20 15:47:55 +01007866
Takashi Iwai1d045db2011-07-07 18:23:21 +02007867 return 0;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02007868
7869 error:
7870 alc_free(codec);
7871 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02007872}
7873
7874/*
7875 * ALC861
7876 */
7877
Takashi Iwai1d045db2011-07-07 18:23:21 +02007878static int alc861_parse_auto_config(struct hda_codec *codec)
7879{
Takashi Iwai1d045db2011-07-07 18:23:21 +02007880 static const hda_nid_t alc861_ignore[] = { 0x1d, 0 };
Takashi Iwai3e6179b2011-07-08 16:55:13 +02007881 static const hda_nid_t alc861_ssids[] = { 0x0e, 0x0f, 0x0b, 0 };
7882 return alc_parse_auto_config(codec, alc861_ignore, alc861_ssids);
Takashi Iwai1d045db2011-07-07 18:23:21 +02007883}
7884
Takashi Iwai1d045db2011-07-07 18:23:21 +02007885/* Pin config fixes */
7886enum {
Takashi Iwaie652f4c2012-02-13 11:56:25 +01007887 ALC861_FIXUP_FSC_AMILO_PI1505,
7888 ALC861_FIXUP_AMP_VREF_0F,
7889 ALC861_FIXUP_NO_JACK_DETECT,
7890 ALC861_FIXUP_ASUS_A6RP,
Takashi Iwai6ddf0fd2013-11-29 12:47:34 +01007891 ALC660_FIXUP_ASUS_W7J,
Takashi Iwai1d045db2011-07-07 18:23:21 +02007892};
7893
Takashi Iwai31150f22012-01-30 10:54:08 +01007894/* On some laptops, VREF of pin 0x0f is abused for controlling the main amp */
7895static void alc861_fixup_asus_amp_vref_0f(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01007896 const struct hda_fixup *fix, int action)
Takashi Iwai31150f22012-01-30 10:54:08 +01007897{
7898 struct alc_spec *spec = codec->spec;
7899 unsigned int val;
7900
Takashi Iwai1727a772013-01-10 09:52:52 +01007901 if (action != HDA_FIXUP_ACT_INIT)
Takashi Iwai31150f22012-01-30 10:54:08 +01007902 return;
Takashi Iwaid3f02d62013-01-10 10:12:22 +01007903 val = snd_hda_codec_get_pin_target(codec, 0x0f);
Takashi Iwai31150f22012-01-30 10:54:08 +01007904 if (!(val & (AC_PINCTL_IN_EN | AC_PINCTL_OUT_EN)))
7905 val |= AC_PINCTL_IN_EN;
7906 val |= AC_PINCTL_VREF_50;
Takashi Iwaicdd03ce2012-04-20 12:34:50 +02007907 snd_hda_set_pin_ctl(codec, 0x0f, val);
Takashi Iwai08c189f2012-12-19 15:22:24 +01007908 spec->gen.keep_vref_in_automute = 1;
Takashi Iwai31150f22012-01-30 10:54:08 +01007909}
7910
Takashi Iwaie652f4c2012-02-13 11:56:25 +01007911/* suppress the jack-detection */
7912static void alc_fixup_no_jack_detect(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01007913 const struct hda_fixup *fix, int action)
Takashi Iwaie652f4c2012-02-13 11:56:25 +01007914{
Takashi Iwai1727a772013-01-10 09:52:52 +01007915 if (action == HDA_FIXUP_ACT_PRE_PROBE)
Takashi Iwaie652f4c2012-02-13 11:56:25 +01007916 codec->no_jack_detect = 1;
Jesper Juhl7d7eb9e2012-04-12 22:11:25 +02007917}
Takashi Iwaie652f4c2012-02-13 11:56:25 +01007918
Takashi Iwai1727a772013-01-10 09:52:52 +01007919static const struct hda_fixup alc861_fixups[] = {
Takashi Iwaie652f4c2012-02-13 11:56:25 +01007920 [ALC861_FIXUP_FSC_AMILO_PI1505] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01007921 .type = HDA_FIXUP_PINS,
7922 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai1d045db2011-07-07 18:23:21 +02007923 { 0x0b, 0x0221101f }, /* HP */
7924 { 0x0f, 0x90170310 }, /* speaker */
7925 { }
7926 }
7927 },
Takashi Iwaie652f4c2012-02-13 11:56:25 +01007928 [ALC861_FIXUP_AMP_VREF_0F] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01007929 .type = HDA_FIXUP_FUNC,
Takashi Iwai31150f22012-01-30 10:54:08 +01007930 .v.func = alc861_fixup_asus_amp_vref_0f,
Takashi Iwai3b25eb62012-01-25 09:55:46 +01007931 },
Takashi Iwaie652f4c2012-02-13 11:56:25 +01007932 [ALC861_FIXUP_NO_JACK_DETECT] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01007933 .type = HDA_FIXUP_FUNC,
Takashi Iwaie652f4c2012-02-13 11:56:25 +01007934 .v.func = alc_fixup_no_jack_detect,
7935 },
7936 [ALC861_FIXUP_ASUS_A6RP] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01007937 .type = HDA_FIXUP_FUNC,
Takashi Iwaie652f4c2012-02-13 11:56:25 +01007938 .v.func = alc861_fixup_asus_amp_vref_0f,
7939 .chained = true,
7940 .chain_id = ALC861_FIXUP_NO_JACK_DETECT,
Takashi Iwai6ddf0fd2013-11-29 12:47:34 +01007941 },
7942 [ALC660_FIXUP_ASUS_W7J] = {
7943 .type = HDA_FIXUP_VERBS,
7944 .v.verbs = (const struct hda_verb[]) {
7945 /* ASUS W7J needs a magic pin setup on unused NID 0x10
7946 * for enabling outputs
7947 */
7948 {0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
7949 { }
7950 },
Takashi Iwaie652f4c2012-02-13 11:56:25 +01007951 }
Takashi Iwai1d045db2011-07-07 18:23:21 +02007952};
7953
7954static const struct snd_pci_quirk alc861_fixup_tbl[] = {
Takashi Iwai6ddf0fd2013-11-29 12:47:34 +01007955 SND_PCI_QUIRK(0x1043, 0x1253, "ASUS W7J", ALC660_FIXUP_ASUS_W7J),
Takashi Iwaie7ca2372013-12-02 15:27:19 +01007956 SND_PCI_QUIRK(0x1043, 0x1263, "ASUS Z35HL", ALC660_FIXUP_ASUS_W7J),
Takashi Iwaie652f4c2012-02-13 11:56:25 +01007957 SND_PCI_QUIRK(0x1043, 0x1393, "ASUS A6Rp", ALC861_FIXUP_ASUS_A6RP),
7958 SND_PCI_QUIRK_VENDOR(0x1043, "ASUS laptop", ALC861_FIXUP_AMP_VREF_0F),
7959 SND_PCI_QUIRK(0x1462, 0x7254, "HP DX2200", ALC861_FIXUP_NO_JACK_DETECT),
7960 SND_PCI_QUIRK(0x1584, 0x2b01, "Haier W18", ALC861_FIXUP_AMP_VREF_0F),
7961 SND_PCI_QUIRK(0x1584, 0x0000, "Uniwill ECS M31EI", ALC861_FIXUP_AMP_VREF_0F),
7962 SND_PCI_QUIRK(0x1734, 0x10c7, "FSC Amilo Pi1505", ALC861_FIXUP_FSC_AMILO_PI1505),
Takashi Iwai1d045db2011-07-07 18:23:21 +02007963 {}
7964};
7965
7966/*
7967 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02007968static int patch_alc861(struct hda_codec *codec)
7969{
7970 struct alc_spec *spec;
Takashi Iwai1d045db2011-07-07 18:23:21 +02007971 int err;
7972
Takashi Iwai3de95172012-05-07 18:03:15 +02007973 err = alc_alloc_spec(codec, 0x15);
7974 if (err < 0)
7975 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02007976
Takashi Iwai3de95172012-05-07 18:03:15 +02007977 spec = codec->spec;
Takashi Iwai7504b6c2013-03-18 11:25:51 +01007978 spec->gen.beep_nid = 0x23;
Takashi Iwai1d045db2011-07-07 18:23:21 +02007979
Takashi Iwai225068a2015-05-29 10:42:14 +02007980#ifdef CONFIG_PM
7981 spec->power_hook = alc_power_eapd;
7982#endif
7983
Takashi Iwaic9af7532019-05-10 11:01:43 +02007984 alc_pre_init(codec);
7985
Takashi Iwai1727a772013-01-10 09:52:52 +01007986 snd_hda_pick_fixup(codec, NULL, alc861_fixup_tbl, alc861_fixups);
7987 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
Takashi Iwai1d045db2011-07-07 18:23:21 +02007988
Takashi Iwaicb4e4822011-08-23 17:34:25 +02007989 /* automatic parse from the BIOS config */
7990 err = alc861_parse_auto_config(codec);
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02007991 if (err < 0)
7992 goto error;
Takashi Iwai1d045db2011-07-07 18:23:21 +02007993
Takashi Iwaifea80fa2018-06-20 12:52:46 +02007994 if (!spec->gen.no_analog) {
7995 err = set_beep_amp(spec, 0x23, 0, HDA_OUTPUT);
7996 if (err < 0)
7997 goto error;
7998 }
Takashi Iwai1d045db2011-07-07 18:23:21 +02007999
Takashi Iwai1727a772013-01-10 09:52:52 +01008000 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
Takashi Iwai589876e2012-02-20 15:47:55 +01008001
Takashi Iwai1d045db2011-07-07 18:23:21 +02008002 return 0;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02008003
8004 error:
8005 alc_free(codec);
8006 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02008007}
8008
8009/*
8010 * ALC861-VD support
8011 *
8012 * Based on ALC882
8013 *
8014 * In addition, an independent DAC
8015 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02008016static int alc861vd_parse_auto_config(struct hda_codec *codec)
8017{
Takashi Iwai1d045db2011-07-07 18:23:21 +02008018 static const hda_nid_t alc861vd_ignore[] = { 0x1d, 0 };
Takashi Iwai3e6179b2011-07-08 16:55:13 +02008019 static const hda_nid_t alc861vd_ssids[] = { 0x15, 0x1b, 0x14, 0 };
8020 return alc_parse_auto_config(codec, alc861vd_ignore, alc861vd_ssids);
Takashi Iwai1d045db2011-07-07 18:23:21 +02008021}
8022
Takashi Iwai1d045db2011-07-07 18:23:21 +02008023enum {
Takashi Iwai8fdcb6f2011-08-23 17:28:55 +02008024 ALC660VD_FIX_ASUS_GPIO1,
8025 ALC861VD_FIX_DALLAS,
Takashi Iwai1d045db2011-07-07 18:23:21 +02008026};
8027
Takashi Iwai8fdcb6f2011-08-23 17:28:55 +02008028/* exclude VREF80 */
8029static void alc861vd_fixup_dallas(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01008030 const struct hda_fixup *fix, int action)
Takashi Iwai8fdcb6f2011-08-23 17:28:55 +02008031{
Takashi Iwai1727a772013-01-10 09:52:52 +01008032 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwaib78562b2012-12-17 20:06:49 +01008033 snd_hda_override_pin_caps(codec, 0x18, 0x00000734);
8034 snd_hda_override_pin_caps(codec, 0x19, 0x0000073c);
Takashi Iwai8fdcb6f2011-08-23 17:28:55 +02008035 }
8036}
8037
Takashi Iwaidf73d832018-06-19 23:05:47 +02008038/* reset GPIO1 */
8039static void alc660vd_fixup_asus_gpio1(struct hda_codec *codec,
8040 const struct hda_fixup *fix, int action)
8041{
8042 struct alc_spec *spec = codec->spec;
8043
8044 if (action == HDA_FIXUP_ACT_PRE_PROBE)
8045 spec->gpio_mask |= 0x02;
8046 alc_fixup_gpio(codec, action, 0x01);
8047}
8048
Takashi Iwai1727a772013-01-10 09:52:52 +01008049static const struct hda_fixup alc861vd_fixups[] = {
Takashi Iwai1d045db2011-07-07 18:23:21 +02008050 [ALC660VD_FIX_ASUS_GPIO1] = {
Takashi Iwaidf73d832018-06-19 23:05:47 +02008051 .type = HDA_FIXUP_FUNC,
8052 .v.func = alc660vd_fixup_asus_gpio1,
Takashi Iwai1d045db2011-07-07 18:23:21 +02008053 },
Takashi Iwai8fdcb6f2011-08-23 17:28:55 +02008054 [ALC861VD_FIX_DALLAS] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01008055 .type = HDA_FIXUP_FUNC,
Takashi Iwai8fdcb6f2011-08-23 17:28:55 +02008056 .v.func = alc861vd_fixup_dallas,
8057 },
Takashi Iwai1d045db2011-07-07 18:23:21 +02008058};
8059
8060static const struct snd_pci_quirk alc861vd_fixup_tbl[] = {
Takashi Iwai8fdcb6f2011-08-23 17:28:55 +02008061 SND_PCI_QUIRK(0x103c, 0x30bf, "HP TX1000", ALC861VD_FIX_DALLAS),
Takashi Iwai1d045db2011-07-07 18:23:21 +02008062 SND_PCI_QUIRK(0x1043, 0x1339, "ASUS A7-K", ALC660VD_FIX_ASUS_GPIO1),
Takashi Iwai8fdcb6f2011-08-23 17:28:55 +02008063 SND_PCI_QUIRK(0x1179, 0xff31, "Toshiba L30-149", ALC861VD_FIX_DALLAS),
Takashi Iwai1d045db2011-07-07 18:23:21 +02008064 {}
8065};
8066
Takashi Iwai1d045db2011-07-07 18:23:21 +02008067/*
8068 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02008069static int patch_alc861vd(struct hda_codec *codec)
8070{
8071 struct alc_spec *spec;
Takashi Iwaicb4e4822011-08-23 17:34:25 +02008072 int err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02008073
Takashi Iwai3de95172012-05-07 18:03:15 +02008074 err = alc_alloc_spec(codec, 0x0b);
8075 if (err < 0)
8076 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02008077
Takashi Iwai3de95172012-05-07 18:03:15 +02008078 spec = codec->spec;
Takashi Iwai7504b6c2013-03-18 11:25:51 +01008079 spec->gen.beep_nid = 0x23;
Takashi Iwai1d045db2011-07-07 18:23:21 +02008080
Takashi Iwai225068a2015-05-29 10:42:14 +02008081 spec->shutup = alc_eapd_shutup;
8082
Takashi Iwaic9af7532019-05-10 11:01:43 +02008083 alc_pre_init(codec);
8084
Takashi Iwai1727a772013-01-10 09:52:52 +01008085 snd_hda_pick_fixup(codec, NULL, alc861vd_fixup_tbl, alc861vd_fixups);
8086 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
Takashi Iwai1d045db2011-07-07 18:23:21 +02008087
Takashi Iwaicb4e4822011-08-23 17:34:25 +02008088 /* automatic parse from the BIOS config */
8089 err = alc861vd_parse_auto_config(codec);
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02008090 if (err < 0)
8091 goto error;
Takashi Iwai1d045db2011-07-07 18:23:21 +02008092
Takashi Iwaifea80fa2018-06-20 12:52:46 +02008093 if (!spec->gen.no_analog) {
8094 err = set_beep_amp(spec, 0x0b, 0x05, HDA_INPUT);
8095 if (err < 0)
8096 goto error;
8097 }
Takashi Iwai1d045db2011-07-07 18:23:21 +02008098
Takashi Iwai1727a772013-01-10 09:52:52 +01008099 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
Takashi Iwai589876e2012-02-20 15:47:55 +01008100
Takashi Iwai1d045db2011-07-07 18:23:21 +02008101 return 0;
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02008102
8103 error:
8104 alc_free(codec);
8105 return err;
Takashi Iwai1d045db2011-07-07 18:23:21 +02008106}
8107
8108/*
8109 * ALC662 support
8110 *
8111 * ALC662 is almost identical with ALC880 but has cleaner and more flexible
8112 * configuration. Each pin widget can choose any input DACs and a mixer.
8113 * Each ADC is connected from a mixer of all inputs. This makes possible
8114 * 6-channel independent captures.
8115 *
8116 * In addition, an independent DAC for the multi-playback (not used in this
8117 * driver yet).
8118 */
Takashi Iwai1d045db2011-07-07 18:23:21 +02008119
8120/*
8121 * BIOS auto configuration
8122 */
8123
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008124static int alc662_parse_auto_config(struct hda_codec *codec)
8125{
Takashi Iwai4c6d72d2011-05-02 11:30:18 +02008126 static const hda_nid_t alc662_ignore[] = { 0x1d, 0 };
Takashi Iwai3e6179b2011-07-08 16:55:13 +02008127 static const hda_nid_t alc663_ssids[] = { 0x15, 0x1b, 0x14, 0x21 };
8128 static const hda_nid_t alc662_ssids[] = { 0x15, 0x1b, 0x14, 0 };
8129 const hda_nid_t *ssids;
Takashi Iwaiee979a142008-09-02 15:42:20 +02008130
Takashi Iwai7639a062015-03-03 10:07:24 +01008131 if (codec->core.vendor_id == 0x10ec0272 || codec->core.vendor_id == 0x10ec0663 ||
8132 codec->core.vendor_id == 0x10ec0665 || codec->core.vendor_id == 0x10ec0670 ||
8133 codec->core.vendor_id == 0x10ec0671)
Takashi Iwai3e6179b2011-07-08 16:55:13 +02008134 ssids = alc663_ssids;
Kailang Yang6227cdc2010-02-25 08:36:52 +01008135 else
Takashi Iwai3e6179b2011-07-08 16:55:13 +02008136 ssids = alc662_ssids;
8137 return alc_parse_auto_config(codec, alc662_ignore, ssids);
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008138}
8139
Todd Broch6be79482010-12-07 16:51:05 -08008140static void alc272_fixup_mario(struct hda_codec *codec,
Takashi Iwai1727a772013-01-10 09:52:52 +01008141 const struct hda_fixup *fix, int action)
Takashi Iwai6fc398c2011-01-13 14:36:37 +01008142{
Takashi Iwai9bb1f062013-01-10 17:14:29 +01008143 if (action != HDA_FIXUP_ACT_PRE_PROBE)
Takashi Iwai6fc398c2011-01-13 14:36:37 +01008144 return;
Todd Broch6be79482010-12-07 16:51:05 -08008145 if (snd_hda_override_amp_caps(codec, 0x2, HDA_OUTPUT,
8146 (0x3b << AC_AMPCAP_OFFSET_SHIFT) |
8147 (0x3b << AC_AMPCAP_NUM_STEPS_SHIFT) |
8148 (0x03 << AC_AMPCAP_STEP_SIZE_SHIFT) |
8149 (0 << AC_AMPCAP_MUTE_SHIFT)))
Takashi Iwai4e76a882014-02-25 12:21:03 +01008150 codec_warn(codec, "failed to override amp caps for NID 0x2\n");
Todd Broch6be79482010-12-07 16:51:05 -08008151}
8152
Takashi Iwai8e383952013-10-30 17:41:12 +01008153static const struct snd_pcm_chmap_elem asus_pcm_2_1_chmaps[] = {
8154 { .channels = 2,
8155 .map = { SNDRV_CHMAP_FL, SNDRV_CHMAP_FR } },
8156 { .channels = 4,
8157 .map = { SNDRV_CHMAP_FL, SNDRV_CHMAP_FR,
8158 SNDRV_CHMAP_NA, SNDRV_CHMAP_LFE } }, /* LFE only on right */
8159 { }
8160};
8161
8162/* override the 2.1 chmap */
Takashi Iwaieb9ca3a2013-11-28 15:24:34 +01008163static void alc_fixup_bass_chmap(struct hda_codec *codec,
Takashi Iwai8e383952013-10-30 17:41:12 +01008164 const struct hda_fixup *fix, int action)
8165{
8166 if (action == HDA_FIXUP_ACT_BUILD) {
8167 struct alc_spec *spec = codec->spec;
Takashi Iwaibbbc7e82015-02-27 17:43:19 +01008168 spec->gen.pcm_rec[0]->stream[0].chmap = asus_pcm_2_1_chmaps;
Takashi Iwai8e383952013-10-30 17:41:12 +01008169 }
8170}
8171
Takashi Iwaibf686652014-01-13 16:18:25 +01008172/* avoid D3 for keeping GPIO up */
8173static unsigned int gpio_led_power_filter(struct hda_codec *codec,
8174 hda_nid_t nid,
8175 unsigned int power_state)
8176{
8177 struct alc_spec *spec = codec->spec;
Takashi Iwaid261eec2018-06-19 22:32:29 +02008178 if (nid == codec->core.afg && power_state == AC_PWRST_D3 && spec->gpio_data)
Takashi Iwaibf686652014-01-13 16:18:25 +01008179 return AC_PWRST_D0;
8180 return power_state;
8181}
8182
Takashi Iwai3e887f32014-01-10 17:50:58 +01008183static void alc662_fixup_led_gpio1(struct hda_codec *codec,
8184 const struct hda_fixup *fix, int action)
8185{
8186 struct alc_spec *spec = codec->spec;
Takashi Iwai3e887f32014-01-10 17:50:58 +01008187
Takashi Iwai01e4a272018-06-19 22:47:30 +02008188 alc_fixup_hp_gpio_led(codec, action, 0x01, 0);
Takashi Iwai3e887f32014-01-10 17:50:58 +01008189 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
Takashi Iwai0f32fd192014-11-19 12:16:14 +01008190 spec->mute_led_polarity = 1;
Takashi Iwaibf686652014-01-13 16:18:25 +01008191 codec->power_filter = gpio_led_power_filter;
Takashi Iwai3e887f32014-01-10 17:50:58 +01008192 }
8193}
8194
Kailang Yangc6790c82016-11-25 16:15:17 +08008195static void alc662_usi_automute_hook(struct hda_codec *codec,
8196 struct hda_jack_callback *jack)
8197{
8198 struct alc_spec *spec = codec->spec;
8199 int vref;
8200 msleep(200);
8201 snd_hda_gen_hp_automute(codec, jack);
8202
8203 vref = spec->gen.hp_jack_present ? PIN_VREF80 : 0;
8204 msleep(100);
8205 snd_hda_codec_write(codec, 0x19, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
8206 vref);
8207}
8208
8209static void alc662_fixup_usi_headset_mic(struct hda_codec *codec,
8210 const struct hda_fixup *fix, int action)
8211{
8212 struct alc_spec *spec = codec->spec;
8213 if (action == HDA_FIXUP_ACT_PRE_PROBE) {
8214 spec->parse_flags |= HDA_PINCFG_HEADSET_MIC;
8215 spec->gen.hp_automute_hook = alc662_usi_automute_hook;
8216 }
8217}
8218
Kailang Yangf3f91852014-10-24 15:43:46 +08008219static struct coef_fw alc668_coefs[] = {
8220 WRITE_COEF(0x01, 0xbebe), WRITE_COEF(0x02, 0xaaaa), WRITE_COEF(0x03, 0x0),
8221 WRITE_COEF(0x04, 0x0180), WRITE_COEF(0x06, 0x0), WRITE_COEF(0x07, 0x0f80),
8222 WRITE_COEF(0x08, 0x0031), WRITE_COEF(0x0a, 0x0060), WRITE_COEF(0x0b, 0x0),
8223 WRITE_COEF(0x0c, 0x7cf7), WRITE_COEF(0x0d, 0x1080), WRITE_COEF(0x0e, 0x7f7f),
8224 WRITE_COEF(0x0f, 0xcccc), WRITE_COEF(0x10, 0xddcc), WRITE_COEF(0x11, 0x0001),
8225 WRITE_COEF(0x13, 0x0), WRITE_COEF(0x14, 0x2aa0), WRITE_COEF(0x17, 0xa940),
8226 WRITE_COEF(0x19, 0x0), WRITE_COEF(0x1a, 0x0), WRITE_COEF(0x1b, 0x0),
8227 WRITE_COEF(0x1c, 0x0), WRITE_COEF(0x1d, 0x0), WRITE_COEF(0x1e, 0x7418),
8228 WRITE_COEF(0x1f, 0x0804), WRITE_COEF(0x20, 0x4200), WRITE_COEF(0x21, 0x0468),
8229 WRITE_COEF(0x22, 0x8ccc), WRITE_COEF(0x23, 0x0250), WRITE_COEF(0x24, 0x7418),
8230 WRITE_COEF(0x27, 0x0), WRITE_COEF(0x28, 0x8ccc), WRITE_COEF(0x2a, 0xff00),
8231 WRITE_COEF(0x2b, 0x8000), WRITE_COEF(0xa7, 0xff00), WRITE_COEF(0xa8, 0x8000),
8232 WRITE_COEF(0xaa, 0x2e17), WRITE_COEF(0xab, 0xa0c0), WRITE_COEF(0xac, 0x0),
8233 WRITE_COEF(0xad, 0x0), WRITE_COEF(0xae, 0x2ac6), WRITE_COEF(0xaf, 0xa480),
8234 WRITE_COEF(0xb0, 0x0), WRITE_COEF(0xb1, 0x0), WRITE_COEF(0xb2, 0x0),
8235 WRITE_COEF(0xb3, 0x0), WRITE_COEF(0xb4, 0x0), WRITE_COEF(0xb5, 0x1040),
8236 WRITE_COEF(0xb6, 0xd697), WRITE_COEF(0xb7, 0x902b), WRITE_COEF(0xb8, 0xd697),
8237 WRITE_COEF(0xb9, 0x902b), WRITE_COEF(0xba, 0xb8ba), WRITE_COEF(0xbb, 0xaaab),
8238 WRITE_COEF(0xbc, 0xaaaf), WRITE_COEF(0xbd, 0x6aaa), WRITE_COEF(0xbe, 0x1c02),
8239 WRITE_COEF(0xc0, 0x00ff), WRITE_COEF(0xc1, 0x0fa6),
8240 {}
8241};
8242
8243static void alc668_restore_default_value(struct hda_codec *codec)
8244{
8245 alc_process_coef_fw(codec, alc668_coefs);
8246}
8247
David Henningsson6cb3b702010-09-09 08:51:44 +02008248enum {
Daniel T Chen2df03512010-10-10 22:39:28 -04008249 ALC662_FIXUP_ASPIRE,
Takashi Iwai3e887f32014-01-10 17:50:58 +01008250 ALC662_FIXUP_LED_GPIO1,
David Henningsson6cb3b702010-09-09 08:51:44 +02008251 ALC662_FIXUP_IDEAPAD,
Todd Broch6be79482010-12-07 16:51:05 -08008252 ALC272_FIXUP_MARIO,
Anisse Astierd2ebd472011-01-20 12:36:21 +01008253 ALC662_FIXUP_CZC_P10T,
David Henningsson94024cd2011-04-29 14:10:55 +02008254 ALC662_FIXUP_SKU_IGNORE,
Takashi Iwaie59ea3e2011-06-29 17:21:00 +02008255 ALC662_FIXUP_HP_RP5800,
Takashi Iwai53c334a2011-08-23 18:27:14 +02008256 ALC662_FIXUP_ASUS_MODE1,
8257 ALC662_FIXUP_ASUS_MODE2,
8258 ALC662_FIXUP_ASUS_MODE3,
8259 ALC662_FIXUP_ASUS_MODE4,
8260 ALC662_FIXUP_ASUS_MODE5,
8261 ALC662_FIXUP_ASUS_MODE6,
8262 ALC662_FIXUP_ASUS_MODE7,
8263 ALC662_FIXUP_ASUS_MODE8,
Takashi Iwai1565cc32012-02-13 12:03:25 +01008264 ALC662_FIXUP_NO_JACK_DETECT,
David Henningssonedfe3bf2012-06-12 13:15:12 +02008265 ALC662_FIXUP_ZOTAC_Z68,
Takashi Iwai125821a2012-06-22 14:30:29 +02008266 ALC662_FIXUP_INV_DMIC,
David Henningsson1f8b46c2015-05-12 14:38:15 +02008267 ALC662_FIXUP_DELL_MIC_NO_PRESENCE,
David Henningsson73bdd592013-04-15 15:44:14 +02008268 ALC668_FIXUP_DELL_MIC_NO_PRESENCE,
David Henningsson1f8b46c2015-05-12 14:38:15 +02008269 ALC662_FIXUP_HEADSET_MODE,
David Henningsson73bdd592013-04-15 15:44:14 +02008270 ALC668_FIXUP_HEADSET_MODE,
David Henningsson8e54b4a2014-02-07 09:31:07 +01008271 ALC662_FIXUP_BASS_MODE4_CHMAP,
David Henningsson61a75f12014-02-07 09:31:08 +01008272 ALC662_FIXUP_BASS_16,
Takashi Iwaia30c9aa2013-11-21 08:00:20 +01008273 ALC662_FIXUP_BASS_1A,
David Henningsson8e54b4a2014-02-07 09:31:07 +01008274 ALC662_FIXUP_BASS_CHMAP,
Hui Wang493a52a2014-01-14 14:07:36 +08008275 ALC668_FIXUP_AUTO_MUTE,
Gabriele Mazzotta5e6db662014-05-20 18:58:26 +02008276 ALC668_FIXUP_DELL_DISABLE_AAMIX,
Gabriele Mazzotta033b0a72014-05-26 17:11:46 +02008277 ALC668_FIXUP_DELL_XPS13,
Bobi Mihalca9d4dc582016-03-23 13:26:11 +02008278 ALC662_FIXUP_ASUS_Nx50,
Mikhail Paulyshkafc7438b2017-04-21 08:52:42 +02008279 ALC668_FIXUP_ASUS_Nx51_HEADSET_MODE,
Yura Pakhuchiy3231e202016-05-07 23:53:36 +07008280 ALC668_FIXUP_ASUS_Nx51,
Takashi Iwai5b7c5e12018-10-09 14:20:17 +02008281 ALC668_FIXUP_MIC_COEF,
Takashi Iwai11ba6112018-10-07 09:44:17 +02008282 ALC668_FIXUP_ASUS_G751,
Kailang Yang78f4f7c2016-06-07 11:31:34 +08008283 ALC891_FIXUP_HEADSET_MODE,
8284 ALC891_FIXUP_DELL_MIC_NO_PRESENCE,
Shrirang Bagul9b51fe32016-08-01 13:16:17 +08008285 ALC662_FIXUP_ACER_VERITON,
Takashi Iwai1a3f0992016-10-20 12:14:51 +02008286 ALC892_FIXUP_ASROCK_MOBO,
Kailang Yangc6790c82016-11-25 16:15:17 +08008287 ALC662_FIXUP_USI_FUNC,
8288 ALC662_FIXUP_USI_HEADSET_MODE,
Kailang Yangca169cc2017-04-25 16:17:40 +08008289 ALC662_FIXUP_LENOVO_MULTI_CODECS,
David Henningsson6cb3b702010-09-09 08:51:44 +02008290};
8291
Takashi Iwai1727a772013-01-10 09:52:52 +01008292static const struct hda_fixup alc662_fixups[] = {
Daniel T Chen2df03512010-10-10 22:39:28 -04008293 [ALC662_FIXUP_ASPIRE] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01008294 .type = HDA_FIXUP_PINS,
8295 .v.pins = (const struct hda_pintbl[]) {
Daniel T Chen2df03512010-10-10 22:39:28 -04008296 { 0x15, 0x99130112 }, /* subwoofer */
8297 { }
8298 }
8299 },
Takashi Iwai3e887f32014-01-10 17:50:58 +01008300 [ALC662_FIXUP_LED_GPIO1] = {
8301 .type = HDA_FIXUP_FUNC,
8302 .v.func = alc662_fixup_led_gpio1,
8303 },
David Henningsson6cb3b702010-09-09 08:51:44 +02008304 [ALC662_FIXUP_IDEAPAD] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01008305 .type = HDA_FIXUP_PINS,
8306 .v.pins = (const struct hda_pintbl[]) {
David Henningsson6cb3b702010-09-09 08:51:44 +02008307 { 0x17, 0x99130112 }, /* subwoofer */
8308 { }
Takashi Iwai3e887f32014-01-10 17:50:58 +01008309 },
8310 .chained = true,
8311 .chain_id = ALC662_FIXUP_LED_GPIO1,
David Henningsson6cb3b702010-09-09 08:51:44 +02008312 },
Todd Broch6be79482010-12-07 16:51:05 -08008313 [ALC272_FIXUP_MARIO] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01008314 .type = HDA_FIXUP_FUNC,
Takashi Iwaib5bfbc62011-01-13 14:22:32 +01008315 .v.func = alc272_fixup_mario,
Anisse Astierd2ebd472011-01-20 12:36:21 +01008316 },
8317 [ALC662_FIXUP_CZC_P10T] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01008318 .type = HDA_FIXUP_VERBS,
Anisse Astierd2ebd472011-01-20 12:36:21 +01008319 .v.verbs = (const struct hda_verb[]) {
8320 {0x14, AC_VERB_SET_EAPD_BTLENABLE, 0},
8321 {}
8322 }
8323 },
David Henningsson94024cd2011-04-29 14:10:55 +02008324 [ALC662_FIXUP_SKU_IGNORE] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01008325 .type = HDA_FIXUP_FUNC,
Takashi Iwai23d30f22012-05-07 17:17:32 +02008326 .v.func = alc_fixup_sku_ignore,
Takashi Iwaic6b35872011-03-28 12:05:31 +02008327 },
Takashi Iwaie59ea3e2011-06-29 17:21:00 +02008328 [ALC662_FIXUP_HP_RP5800] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01008329 .type = HDA_FIXUP_PINS,
8330 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwaie59ea3e2011-06-29 17:21:00 +02008331 { 0x14, 0x0221201f }, /* HP out */
8332 { }
8333 },
8334 .chained = true,
8335 .chain_id = ALC662_FIXUP_SKU_IGNORE
8336 },
Takashi Iwai53c334a2011-08-23 18:27:14 +02008337 [ALC662_FIXUP_ASUS_MODE1] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01008338 .type = HDA_FIXUP_PINS,
8339 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai53c334a2011-08-23 18:27:14 +02008340 { 0x14, 0x99130110 }, /* speaker */
8341 { 0x18, 0x01a19c20 }, /* mic */
8342 { 0x19, 0x99a3092f }, /* int-mic */
8343 { 0x21, 0x0121401f }, /* HP out */
8344 { }
8345 },
8346 .chained = true,
8347 .chain_id = ALC662_FIXUP_SKU_IGNORE
8348 },
8349 [ALC662_FIXUP_ASUS_MODE2] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01008350 .type = HDA_FIXUP_PINS,
8351 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai2996bdb2011-08-18 16:02:24 +02008352 { 0x14, 0x99130110 }, /* speaker */
8353 { 0x18, 0x01a19820 }, /* mic */
8354 { 0x19, 0x99a3092f }, /* int-mic */
8355 { 0x1b, 0x0121401f }, /* HP out */
8356 { }
8357 },
Takashi Iwai53c334a2011-08-23 18:27:14 +02008358 .chained = true,
8359 .chain_id = ALC662_FIXUP_SKU_IGNORE
8360 },
8361 [ALC662_FIXUP_ASUS_MODE3] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01008362 .type = HDA_FIXUP_PINS,
8363 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai53c334a2011-08-23 18:27:14 +02008364 { 0x14, 0x99130110 }, /* speaker */
8365 { 0x15, 0x0121441f }, /* HP */
8366 { 0x18, 0x01a19840 }, /* mic */
8367 { 0x19, 0x99a3094f }, /* int-mic */
8368 { 0x21, 0x01211420 }, /* HP2 */
8369 { }
8370 },
8371 .chained = true,
8372 .chain_id = ALC662_FIXUP_SKU_IGNORE
8373 },
8374 [ALC662_FIXUP_ASUS_MODE4] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01008375 .type = HDA_FIXUP_PINS,
8376 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai53c334a2011-08-23 18:27:14 +02008377 { 0x14, 0x99130110 }, /* speaker */
8378 { 0x16, 0x99130111 }, /* speaker */
8379 { 0x18, 0x01a19840 }, /* mic */
8380 { 0x19, 0x99a3094f }, /* int-mic */
8381 { 0x21, 0x0121441f }, /* HP */
8382 { }
8383 },
8384 .chained = true,
8385 .chain_id = ALC662_FIXUP_SKU_IGNORE
8386 },
8387 [ALC662_FIXUP_ASUS_MODE5] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01008388 .type = HDA_FIXUP_PINS,
8389 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai53c334a2011-08-23 18:27:14 +02008390 { 0x14, 0x99130110 }, /* speaker */
8391 { 0x15, 0x0121441f }, /* HP */
8392 { 0x16, 0x99130111 }, /* speaker */
8393 { 0x18, 0x01a19840 }, /* mic */
8394 { 0x19, 0x99a3094f }, /* int-mic */
8395 { }
8396 },
8397 .chained = true,
8398 .chain_id = ALC662_FIXUP_SKU_IGNORE
8399 },
8400 [ALC662_FIXUP_ASUS_MODE6] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01008401 .type = HDA_FIXUP_PINS,
8402 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai53c334a2011-08-23 18:27:14 +02008403 { 0x14, 0x99130110 }, /* speaker */
8404 { 0x15, 0x01211420 }, /* HP2 */
8405 { 0x18, 0x01a19840 }, /* mic */
8406 { 0x19, 0x99a3094f }, /* int-mic */
8407 { 0x1b, 0x0121441f }, /* HP */
8408 { }
8409 },
8410 .chained = true,
8411 .chain_id = ALC662_FIXUP_SKU_IGNORE
8412 },
8413 [ALC662_FIXUP_ASUS_MODE7] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01008414 .type = HDA_FIXUP_PINS,
8415 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai53c334a2011-08-23 18:27:14 +02008416 { 0x14, 0x99130110 }, /* speaker */
8417 { 0x17, 0x99130111 }, /* speaker */
8418 { 0x18, 0x01a19840 }, /* mic */
8419 { 0x19, 0x99a3094f }, /* int-mic */
8420 { 0x1b, 0x01214020 }, /* HP */
8421 { 0x21, 0x0121401f }, /* HP */
8422 { }
8423 },
8424 .chained = true,
8425 .chain_id = ALC662_FIXUP_SKU_IGNORE
8426 },
8427 [ALC662_FIXUP_ASUS_MODE8] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01008428 .type = HDA_FIXUP_PINS,
8429 .v.pins = (const struct hda_pintbl[]) {
Takashi Iwai53c334a2011-08-23 18:27:14 +02008430 { 0x14, 0x99130110 }, /* speaker */
8431 { 0x12, 0x99a30970 }, /* int-mic */
8432 { 0x15, 0x01214020 }, /* HP */
8433 { 0x17, 0x99130111 }, /* speaker */
8434 { 0x18, 0x01a19840 }, /* mic */
8435 { 0x21, 0x0121401f }, /* HP */
8436 { }
8437 },
8438 .chained = true,
8439 .chain_id = ALC662_FIXUP_SKU_IGNORE
Takashi Iwai2996bdb2011-08-18 16:02:24 +02008440 },
Takashi Iwai1565cc32012-02-13 12:03:25 +01008441 [ALC662_FIXUP_NO_JACK_DETECT] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01008442 .type = HDA_FIXUP_FUNC,
Takashi Iwai1565cc32012-02-13 12:03:25 +01008443 .v.func = alc_fixup_no_jack_detect,
8444 },
David Henningssonedfe3bf2012-06-12 13:15:12 +02008445 [ALC662_FIXUP_ZOTAC_Z68] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01008446 .type = HDA_FIXUP_PINS,
8447 .v.pins = (const struct hda_pintbl[]) {
David Henningssonedfe3bf2012-06-12 13:15:12 +02008448 { 0x1b, 0x02214020 }, /* Front HP */
8449 { }
8450 }
8451 },
Takashi Iwai125821a2012-06-22 14:30:29 +02008452 [ALC662_FIXUP_INV_DMIC] = {
Takashi Iwai1727a772013-01-10 09:52:52 +01008453 .type = HDA_FIXUP_FUNC,
David Henningsson9d36a7d2014-10-07 10:18:42 +02008454 .v.func = alc_fixup_inv_dmic,
Takashi Iwai125821a2012-06-22 14:30:29 +02008455 },
Gabriele Mazzotta033b0a72014-05-26 17:11:46 +02008456 [ALC668_FIXUP_DELL_XPS13] = {
8457 .type = HDA_FIXUP_FUNC,
8458 .v.func = alc_fixup_dell_xps13,
8459 .chained = true,
8460 .chain_id = ALC668_FIXUP_DELL_DISABLE_AAMIX
8461 },
Gabriele Mazzotta5e6db662014-05-20 18:58:26 +02008462 [ALC668_FIXUP_DELL_DISABLE_AAMIX] = {
8463 .type = HDA_FIXUP_FUNC,
8464 .v.func = alc_fixup_disable_aamix,
8465 .chained = true,
8466 .chain_id = ALC668_FIXUP_DELL_MIC_NO_PRESENCE
8467 },
Hui Wang493a52a2014-01-14 14:07:36 +08008468 [ALC668_FIXUP_AUTO_MUTE] = {
8469 .type = HDA_FIXUP_FUNC,
8470 .v.func = alc_fixup_auto_mute_via_amp,
8471 .chained = true,
8472 .chain_id = ALC668_FIXUP_DELL_MIC_NO_PRESENCE
8473 },
David Henningsson1f8b46c2015-05-12 14:38:15 +02008474 [ALC662_FIXUP_DELL_MIC_NO_PRESENCE] = {
8475 .type = HDA_FIXUP_PINS,
8476 .v.pins = (const struct hda_pintbl[]) {
8477 { 0x19, 0x03a1113c }, /* use as headset mic, without its own jack detect */
8478 /* headphone mic by setting pin control of 0x1b (headphone out) to in + vref_50 */
8479 { }
8480 },
8481 .chained = true,
8482 .chain_id = ALC662_FIXUP_HEADSET_MODE
8483 },
8484 [ALC662_FIXUP_HEADSET_MODE] = {
8485 .type = HDA_FIXUP_FUNC,
8486 .v.func = alc_fixup_headset_mode_alc662,
8487 },
David Henningsson73bdd592013-04-15 15:44:14 +02008488 [ALC668_FIXUP_DELL_MIC_NO_PRESENCE] = {
8489 .type = HDA_FIXUP_PINS,
8490 .v.pins = (const struct hda_pintbl[]) {
8491 { 0x19, 0x03a1913d }, /* use as headphone mic, without its own jack detect */
8492 { 0x1b, 0x03a1113c }, /* use as headset mic, without its own jack detect */
8493 { }
8494 },
8495 .chained = true,
8496 .chain_id = ALC668_FIXUP_HEADSET_MODE
8497 },
8498 [ALC668_FIXUP_HEADSET_MODE] = {
8499 .type = HDA_FIXUP_FUNC,
8500 .v.func = alc_fixup_headset_mode_alc668,
8501 },
David Henningsson8e54b4a2014-02-07 09:31:07 +01008502 [ALC662_FIXUP_BASS_MODE4_CHMAP] = {
Takashi Iwai8e383952013-10-30 17:41:12 +01008503 .type = HDA_FIXUP_FUNC,
Takashi Iwaieb9ca3a2013-11-28 15:24:34 +01008504 .v.func = alc_fixup_bass_chmap,
Takashi Iwai8e383952013-10-30 17:41:12 +01008505 .chained = true,
8506 .chain_id = ALC662_FIXUP_ASUS_MODE4
8507 },
David Henningsson61a75f12014-02-07 09:31:08 +01008508 [ALC662_FIXUP_BASS_16] = {
8509 .type = HDA_FIXUP_PINS,
8510 .v.pins = (const struct hda_pintbl[]) {
8511 {0x16, 0x80106111}, /* bass speaker */
8512 {}
8513 },
8514 .chained = true,
8515 .chain_id = ALC662_FIXUP_BASS_CHMAP,
8516 },
Takashi Iwaia30c9aa2013-11-21 08:00:20 +01008517 [ALC662_FIXUP_BASS_1A] = {
8518 .type = HDA_FIXUP_PINS,
8519 .v.pins = (const struct hda_pintbl[]) {
8520 {0x1a, 0x80106111}, /* bass speaker */
8521 {}
8522 },
David Henningsson8e54b4a2014-02-07 09:31:07 +01008523 .chained = true,
8524 .chain_id = ALC662_FIXUP_BASS_CHMAP,
Takashi Iwaia30c9aa2013-11-21 08:00:20 +01008525 },
David Henningsson8e54b4a2014-02-07 09:31:07 +01008526 [ALC662_FIXUP_BASS_CHMAP] = {
Takashi Iwaia30c9aa2013-11-21 08:00:20 +01008527 .type = HDA_FIXUP_FUNC,
Takashi Iwaieb9ca3a2013-11-28 15:24:34 +01008528 .v.func = alc_fixup_bass_chmap,
Takashi Iwaia30c9aa2013-11-21 08:00:20 +01008529 },
Bobi Mihalca9d4dc582016-03-23 13:26:11 +02008530 [ALC662_FIXUP_ASUS_Nx50] = {
8531 .type = HDA_FIXUP_FUNC,
8532 .v.func = alc_fixup_auto_mute_via_amp,
8533 .chained = true,
8534 .chain_id = ALC662_FIXUP_BASS_1A
8535 },
Mikhail Paulyshkafc7438b2017-04-21 08:52:42 +02008536 [ALC668_FIXUP_ASUS_Nx51_HEADSET_MODE] = {
8537 .type = HDA_FIXUP_FUNC,
8538 .v.func = alc_fixup_headset_mode_alc668,
8539 .chain_id = ALC662_FIXUP_BASS_CHMAP
8540 },
Yura Pakhuchiy3231e202016-05-07 23:53:36 +07008541 [ALC668_FIXUP_ASUS_Nx51] = {
8542 .type = HDA_FIXUP_PINS,
8543 .v.pins = (const struct hda_pintbl[]) {
Mikhail Paulyshkafc7438b2017-04-21 08:52:42 +02008544 { 0x19, 0x03a1913d }, /* use as headphone mic, without its own jack detect */
8545 { 0x1a, 0x90170151 }, /* bass speaker */
8546 { 0x1b, 0x03a1113c }, /* use as headset mic, without its own jack detect */
Yura Pakhuchiy3231e202016-05-07 23:53:36 +07008547 {}
8548 },
8549 .chained = true,
Mikhail Paulyshkafc7438b2017-04-21 08:52:42 +02008550 .chain_id = ALC668_FIXUP_ASUS_Nx51_HEADSET_MODE,
Yura Pakhuchiy3231e202016-05-07 23:53:36 +07008551 },
Takashi Iwai5b7c5e12018-10-09 14:20:17 +02008552 [ALC668_FIXUP_MIC_COEF] = {
Takashi Iwai11ba6112018-10-07 09:44:17 +02008553 .type = HDA_FIXUP_VERBS,
8554 .v.verbs = (const struct hda_verb[]) {
8555 { 0x20, AC_VERB_SET_COEF_INDEX, 0xc3 },
8556 { 0x20, AC_VERB_SET_PROC_COEF, 0x4000 },
8557 {}
8558 },
8559 },
Takashi Iwai5b7c5e12018-10-09 14:20:17 +02008560 [ALC668_FIXUP_ASUS_G751] = {
8561 .type = HDA_FIXUP_PINS,
8562 .v.pins = (const struct hda_pintbl[]) {
8563 { 0x16, 0x0421101f }, /* HP */
8564 {}
8565 },
8566 .chained = true,
8567 .chain_id = ALC668_FIXUP_MIC_COEF
8568 },
Kailang Yang78f4f7c2016-06-07 11:31:34 +08008569 [ALC891_FIXUP_HEADSET_MODE] = {
8570 .type = HDA_FIXUP_FUNC,
8571 .v.func = alc_fixup_headset_mode,
8572 },
8573 [ALC891_FIXUP_DELL_MIC_NO_PRESENCE] = {
8574 .type = HDA_FIXUP_PINS,
8575 .v.pins = (const struct hda_pintbl[]) {
8576 { 0x19, 0x03a1913d }, /* use as headphone mic, without its own jack detect */
8577 { 0x1b, 0x03a1113c }, /* use as headset mic, without its own jack detect */
8578 { }
8579 },
8580 .chained = true,
8581 .chain_id = ALC891_FIXUP_HEADSET_MODE
8582 },
Shrirang Bagul9b51fe32016-08-01 13:16:17 +08008583 [ALC662_FIXUP_ACER_VERITON] = {
8584 .type = HDA_FIXUP_PINS,
8585 .v.pins = (const struct hda_pintbl[]) {
8586 { 0x15, 0x50170120 }, /* no internal speaker */
8587 { }
8588 }
8589 },
Takashi Iwai1a3f0992016-10-20 12:14:51 +02008590 [ALC892_FIXUP_ASROCK_MOBO] = {
8591 .type = HDA_FIXUP_PINS,
8592 .v.pins = (const struct hda_pintbl[]) {
8593 { 0x15, 0x40f000f0 }, /* disabled */
8594 { 0x16, 0x40f000f0 }, /* disabled */
Takashi Iwai1a3f0992016-10-20 12:14:51 +02008595 { }
8596 }
8597 },
Kailang Yangc6790c82016-11-25 16:15:17 +08008598 [ALC662_FIXUP_USI_FUNC] = {
8599 .type = HDA_FIXUP_FUNC,
8600 .v.func = alc662_fixup_usi_headset_mic,
8601 },
8602 [ALC662_FIXUP_USI_HEADSET_MODE] = {
8603 .type = HDA_FIXUP_PINS,
8604 .v.pins = (const struct hda_pintbl[]) {
8605 { 0x19, 0x02a1913c }, /* use as headset mic, without its own jack detect */
8606 { 0x18, 0x01a1903d },
8607 { }
8608 },
8609 .chained = true,
8610 .chain_id = ALC662_FIXUP_USI_FUNC
8611 },
Kailang Yangca169cc2017-04-25 16:17:40 +08008612 [ALC662_FIXUP_LENOVO_MULTI_CODECS] = {
8613 .type = HDA_FIXUP_FUNC,
8614 .v.func = alc233_alc662_fixup_lenovo_dual_codecs,
8615 },
David Henningsson6cb3b702010-09-09 08:51:44 +02008616};
8617
Takashi Iwaia9111322011-05-02 11:30:18 +02008618static const struct snd_pci_quirk alc662_fixup_tbl[] = {
Takashi Iwai53c334a2011-08-23 18:27:14 +02008619 SND_PCI_QUIRK(0x1019, 0x9087, "ECS", ALC662_FIXUP_ASUS_MODE2),
Takashi Iwaid3d38352013-08-19 20:05:50 +02008620 SND_PCI_QUIRK(0x1025, 0x022f, "Acer Aspire One", ALC662_FIXUP_INV_DMIC),
David Henningsson02f6ff92015-12-07 11:29:31 +01008621 SND_PCI_QUIRK(0x1025, 0x0241, "Packard Bell DOTS", ALC662_FIXUP_INV_DMIC),
David Henningssona6c47a82011-02-10 15:39:19 +01008622 SND_PCI_QUIRK(0x1025, 0x0308, "Acer Aspire 8942G", ALC662_FIXUP_ASPIRE),
David Henningsson94024cd2011-04-29 14:10:55 +02008623 SND_PCI_QUIRK(0x1025, 0x031c, "Gateway NV79", ALC662_FIXUP_SKU_IGNORE),
Takashi Iwai125821a2012-06-22 14:30:29 +02008624 SND_PCI_QUIRK(0x1025, 0x0349, "eMachines eM250", ALC662_FIXUP_INV_DMIC),
Takashi Iwai18019282013-08-16 08:17:05 +02008625 SND_PCI_QUIRK(0x1025, 0x034a, "Gateway LT27", ALC662_FIXUP_INV_DMIC),
Daniel T Chen2df03512010-10-10 22:39:28 -04008626 SND_PCI_QUIRK(0x1025, 0x038b, "Acer Aspire 8943G", ALC662_FIXUP_ASPIRE),
David Henningsson73bdd592013-04-15 15:44:14 +02008627 SND_PCI_QUIRK(0x1028, 0x05d8, "Dell", ALC668_FIXUP_DELL_MIC_NO_PRESENCE),
8628 SND_PCI_QUIRK(0x1028, 0x05db, "Dell", ALC668_FIXUP_DELL_MIC_NO_PRESENCE),
Takashi Iwaic5d019c2014-07-22 17:33:32 +02008629 SND_PCI_QUIRK(0x1028, 0x05fe, "Dell XPS 15", ALC668_FIXUP_DELL_XPS13),
Gabriele Mazzotta033b0a72014-05-26 17:11:46 +02008630 SND_PCI_QUIRK(0x1028, 0x060a, "Dell XPS 13", ALC668_FIXUP_DELL_XPS13),
Niranjan Sivakumar467e1432015-09-05 18:20:35 +02008631 SND_PCI_QUIRK(0x1028, 0x060d, "Dell M3800", ALC668_FIXUP_DELL_XPS13),
David Henningsson09d20142013-11-20 11:43:30 +01008632 SND_PCI_QUIRK(0x1028, 0x0625, "Dell", ALC668_FIXUP_DELL_MIC_NO_PRESENCE),
David Henningssonad8ff99e2013-11-07 09:28:59 +01008633 SND_PCI_QUIRK(0x1028, 0x0626, "Dell", ALC668_FIXUP_DELL_MIC_NO_PRESENCE),
Kailang Yang8dc9abb2014-04-16 15:53:12 +08008634 SND_PCI_QUIRK(0x1028, 0x0696, "Dell", ALC668_FIXUP_DELL_MIC_NO_PRESENCE),
8635 SND_PCI_QUIRK(0x1028, 0x0698, "Dell", ALC668_FIXUP_DELL_MIC_NO_PRESENCE),
Kailang Yang6a98e342014-10-27 16:20:30 +08008636 SND_PCI_QUIRK(0x1028, 0x069f, "Dell", ALC668_FIXUP_DELL_MIC_NO_PRESENCE),
Takashi Iwaie59ea3e2011-06-29 17:21:00 +02008637 SND_PCI_QUIRK(0x103c, 0x1632, "HP RP5800", ALC662_FIXUP_HP_RP5800),
Kaho Ng2da2dc92016-05-09 00:27:49 +08008638 SND_PCI_QUIRK(0x1043, 0x1080, "Asus UX501VW", ALC668_FIXUP_HEADSET_MODE),
Bobi Mihalca83a9efb2016-03-23 13:32:33 +02008639 SND_PCI_QUIRK(0x1043, 0x11cd, "Asus N550", ALC662_FIXUP_ASUS_Nx50),
Takashi Iwaidb8948e2016-01-18 09:17:30 +01008640 SND_PCI_QUIRK(0x1043, 0x13df, "Asus N550JX", ALC662_FIXUP_BASS_1A),
Bobi Mihalca9d4dc582016-03-23 13:26:11 +02008641 SND_PCI_QUIRK(0x1043, 0x129d, "Asus N750", ALC662_FIXUP_ASUS_Nx50),
Takashi Iwai11ba6112018-10-07 09:44:17 +02008642 SND_PCI_QUIRK(0x1043, 0x12ff, "ASUS G751", ALC668_FIXUP_ASUS_G751),
David Henningsson8e54b4a2014-02-07 09:31:07 +01008643 SND_PCI_QUIRK(0x1043, 0x1477, "ASUS N56VZ", ALC662_FIXUP_BASS_MODE4_CHMAP),
David Henningsson61a75f12014-02-07 09:31:08 +01008644 SND_PCI_QUIRK(0x1043, 0x15a7, "ASUS UX51VZH", ALC662_FIXUP_BASS_16),
Yura Pakhuchiy3231e202016-05-07 23:53:36 +07008645 SND_PCI_QUIRK(0x1043, 0x177d, "ASUS N551", ALC668_FIXUP_ASUS_Nx51),
8646 SND_PCI_QUIRK(0x1043, 0x17bd, "ASUS N751", ALC668_FIXUP_ASUS_Nx51),
Takashi Iwaic7efff92017-01-04 21:38:16 +01008647 SND_PCI_QUIRK(0x1043, 0x1963, "ASUS X71SL", ALC662_FIXUP_ASUS_MODE8),
David Henningsson61a75f12014-02-07 09:31:08 +01008648 SND_PCI_QUIRK(0x1043, 0x1b73, "ASUS N55SF", ALC662_FIXUP_BASS_16),
David Henningsson8e54b4a2014-02-07 09:31:07 +01008649 SND_PCI_QUIRK(0x1043, 0x1bf3, "ASUS N76VZ", ALC662_FIXUP_BASS_MODE4_CHMAP),
Takashi Iwai1565cc32012-02-13 12:03:25 +01008650 SND_PCI_QUIRK(0x1043, 0x8469, "ASUS mobo", ALC662_FIXUP_NO_JACK_DETECT),
Takashi Iwai53c334a2011-08-23 18:27:14 +02008651 SND_PCI_QUIRK(0x105b, 0x0cd6, "Foxconn", ALC662_FIXUP_ASUS_MODE2),
Daniel T Chena0e90ac2010-11-20 10:20:35 -05008652 SND_PCI_QUIRK(0x144d, 0xc051, "Samsung R720", ALC662_FIXUP_IDEAPAD),
Kailang Yangc6790c82016-11-25 16:15:17 +08008653 SND_PCI_QUIRK(0x14cd, 0x5003, "USI", ALC662_FIXUP_USI_HEADSET_MODE),
Kailang Yangca169cc2017-04-25 16:17:40 +08008654 SND_PCI_QUIRK(0x17aa, 0x1036, "Lenovo P520", ALC662_FIXUP_LENOVO_MULTI_CODECS),
Valentine Sinitsynd4118582010-10-01 22:24:08 +06008655 SND_PCI_QUIRK(0x17aa, 0x38af, "Lenovo Ideapad Y550P", ALC662_FIXUP_IDEAPAD),
David Henningsson6cb3b702010-09-09 08:51:44 +02008656 SND_PCI_QUIRK(0x17aa, 0x3a0d, "Lenovo Ideapad Y550", ALC662_FIXUP_IDEAPAD),
Takashi Iwai1a3f0992016-10-20 12:14:51 +02008657 SND_PCI_QUIRK(0x1849, 0x5892, "ASRock B150M", ALC892_FIXUP_ASROCK_MOBO),
David Henningssonedfe3bf2012-06-12 13:15:12 +02008658 SND_PCI_QUIRK(0x19da, 0xa130, "Zotac Z68", ALC662_FIXUP_ZOTAC_Z68),
Shrirang Bagul9b51fe32016-08-01 13:16:17 +08008659 SND_PCI_QUIRK(0x1b0a, 0x01b8, "ACER Veriton", ALC662_FIXUP_ACER_VERITON),
Anisse Astierd2ebd472011-01-20 12:36:21 +01008660 SND_PCI_QUIRK(0x1b35, 0x2206, "CZC P10T", ALC662_FIXUP_CZC_P10T),
Takashi Iwai53c334a2011-08-23 18:27:14 +02008661
8662#if 0
8663 /* Below is a quirk table taken from the old code.
8664 * Basically the device should work as is without the fixup table.
8665 * If BIOS doesn't give a proper info, enable the corresponding
8666 * fixup entry.
Jesper Juhl7d7eb9e2012-04-12 22:11:25 +02008667 */
Takashi Iwai53c334a2011-08-23 18:27:14 +02008668 SND_PCI_QUIRK(0x1043, 0x1000, "ASUS N50Vm", ALC662_FIXUP_ASUS_MODE1),
8669 SND_PCI_QUIRK(0x1043, 0x1092, "ASUS NB", ALC662_FIXUP_ASUS_MODE3),
8670 SND_PCI_QUIRK(0x1043, 0x1173, "ASUS K73Jn", ALC662_FIXUP_ASUS_MODE1),
8671 SND_PCI_QUIRK(0x1043, 0x11c3, "ASUS M70V", ALC662_FIXUP_ASUS_MODE3),
8672 SND_PCI_QUIRK(0x1043, 0x11d3, "ASUS NB", ALC662_FIXUP_ASUS_MODE1),
8673 SND_PCI_QUIRK(0x1043, 0x11f3, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
8674 SND_PCI_QUIRK(0x1043, 0x1203, "ASUS NB", ALC662_FIXUP_ASUS_MODE1),
8675 SND_PCI_QUIRK(0x1043, 0x1303, "ASUS G60J", ALC662_FIXUP_ASUS_MODE1),
8676 SND_PCI_QUIRK(0x1043, 0x1333, "ASUS G60Jx", ALC662_FIXUP_ASUS_MODE1),
8677 SND_PCI_QUIRK(0x1043, 0x1339, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
8678 SND_PCI_QUIRK(0x1043, 0x13e3, "ASUS N71JA", ALC662_FIXUP_ASUS_MODE7),
8679 SND_PCI_QUIRK(0x1043, 0x1463, "ASUS N71", ALC662_FIXUP_ASUS_MODE7),
8680 SND_PCI_QUIRK(0x1043, 0x14d3, "ASUS G72", ALC662_FIXUP_ASUS_MODE8),
8681 SND_PCI_QUIRK(0x1043, 0x1563, "ASUS N90", ALC662_FIXUP_ASUS_MODE3),
8682 SND_PCI_QUIRK(0x1043, 0x15d3, "ASUS N50SF F50SF", ALC662_FIXUP_ASUS_MODE1),
8683 SND_PCI_QUIRK(0x1043, 0x16c3, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
8684 SND_PCI_QUIRK(0x1043, 0x16f3, "ASUS K40C K50C", ALC662_FIXUP_ASUS_MODE2),
8685 SND_PCI_QUIRK(0x1043, 0x1733, "ASUS N81De", ALC662_FIXUP_ASUS_MODE1),
8686 SND_PCI_QUIRK(0x1043, 0x1753, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
8687 SND_PCI_QUIRK(0x1043, 0x1763, "ASUS NB", ALC662_FIXUP_ASUS_MODE6),
8688 SND_PCI_QUIRK(0x1043, 0x1765, "ASUS NB", ALC662_FIXUP_ASUS_MODE6),
8689 SND_PCI_QUIRK(0x1043, 0x1783, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
8690 SND_PCI_QUIRK(0x1043, 0x1793, "ASUS F50GX", ALC662_FIXUP_ASUS_MODE1),
8691 SND_PCI_QUIRK(0x1043, 0x17b3, "ASUS F70SL", ALC662_FIXUP_ASUS_MODE3),
8692 SND_PCI_QUIRK(0x1043, 0x17f3, "ASUS X58LE", ALC662_FIXUP_ASUS_MODE2),
8693 SND_PCI_QUIRK(0x1043, 0x1813, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
8694 SND_PCI_QUIRK(0x1043, 0x1823, "ASUS NB", ALC662_FIXUP_ASUS_MODE5),
8695 SND_PCI_QUIRK(0x1043, 0x1833, "ASUS NB", ALC662_FIXUP_ASUS_MODE6),
8696 SND_PCI_QUIRK(0x1043, 0x1843, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
8697 SND_PCI_QUIRK(0x1043, 0x1853, "ASUS F50Z", ALC662_FIXUP_ASUS_MODE1),
8698 SND_PCI_QUIRK(0x1043, 0x1864, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
8699 SND_PCI_QUIRK(0x1043, 0x1876, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
8700 SND_PCI_QUIRK(0x1043, 0x1893, "ASUS M50Vm", ALC662_FIXUP_ASUS_MODE3),
8701 SND_PCI_QUIRK(0x1043, 0x1894, "ASUS X55", ALC662_FIXUP_ASUS_MODE3),
8702 SND_PCI_QUIRK(0x1043, 0x18b3, "ASUS N80Vc", ALC662_FIXUP_ASUS_MODE1),
8703 SND_PCI_QUIRK(0x1043, 0x18c3, "ASUS VX5", ALC662_FIXUP_ASUS_MODE1),
8704 SND_PCI_QUIRK(0x1043, 0x18d3, "ASUS N81Te", ALC662_FIXUP_ASUS_MODE1),
8705 SND_PCI_QUIRK(0x1043, 0x18f3, "ASUS N505Tp", ALC662_FIXUP_ASUS_MODE1),
8706 SND_PCI_QUIRK(0x1043, 0x1903, "ASUS F5GL", ALC662_FIXUP_ASUS_MODE1),
8707 SND_PCI_QUIRK(0x1043, 0x1913, "ASUS NB", ALC662_FIXUP_ASUS_MODE2),
8708 SND_PCI_QUIRK(0x1043, 0x1933, "ASUS F80Q", ALC662_FIXUP_ASUS_MODE2),
8709 SND_PCI_QUIRK(0x1043, 0x1943, "ASUS Vx3V", ALC662_FIXUP_ASUS_MODE1),
8710 SND_PCI_QUIRK(0x1043, 0x1953, "ASUS NB", ALC662_FIXUP_ASUS_MODE1),
8711 SND_PCI_QUIRK(0x1043, 0x1963, "ASUS X71C", ALC662_FIXUP_ASUS_MODE3),
8712 SND_PCI_QUIRK(0x1043, 0x1983, "ASUS N5051A", ALC662_FIXUP_ASUS_MODE1),
8713 SND_PCI_QUIRK(0x1043, 0x1993, "ASUS N20", ALC662_FIXUP_ASUS_MODE1),
8714 SND_PCI_QUIRK(0x1043, 0x19b3, "ASUS F7Z", ALC662_FIXUP_ASUS_MODE1),
8715 SND_PCI_QUIRK(0x1043, 0x19c3, "ASUS F5Z/F6x", ALC662_FIXUP_ASUS_MODE2),
8716 SND_PCI_QUIRK(0x1043, 0x19e3, "ASUS NB", ALC662_FIXUP_ASUS_MODE1),
8717 SND_PCI_QUIRK(0x1043, 0x19f3, "ASUS NB", ALC662_FIXUP_ASUS_MODE4),
8718#endif
David Henningsson6cb3b702010-09-09 08:51:44 +02008719 {}
8720};
8721
Takashi Iwai1727a772013-01-10 09:52:52 +01008722static const struct hda_model_fixup alc662_fixup_models[] = {
Takashi Iwaiaa3841b2018-06-26 07:42:40 +02008723 {.id = ALC662_FIXUP_ASPIRE, .name = "aspire"},
8724 {.id = ALC662_FIXUP_IDEAPAD, .name = "ideapad"},
Todd Broch6be79482010-12-07 16:51:05 -08008725 {.id = ALC272_FIXUP_MARIO, .name = "mario"},
Takashi Iwaiaa3841b2018-06-26 07:42:40 +02008726 {.id = ALC662_FIXUP_HP_RP5800, .name = "hp-rp5800"},
Takashi Iwai53c334a2011-08-23 18:27:14 +02008727 {.id = ALC662_FIXUP_ASUS_MODE1, .name = "asus-mode1"},
8728 {.id = ALC662_FIXUP_ASUS_MODE2, .name = "asus-mode2"},
8729 {.id = ALC662_FIXUP_ASUS_MODE3, .name = "asus-mode3"},
8730 {.id = ALC662_FIXUP_ASUS_MODE4, .name = "asus-mode4"},
8731 {.id = ALC662_FIXUP_ASUS_MODE5, .name = "asus-mode5"},
8732 {.id = ALC662_FIXUP_ASUS_MODE6, .name = "asus-mode6"},
8733 {.id = ALC662_FIXUP_ASUS_MODE7, .name = "asus-mode7"},
8734 {.id = ALC662_FIXUP_ASUS_MODE8, .name = "asus-mode8"},
Takashi Iwaiaa3841b2018-06-26 07:42:40 +02008735 {.id = ALC662_FIXUP_ZOTAC_Z68, .name = "zotac-z68"},
Takashi Iwai6e72aa52012-06-25 10:52:25 +02008736 {.id = ALC662_FIXUP_INV_DMIC, .name = "inv-dmic"},
Takashi Iwaiaa3841b2018-06-26 07:42:40 +02008737 {.id = ALC662_FIXUP_DELL_MIC_NO_PRESENCE, .name = "alc662-headset-multi"},
David Henningssone32aa852013-06-17 11:04:02 +02008738 {.id = ALC668_FIXUP_DELL_MIC_NO_PRESENCE, .name = "dell-headset-multi"},
Takashi Iwaiaa3841b2018-06-26 07:42:40 +02008739 {.id = ALC662_FIXUP_HEADSET_MODE, .name = "alc662-headset"},
8740 {.id = ALC668_FIXUP_HEADSET_MODE, .name = "alc668-headset"},
8741 {.id = ALC662_FIXUP_BASS_16, .name = "bass16"},
8742 {.id = ALC662_FIXUP_BASS_1A, .name = "bass1a"},
8743 {.id = ALC668_FIXUP_AUTO_MUTE, .name = "automute"},
8744 {.id = ALC668_FIXUP_DELL_XPS13, .name = "dell-xps13"},
8745 {.id = ALC662_FIXUP_ASUS_Nx50, .name = "asus-nx50"},
8746 {.id = ALC668_FIXUP_ASUS_Nx51, .name = "asus-nx51"},
Takashi Iwai40c51672018-10-07 09:47:39 +02008747 {.id = ALC668_FIXUP_ASUS_G751, .name = "asus-g751"},
Takashi Iwaiaa3841b2018-06-26 07:42:40 +02008748 {.id = ALC891_FIXUP_HEADSET_MODE, .name = "alc891-headset"},
8749 {.id = ALC891_FIXUP_DELL_MIC_NO_PRESENCE, .name = "alc891-headset-multi"},
8750 {.id = ALC662_FIXUP_ACER_VERITON, .name = "acer-veriton"},
8751 {.id = ALC892_FIXUP_ASROCK_MOBO, .name = "asrock-mobo"},
8752 {.id = ALC662_FIXUP_USI_HEADSET_MODE, .name = "usi-headset"},
Takashi Iwaiba90d6a2017-05-22 16:38:47 +02008753 {.id = ALC662_FIXUP_LENOVO_MULTI_CODECS, .name = "dual-codecs"},
Todd Broch6be79482010-12-07 16:51:05 -08008754 {}
8755};
David Henningsson6cb3b702010-09-09 08:51:44 +02008756
Hui Wang532895c2014-05-29 15:59:19 +08008757static const struct snd_hda_pin_quirk alc662_pin_fixup_tbl[] = {
Kailang Yang78f4f7c2016-06-07 11:31:34 +08008758 SND_HDA_PIN_QUIRK(0x10ec0867, 0x1028, "Dell", ALC891_FIXUP_DELL_MIC_NO_PRESENCE,
8759 {0x17, 0x02211010},
8760 {0x18, 0x01a19030},
8761 {0x1a, 0x01813040},
8762 {0x21, 0x01014020}),
David Henningsson1f8b46c2015-05-12 14:38:15 +02008763 SND_HDA_PIN_QUIRK(0x10ec0662, 0x1028, "Dell", ALC662_FIXUP_DELL_MIC_NO_PRESENCE,
David Henningsson1f8b46c2015-05-12 14:38:15 +02008764 {0x14, 0x01014010},
David Henningsson1f8b46c2015-05-12 14:38:15 +02008765 {0x18, 0x01a19020},
David Henningsson1f8b46c2015-05-12 14:38:15 +02008766 {0x1a, 0x0181302f},
Hui Wang11580292015-08-03 11:03:49 +08008767 {0x1b, 0x0221401f}),
David Henningsson76c21322014-06-24 14:46:54 +02008768 SND_HDA_PIN_QUIRK(0x10ec0668, 0x1028, "Dell", ALC668_FIXUP_AUTO_MUTE,
8769 {0x12, 0x99a30130},
8770 {0x14, 0x90170110},
8771 {0x15, 0x0321101f},
Hui Wang11580292015-08-03 11:03:49 +08008772 {0x16, 0x03011020}),
David Henningsson76c21322014-06-24 14:46:54 +02008773 SND_HDA_PIN_QUIRK(0x10ec0668, 0x1028, "Dell", ALC668_FIXUP_AUTO_MUTE,
8774 {0x12, 0x99a30140},
8775 {0x14, 0x90170110},
8776 {0x15, 0x0321101f},
Hui Wang11580292015-08-03 11:03:49 +08008777 {0x16, 0x03011020}),
David Henningsson76c21322014-06-24 14:46:54 +02008778 SND_HDA_PIN_QUIRK(0x10ec0668, 0x1028, "Dell", ALC668_FIXUP_AUTO_MUTE,
8779 {0x12, 0x99a30150},
8780 {0x14, 0x90170110},
8781 {0x15, 0x0321101f},
Hui Wang11580292015-08-03 11:03:49 +08008782 {0x16, 0x03011020}),
David Henningsson76c21322014-06-24 14:46:54 +02008783 SND_HDA_PIN_QUIRK(0x10ec0668, 0x1028, "Dell", ALC668_FIXUP_AUTO_MUTE,
David Henningsson76c21322014-06-24 14:46:54 +02008784 {0x14, 0x90170110},
8785 {0x15, 0x0321101f},
Hui Wang11580292015-08-03 11:03:49 +08008786 {0x16, 0x03011020}),
David Henningsson76c21322014-06-24 14:46:54 +02008787 SND_HDA_PIN_QUIRK(0x10ec0668, 0x1028, "Dell XPS 15", ALC668_FIXUP_AUTO_MUTE,
8788 {0x12, 0x90a60130},
8789 {0x14, 0x90170110},
Hui Wang11580292015-08-03 11:03:49 +08008790 {0x15, 0x0321101f}),
Hui Wang532895c2014-05-29 15:59:19 +08008791 {}
8792};
8793
Takashi Iwai1d045db2011-07-07 18:23:21 +02008794/*
8795 */
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008796static int patch_alc662(struct hda_codec *codec)
8797{
8798 struct alc_spec *spec;
Takashi Iwai3de95172012-05-07 18:03:15 +02008799 int err;
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008800
Takashi Iwai3de95172012-05-07 18:03:15 +02008801 err = alc_alloc_spec(codec, 0x0b);
8802 if (err < 0)
8803 return err;
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008804
Takashi Iwai3de95172012-05-07 18:03:15 +02008805 spec = codec->spec;
Takashi Iwai1f0f4b82011-06-27 10:52:59 +02008806
Takashi Iwai225068a2015-05-29 10:42:14 +02008807 spec->shutup = alc_eapd_shutup;
8808
Takashi Iwai53c334a2011-08-23 18:27:14 +02008809 /* handle multiple HPs as is */
8810 spec->parse_flags = HDA_PINCFG_NO_HP_FIXUP;
8811
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +02008812 alc_fix_pll_init(codec, 0x20, 0x04, 15);
8813
Takashi Iwai7639a062015-03-03 10:07:24 +01008814 switch (codec->core.vendor_id) {
Kailang Yangf3f91852014-10-24 15:43:46 +08008815 case 0x10ec0668:
8816 spec->init_hook = alc668_restore_default_value;
8817 break;
Kailang Yangf3f91852014-10-24 15:43:46 +08008818 }
Kailang Yang8663ff72012-06-29 09:35:52 +02008819
Takashi Iwaic9af7532019-05-10 11:01:43 +02008820 alc_pre_init(codec);
8821
Takashi Iwai1727a772013-01-10 09:52:52 +01008822 snd_hda_pick_fixup(codec, alc662_fixup_models,
Takashi Iwai8e5a0502012-06-21 15:49:33 +02008823 alc662_fixup_tbl, alc662_fixups);
Hui Wang532895c2014-05-29 15:59:19 +08008824 snd_hda_pick_pin_fixup(codec, alc662_pin_fixup_tbl, alc662_fixups);
Takashi Iwai1727a772013-01-10 09:52:52 +01008825 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
Takashi Iwai8e5a0502012-06-21 15:49:33 +02008826
8827 alc_auto_parse_customize_define(codec);
8828
Takashi Iwai7504b6c2013-03-18 11:25:51 +01008829 if (has_cdefine_beep(codec))
8830 spec->gen.beep_nid = 0x01;
8831
Takashi Iwai1bb7e432011-10-17 16:50:59 +02008832 if ((alc_get_coef0(codec) & (1 << 14)) &&
Takashi Iwai5100cd02014-02-15 10:03:19 +01008833 codec->bus->pci && codec->bus->pci->subsystem_vendor == 0x1025 &&
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02008834 spec->cdefine.platform_type == 1) {
Wei Yongjun6134b1a2013-04-18 11:12:59 +08008835 err = alc_codec_rename(codec, "ALC272X");
8836 if (err < 0)
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02008837 goto error;
Takashi Iwai20ca0c32011-10-17 16:00:35 +02008838 }
Kailang Yang274693f2009-12-03 10:07:50 +01008839
Takashi Iwaib9c51062011-08-24 18:08:07 +02008840 /* automatic parse from the BIOS config */
8841 err = alc662_parse_auto_config(codec);
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02008842 if (err < 0)
8843 goto error;
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008844
Takashi Iwai7504b6c2013-03-18 11:25:51 +01008845 if (!spec->gen.no_analog && spec->gen.beep_nid) {
Takashi Iwai7639a062015-03-03 10:07:24 +01008846 switch (codec->core.vendor_id) {
Kailang Yangda00c242010-03-19 11:23:45 +01008847 case 0x10ec0662:
Takashi Iwaifea80fa2018-06-20 12:52:46 +02008848 err = set_beep_amp(spec, 0x0b, 0x05, HDA_INPUT);
Kailang Yangda00c242010-03-19 11:23:45 +01008849 break;
8850 case 0x10ec0272:
8851 case 0x10ec0663:
8852 case 0x10ec0665:
Kailang Yang9ad54542013-11-26 15:41:40 +08008853 case 0x10ec0668:
Takashi Iwaifea80fa2018-06-20 12:52:46 +02008854 err = set_beep_amp(spec, 0x0b, 0x04, HDA_INPUT);
Kailang Yangda00c242010-03-19 11:23:45 +01008855 break;
8856 case 0x10ec0273:
Takashi Iwaifea80fa2018-06-20 12:52:46 +02008857 err = set_beep_amp(spec, 0x0b, 0x03, HDA_INPUT);
Kailang Yangda00c242010-03-19 11:23:45 +01008858 break;
8859 }
Takashi Iwaifea80fa2018-06-20 12:52:46 +02008860 if (err < 0)
8861 goto error;
Kailang Yangcec27c82010-02-04 14:18:18 +01008862 }
Takashi Iwai2134ea42008-01-10 16:53:55 +01008863
Takashi Iwai1727a772013-01-10 09:52:52 +01008864 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
Takashi Iwai589876e2012-02-20 15:47:55 +01008865
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008866 return 0;
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008867
Takashi Iwaie16fb6d2011-10-17 16:39:09 +02008868 error:
8869 alc_free(codec);
8870 return err;
Kailang Yangb478b992011-05-18 11:51:15 +02008871}
8872
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008873/*
Kailang Yangd1eb57f2010-06-23 16:25:26 +02008874 * ALC680 support
8875 */
Kailang Yangd1eb57f2010-06-23 16:25:26 +02008876
Kailang Yangd1eb57f2010-06-23 16:25:26 +02008877static int alc680_parse_auto_config(struct hda_codec *codec)
8878{
Takashi Iwai3e6179b2011-07-08 16:55:13 +02008879 return alc_parse_auto_config(codec, NULL, NULL);
Kailang Yangd1eb57f2010-06-23 16:25:26 +02008880}
8881
Kailang Yangd1eb57f2010-06-23 16:25:26 +02008882/*
Kailang Yangd1eb57f2010-06-23 16:25:26 +02008883 */
Kailang Yangd1eb57f2010-06-23 16:25:26 +02008884static int patch_alc680(struct hda_codec *codec)
8885{
Kailang Yangd1eb57f2010-06-23 16:25:26 +02008886 int err;
8887
Takashi Iwai1f0f4b82011-06-27 10:52:59 +02008888 /* ALC680 has no aa-loopback mixer */
Takashi Iwai3de95172012-05-07 18:03:15 +02008889 err = alc_alloc_spec(codec, 0);
8890 if (err < 0)
8891 return err;
Takashi Iwai1f0f4b82011-06-27 10:52:59 +02008892
Takashi Iwai1ebec5f2011-08-15 13:21:48 +02008893 /* automatic parse from the BIOS config */
8894 err = alc680_parse_auto_config(codec);
8895 if (err < 0) {
8896 alc_free(codec);
8897 return err;
Kailang Yangd1eb57f2010-06-23 16:25:26 +02008898 }
8899
Kailang Yangd1eb57f2010-06-23 16:25:26 +02008900 return 0;
8901}
8902
8903/*
Linus Torvalds1da177e2005-04-16 15:20:36 -07008904 * patch entries
8905 */
Takashi Iwaib9a94a92015-10-01 16:20:04 +02008906static const struct hda_device_id snd_hda_id_realtek[] = {
Kailang Yang0a6f0602017-06-30 16:00:48 +08008907 HDA_CODEC_ENTRY(0x10ec0215, "ALC215", patch_alc269),
Takashi Iwaib9a94a92015-10-01 16:20:04 +02008908 HDA_CODEC_ENTRY(0x10ec0221, "ALC221", patch_alc269),
Kailang Yang42314302016-02-03 15:03:50 +08008909 HDA_CODEC_ENTRY(0x10ec0225, "ALC225", patch_alc269),
Takashi Iwaib9a94a92015-10-01 16:20:04 +02008910 HDA_CODEC_ENTRY(0x10ec0231, "ALC231", patch_alc269),
8911 HDA_CODEC_ENTRY(0x10ec0233, "ALC233", patch_alc269),
Kailang Yangdcd4f0d2016-05-04 15:50:18 +08008912 HDA_CODEC_ENTRY(0x10ec0234, "ALC234", patch_alc269),
Takashi Iwaib9a94a92015-10-01 16:20:04 +02008913 HDA_CODEC_ENTRY(0x10ec0235, "ALC233", patch_alc269),
Kailang Yang736f20a2017-10-20 15:06:34 +08008914 HDA_CODEC_ENTRY(0x10ec0236, "ALC236", patch_alc269),
Takashi Iwaib9a94a92015-10-01 16:20:04 +02008915 HDA_CODEC_ENTRY(0x10ec0255, "ALC255", patch_alc269),
8916 HDA_CODEC_ENTRY(0x10ec0256, "ALC256", patch_alc269),
Kailang Yangf429e7e2017-12-05 15:38:24 +08008917 HDA_CODEC_ENTRY(0x10ec0257, "ALC257", patch_alc269),
Takashi Iwaib9a94a92015-10-01 16:20:04 +02008918 HDA_CODEC_ENTRY(0x10ec0260, "ALC260", patch_alc260),
8919 HDA_CODEC_ENTRY(0x10ec0262, "ALC262", patch_alc262),
8920 HDA_CODEC_ENTRY(0x10ec0267, "ALC267", patch_alc268),
8921 HDA_CODEC_ENTRY(0x10ec0268, "ALC268", patch_alc268),
8922 HDA_CODEC_ENTRY(0x10ec0269, "ALC269", patch_alc269),
8923 HDA_CODEC_ENTRY(0x10ec0270, "ALC270", patch_alc269),
8924 HDA_CODEC_ENTRY(0x10ec0272, "ALC272", patch_alc662),
Kailang Yangdcd4f0d2016-05-04 15:50:18 +08008925 HDA_CODEC_ENTRY(0x10ec0274, "ALC274", patch_alc269),
Takashi Iwaib9a94a92015-10-01 16:20:04 +02008926 HDA_CODEC_ENTRY(0x10ec0275, "ALC275", patch_alc269),
8927 HDA_CODEC_ENTRY(0x10ec0276, "ALC276", patch_alc269),
8928 HDA_CODEC_ENTRY(0x10ec0280, "ALC280", patch_alc269),
8929 HDA_CODEC_ENTRY(0x10ec0282, "ALC282", patch_alc269),
8930 HDA_CODEC_ENTRY(0x10ec0283, "ALC283", patch_alc269),
8931 HDA_CODEC_ENTRY(0x10ec0284, "ALC284", patch_alc269),
Kailang Yang0a6f0602017-06-30 16:00:48 +08008932 HDA_CODEC_ENTRY(0x10ec0285, "ALC285", patch_alc269),
Takashi Iwaib9a94a92015-10-01 16:20:04 +02008933 HDA_CODEC_ENTRY(0x10ec0286, "ALC286", patch_alc269),
8934 HDA_CODEC_ENTRY(0x10ec0288, "ALC288", patch_alc269),
Kailang Yang0a6f0602017-06-30 16:00:48 +08008935 HDA_CODEC_ENTRY(0x10ec0289, "ALC289", patch_alc269),
Takashi Iwaib9a94a92015-10-01 16:20:04 +02008936 HDA_CODEC_ENTRY(0x10ec0290, "ALC290", patch_alc269),
8937 HDA_CODEC_ENTRY(0x10ec0292, "ALC292", patch_alc269),
8938 HDA_CODEC_ENTRY(0x10ec0293, "ALC293", patch_alc269),
Kailang Yangdcd4f0d2016-05-04 15:50:18 +08008939 HDA_CODEC_ENTRY(0x10ec0294, "ALC294", patch_alc269),
Kailang Yang7d727862016-05-24 16:46:07 +08008940 HDA_CODEC_ENTRY(0x10ec0295, "ALC295", patch_alc269),
Takashi Iwaib9a94a92015-10-01 16:20:04 +02008941 HDA_CODEC_ENTRY(0x10ec0298, "ALC298", patch_alc269),
Kailang Yang28f1f9b2017-01-04 14:49:07 +08008942 HDA_CODEC_ENTRY(0x10ec0299, "ALC299", patch_alc269),
Kailang Yang1078bef2018-11-08 16:36:15 +08008943 HDA_CODEC_ENTRY(0x10ec0300, "ALC300", patch_alc269),
Takashi Iwaib9a94a92015-10-01 16:20:04 +02008944 HDA_CODEC_REV_ENTRY(0x10ec0861, 0x100340, "ALC660", patch_alc861),
8945 HDA_CODEC_ENTRY(0x10ec0660, "ALC660-VD", patch_alc861vd),
8946 HDA_CODEC_ENTRY(0x10ec0861, "ALC861", patch_alc861),
8947 HDA_CODEC_ENTRY(0x10ec0862, "ALC861-VD", patch_alc861vd),
8948 HDA_CODEC_REV_ENTRY(0x10ec0662, 0x100002, "ALC662 rev2", patch_alc882),
8949 HDA_CODEC_REV_ENTRY(0x10ec0662, 0x100101, "ALC662 rev1", patch_alc662),
8950 HDA_CODEC_REV_ENTRY(0x10ec0662, 0x100300, "ALC662 rev3", patch_alc662),
8951 HDA_CODEC_ENTRY(0x10ec0663, "ALC663", patch_alc662),
8952 HDA_CODEC_ENTRY(0x10ec0665, "ALC665", patch_alc662),
8953 HDA_CODEC_ENTRY(0x10ec0667, "ALC667", patch_alc662),
8954 HDA_CODEC_ENTRY(0x10ec0668, "ALC668", patch_alc662),
8955 HDA_CODEC_ENTRY(0x10ec0670, "ALC670", patch_alc662),
8956 HDA_CODEC_ENTRY(0x10ec0671, "ALC671", patch_alc662),
8957 HDA_CODEC_ENTRY(0x10ec0680, "ALC680", patch_alc680),
Kailang Yang6fbae352016-05-30 16:44:20 +08008958 HDA_CODEC_ENTRY(0x10ec0700, "ALC700", patch_alc269),
8959 HDA_CODEC_ENTRY(0x10ec0701, "ALC701", patch_alc269),
8960 HDA_CODEC_ENTRY(0x10ec0703, "ALC703", patch_alc269),
Kailang Yang78f4f7c2016-06-07 11:31:34 +08008961 HDA_CODEC_ENTRY(0x10ec0867, "ALC891", patch_alc662),
Takashi Iwaib9a94a92015-10-01 16:20:04 +02008962 HDA_CODEC_ENTRY(0x10ec0880, "ALC880", patch_alc880),
8963 HDA_CODEC_ENTRY(0x10ec0882, "ALC882", patch_alc882),
8964 HDA_CODEC_ENTRY(0x10ec0883, "ALC883", patch_alc882),
8965 HDA_CODEC_REV_ENTRY(0x10ec0885, 0x100101, "ALC889A", patch_alc882),
8966 HDA_CODEC_REV_ENTRY(0x10ec0885, 0x100103, "ALC889A", patch_alc882),
8967 HDA_CODEC_ENTRY(0x10ec0885, "ALC885", patch_alc882),
8968 HDA_CODEC_ENTRY(0x10ec0887, "ALC887", patch_alc882),
8969 HDA_CODEC_REV_ENTRY(0x10ec0888, 0x100101, "ALC1200", patch_alc882),
8970 HDA_CODEC_ENTRY(0x10ec0888, "ALC888", patch_alc882),
8971 HDA_CODEC_ENTRY(0x10ec0889, "ALC889", patch_alc882),
8972 HDA_CODEC_ENTRY(0x10ec0892, "ALC892", patch_alc662),
8973 HDA_CODEC_ENTRY(0x10ec0899, "ALC898", patch_alc882),
8974 HDA_CODEC_ENTRY(0x10ec0900, "ALC1150", patch_alc882),
Kailang Yang65553b12017-07-11 15:15:47 +08008975 HDA_CODEC_ENTRY(0x10ec1168, "ALC1220", patch_alc882),
Kailang Yanga535ad52017-01-16 16:59:26 +08008976 HDA_CODEC_ENTRY(0x10ec1220, "ALC1220", patch_alc882),
Linus Torvalds1da177e2005-04-16 15:20:36 -07008977 {} /* terminator */
8978};
Takashi Iwaib9a94a92015-10-01 16:20:04 +02008979MODULE_DEVICE_TABLE(hdaudio, snd_hda_id_realtek);
Takashi Iwai1289e9e2008-11-27 15:47:11 +01008980
8981MODULE_LICENSE("GPL");
8982MODULE_DESCRIPTION("Realtek HD-audio codec");
8983
Takashi Iwaid8a766a2015-02-17 15:25:37 +01008984static struct hda_codec_driver realtek_driver = {
Takashi Iwaib9a94a92015-10-01 16:20:04 +02008985 .id = snd_hda_id_realtek,
Takashi Iwai1289e9e2008-11-27 15:47:11 +01008986};
8987
Takashi Iwaid8a766a2015-02-17 15:25:37 +01008988module_hda_codec_driver(realtek_driver);