blob: 92a709be3c4e7cea98933d9b95b531da075e3e3a [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * Universal Interface for Intel High Definition Audio Codec
3 *
4 * HD audio interface patch for ALC 260/880/882 codecs
5 *
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 Woithe7cf51e42006-02-09 12:01:26 +01009 * Jonathan Woithe <jwoithe@physics.adelaide.edu.au>
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>
30#include <sound/core.h>
31#include "hda_codec.h"
32#include "hda_local.h"
Harvey Harrison3c9a3202008-02-29 11:59:26 +010033#include "hda_patch.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070034
Kailang Yangccc656c2006-10-17 12:32:26 +020035#define ALC880_FRONT_EVENT 0x01
36#define ALC880_DCVOL_EVENT 0x02
37#define ALC880_HP_EVENT 0x04
38#define ALC880_MIC_EVENT 0x08
Linus Torvalds1da177e2005-04-16 15:20:36 -070039
40/* ALC880 board config type */
41enum {
Linus Torvalds1da177e2005-04-16 15:20:36 -070042 ALC880_3ST,
43 ALC880_3ST_DIG,
44 ALC880_5ST,
45 ALC880_5ST_DIG,
46 ALC880_W810,
Takashi Iwaidfc0ff62005-05-12 14:31:49 +020047 ALC880_Z71V,
Takashi Iwaib6482d42005-06-27 15:32:43 +020048 ALC880_6ST,
Takashi Iwai16ded522005-06-10 19:58:24 +020049 ALC880_6ST_DIG,
50 ALC880_F1734,
51 ALC880_ASUS,
52 ALC880_ASUS_DIG,
53 ALC880_ASUS_W1V,
Kailang Yangdf694da2005-12-05 19:42:22 +010054 ALC880_ASUS_DIG2,
Tobin Davis2cf9f0f2007-02-07 16:04:25 +010055 ALC880_FUJITSU,
Takashi Iwai16ded522005-06-10 19:58:24 +020056 ALC880_UNIWILL_DIG,
Kailang Yangccc656c2006-10-17 12:32:26 +020057 ALC880_UNIWILL,
58 ALC880_UNIWILL_P53,
Kailang Yangdf694da2005-12-05 19:42:22 +010059 ALC880_CLEVO,
60 ALC880_TCL_S700,
Takashi Iwaiae6b8132006-03-03 16:47:17 +010061 ALC880_LG,
Takashi Iwaid6815182006-03-23 16:06:23 +010062 ALC880_LG_LW,
Takashi Iwaidf99cd32008-04-25 15:25:04 +020063 ALC880_MEDION_RIM,
Takashi Iwaie9edcee2005-06-13 14:16:38 +020064#ifdef CONFIG_SND_DEBUG
65 ALC880_TEST,
66#endif
Kailang Yangdf694da2005-12-05 19:42:22 +010067 ALC880_AUTO,
Takashi Iwai16ded522005-06-10 19:58:24 +020068 ALC880_MODEL_LAST /* last tag */
69};
70
71/* ALC260 models */
72enum {
73 ALC260_BASIC,
74 ALC260_HP,
Kailang Yangdf694da2005-12-05 19:42:22 +010075 ALC260_HP_3013,
76 ALC260_FUJITSU_S702X,
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +010077 ALC260_ACER,
Kailang Yangbc9f98a2007-04-12 13:06:07 +020078 ALC260_WILL,
79 ALC260_REPLACER_672V,
Jonathan Woithe7cf51e42006-02-09 12:01:26 +010080#ifdef CONFIG_SND_DEBUG
81 ALC260_TEST,
82#endif
Kailang Yangdf694da2005-12-05 19:42:22 +010083 ALC260_AUTO,
Takashi Iwai16ded522005-06-10 19:58:24 +020084 ALC260_MODEL_LAST /* last tag */
Linus Torvalds1da177e2005-04-16 15:20:36 -070085};
86
Kailang Yangdf694da2005-12-05 19:42:22 +010087/* ALC262 models */
88enum {
89 ALC262_BASIC,
Kailang Yangccc656c2006-10-17 12:32:26 +020090 ALC262_HIPPO,
91 ALC262_HIPPO_1,
Takashi Iwai834be882006-03-01 14:16:17 +010092 ALC262_FUJITSU,
Takashi Iwai9c7f8522006-06-28 15:08:22 +020093 ALC262_HP_BPC,
Kailang Yangcd7509a2007-01-26 18:33:17 +010094 ALC262_HP_BPC_D7000_WL,
95 ALC262_HP_BPC_D7000_WF,
Kailang Yang66d2a9d2007-11-14 12:06:21 +010096 ALC262_HP_TC_T5735,
Kailang Yang8c427222008-01-10 13:03:59 +010097 ALC262_HP_RP5700,
Takashi Iwai304dcaa2006-07-25 14:51:16 +020098 ALC262_BENQ_ED8,
Kailang Yang272a5272007-05-14 11:00:38 +020099 ALC262_SONY_ASSAMD,
Kailang Yang83c34212007-07-05 11:43:05 +0200100 ALC262_BENQ_T31,
Tobin Davisf651b502007-10-26 12:40:47 +0200101 ALC262_ULTRA,
Jiang zhe0e31daf2008-03-20 12:12:39 +0100102 ALC262_LENOVO_3000,
Kailang Yangdf694da2005-12-05 19:42:22 +0100103 ALC262_AUTO,
104 ALC262_MODEL_LAST /* last tag */
105};
106
Kailang Yanga361d842007-06-05 12:30:55 +0200107/* ALC268 models */
108enum {
Herton Ronaldo Krzesinskieb5a6622008-04-14 13:46:28 +0200109 ALC267_QUANTA_IL1,
Kailang Yanga361d842007-06-05 12:30:55 +0200110 ALC268_3ST,
Kailang Yangd1a991a2007-08-15 16:21:59 +0200111 ALC268_TOSHIBA,
Takashi Iwaid2738092007-08-16 14:59:45 +0200112 ALC268_ACER,
Takashi Iwai3866f0b2008-01-15 12:37:42 +0100113 ALC268_DELL,
Mirco Tischlerf12462c2008-02-04 12:33:59 +0100114 ALC268_ZEPTO,
Jonathan Woithe86c53bd2008-01-08 12:33:19 +0100115#ifdef CONFIG_SND_DEBUG
116 ALC268_TEST,
117#endif
Kailang Yanga361d842007-06-05 12:30:55 +0200118 ALC268_AUTO,
119 ALC268_MODEL_LAST /* last tag */
120};
121
Kailang Yangf6a92242007-12-13 16:52:54 +0100122/* ALC269 models */
123enum {
124 ALC269_BASIC,
125 ALC269_AUTO,
126 ALC269_MODEL_LAST /* last tag */
127};
128
Kailang Yangdf694da2005-12-05 19:42:22 +0100129/* ALC861 models */
130enum {
131 ALC861_3ST,
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200132 ALC660_3ST,
Kailang Yangdf694da2005-12-05 19:42:22 +0100133 ALC861_3ST_DIG,
134 ALC861_6ST_DIG,
Takashi Iwai22309c32006-08-09 16:57:28 +0200135 ALC861_UNIWILL_M31,
Tobin Davisa53d1ae2006-10-17 12:00:28 +0200136 ALC861_TOSHIBA,
Mariusz Domanski7cdbff92006-10-23 13:42:56 +0200137 ALC861_ASUS,
Takashi Iwai56bb0ca2006-11-22 11:52:52 +0100138 ALC861_ASUS_LAPTOP,
Kailang Yangdf694da2005-12-05 19:42:22 +0100139 ALC861_AUTO,
140 ALC861_MODEL_LAST,
141};
142
Jakub Schmidtkef32610e2007-02-02 18:17:27 +0100143/* ALC861-VD models */
144enum {
145 ALC660VD_3ST,
Mike Crash6963f842007-06-25 12:12:51 +0200146 ALC660VD_3ST_DIG,
Jakub Schmidtkef32610e2007-02-02 18:17:27 +0100147 ALC861VD_3ST,
148 ALC861VD_3ST_DIG,
149 ALC861VD_6ST_DIG,
Kailang Yangbdd148a2007-05-08 15:19:08 +0200150 ALC861VD_LENOVO,
Kailang Yang272a5272007-05-14 11:00:38 +0200151 ALC861VD_DALLAS,
Kailang Yangd1a991a2007-08-15 16:21:59 +0200152 ALC861VD_HP,
Jakub Schmidtkef32610e2007-02-02 18:17:27 +0100153 ALC861VD_AUTO,
154 ALC861VD_MODEL_LAST,
155};
156
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200157/* ALC662 models */
158enum {
159 ALC662_3ST_2ch_DIG,
160 ALC662_3ST_6ch_DIG,
161 ALC662_3ST_6ch,
162 ALC662_5ST_DIG,
163 ALC662_LENOVO_101E,
Kailang Yang291702f2007-10-16 14:28:03 +0200164 ALC662_ASUS_EEEPC_P701,
Kailang Yang8c427222008-01-10 13:03:59 +0100165 ALC662_ASUS_EEEPC_EP20,
Kailang Yang6dda9f42008-05-27 12:05:31 +0200166 ALC663_ASUS_M51VA,
167 ALC663_ASUS_G71V,
168 ALC663_ASUS_H13,
169 ALC663_ASUS_G50V,
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200170 ALC662_AUTO,
171 ALC662_MODEL_LAST,
172};
173
Kailang Yangdf694da2005-12-05 19:42:22 +0100174/* ALC882 models */
175enum {
176 ALC882_3ST_DIG,
177 ALC882_6ST_DIG,
Takashi Iwai4b146cb2006-07-28 14:42:36 +0200178 ALC882_ARIMA,
Kailang Yangbdd148a2007-05-08 15:19:08 +0200179 ALC882_W2JC,
Kailang Yang272a5272007-05-14 11:00:38 +0200180 ALC882_TARGA,
181 ALC882_ASUS_A7J,
Takashi Iwai914759b2007-09-06 14:52:04 +0200182 ALC882_ASUS_A7M,
Tobin Davis9102cd12006-12-15 10:02:12 +0100183 ALC885_MACPRO,
Takashi Iwai87350ad2007-08-16 18:19:38 +0200184 ALC885_MBP3,
Nicola Fagnanic54728d2007-07-19 23:28:52 +0200185 ALC885_IMAC24,
Kailang Yang272a5272007-05-14 11:00:38 +0200186 ALC882_AUTO,
Kailang Yangdf694da2005-12-05 19:42:22 +0100187 ALC882_MODEL_LAST,
188};
189
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200190/* ALC883 models */
191enum {
192 ALC883_3ST_2ch_DIG,
193 ALC883_3ST_6ch_DIG,
194 ALC883_3ST_6ch,
195 ALC883_6ST_DIG,
Kailang Yangccc656c2006-10-17 12:32:26 +0200196 ALC883_TARGA_DIG,
197 ALC883_TARGA_2ch_DIG,
Vladimir Avdoninbab282b2006-08-22 13:31:58 +0200198 ALC883_ACER,
Tobin Davis2880a862007-08-07 11:50:26 +0200199 ALC883_ACER_ASPIRE,
Tobin Davisc07584c2006-10-13 12:32:16 +0200200 ALC883_MEDION,
Kailang Yang272a5272007-05-14 11:00:38 +0200201 ALC883_MEDION_MD2,
Andrew L. Neporadab373bde2006-11-07 11:37:08 +0100202 ALC883_LAPTOP_EAPD,
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200203 ALC883_LENOVO_101E_2ch,
Kailang Yang272a5272007-05-14 11:00:38 +0200204 ALC883_LENOVO_NB0763,
Kailang Yang189609a2007-08-20 11:31:23 +0200205 ALC888_LENOVO_MS7195_DIG,
206 ALC883_HAIER_W66,
Claudio Matsuoka4723c022007-07-13 14:36:19 +0200207 ALC888_3ST_HP,
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +0100208 ALC888_6ST_DELL,
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +0100209 ALC883_MITAC,
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +0100210 ALC883_CLEVO_M720,
Jiang zhefb97dc62008-03-06 11:07:11 +0100211 ALC883_FUJITSU_PI2515,
Jiang zhe17bba1b2008-06-04 12:11:07 +0200212 ALC883_3ST_6ch_INTEL,
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200213 ALC883_AUTO,
214 ALC883_MODEL_LAST,
215};
216
Kailang Yangdf694da2005-12-05 19:42:22 +0100217/* for GPIO Poll */
218#define GPIO_MASK 0x03
219
Linus Torvalds1da177e2005-04-16 15:20:36 -0700220struct alc_spec {
221 /* codec parameterization */
Kailang Yangdf694da2005-12-05 19:42:22 +0100222 struct snd_kcontrol_new *mixers[5]; /* mixer arrays */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700223 unsigned int num_mixers;
224
Kailang Yangdf694da2005-12-05 19:42:22 +0100225 const struct hda_verb *init_verbs[5]; /* initialization verbs
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200226 * don't forget NULL
227 * termination!
Takashi Iwaie9edcee2005-06-13 14:16:38 +0200228 */
229 unsigned int num_init_verbs;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700230
Takashi Iwai16ded522005-06-10 19:58:24 +0200231 char *stream_name_analog; /* analog PCM stream */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700232 struct hda_pcm_stream *stream_analog_playback;
233 struct hda_pcm_stream *stream_analog_capture;
Takashi Iwai63300792008-01-24 15:31:36 +0100234 struct hda_pcm_stream *stream_analog_alt_playback;
235 struct hda_pcm_stream *stream_analog_alt_capture;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700236
Takashi Iwaif12ab1e2007-04-12 15:51:47 +0200237 char *stream_name_digital; /* digital PCM stream */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700238 struct hda_pcm_stream *stream_digital_playback;
239 struct hda_pcm_stream *stream_digital_capture;
240
241 /* playback */
Takashi Iwai16ded522005-06-10 19:58:24 +0200242 struct hda_multi_out multiout; /* playback set-up
243 * max_channels, dacs must be set
244 * dig_out_nid and hp_nid are optional
245 */
Takashi Iwai63300792008-01-24 15:31:36 +0100246 hda_nid_t alt_dac_nid;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700247
248 /* capture */
249 unsigned int num_adc_nids;
250 hda_nid_t *adc_nids;
Takashi Iwaie1406342008-02-11 18:32:32 +0100251 hda_nid_t *capsrc_nids;
Takashi Iwai16ded522005-06-10 19:58:24 +0200252 hda_nid_t dig_in_nid; /* digital-in NID; optional */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700253
254 /* capture source */
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200255 unsigned int num_mux_defs;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700256 const struct hda_input_mux *input_mux;
257 unsigned int cur_mux[3];
258
259 /* channel model */
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +0100260 const struct hda_channel_mode *channel_mode;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700261 int num_channel_mode;
Takashi Iwai4e195a72006-07-28 14:47:34 +0200262 int need_dac_fix;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700263
264 /* PCM information */
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100265 struct hda_pcm pcm_rec[3]; /* used in alc_build_pcms() */
Takashi Iwai41e41f12005-06-08 14:48:49 +0200266
Takashi Iwaie9edcee2005-06-13 14:16:38 +0200267 /* dynamic controls, init_verbs and input_mux */
268 struct auto_pin_cfg autocfg;
269 unsigned int num_kctl_alloc, num_kctl_used;
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +0100270 struct snd_kcontrol_new *kctl_alloc;
Takashi Iwaie9edcee2005-06-13 14:16:38 +0200271 struct hda_input_mux private_imux;
Takashi Iwai41923e42007-10-22 17:20:10 +0200272 hda_nid_t private_dac_nids[AUTO_CFG_MAX_OUTS];
Takashi Iwai834be882006-03-01 14:16:17 +0100273
Takashi Iwaiae6b8132006-03-03 16:47:17 +0100274 /* hooks */
275 void (*init_hook)(struct hda_codec *codec);
276 void (*unsol_event)(struct hda_codec *codec, unsigned int res);
277
Takashi Iwai834be882006-03-01 14:16:17 +0100278 /* for pin sensing */
279 unsigned int sense_updated: 1;
280 unsigned int jack_present: 1;
Takashi Iwaibec15c32008-01-28 18:16:30 +0100281 unsigned int master_sw: 1;
Takashi Iwaicb53c622007-08-10 17:21:45 +0200282
Takashi Iwai2134ea42008-01-10 16:53:55 +0100283 /* for virtual master */
284 hda_nid_t vmaster_nid;
Takashi Iwaicb53c622007-08-10 17:21:45 +0200285#ifdef CONFIG_SND_HDA_POWER_SAVE
286 struct hda_loopback_check loopback;
287#endif
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +0200288
289 /* for PLL fix */
290 hda_nid_t pll_nid;
291 unsigned int pll_coef_idx, pll_coef_bit;
Kailang Yangdf694da2005-12-05 19:42:22 +0100292};
293
294/*
295 * configuration template - to be copied to the spec instance
296 */
297struct alc_config_preset {
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200298 struct snd_kcontrol_new *mixers[5]; /* should be identical size
299 * with spec
300 */
Kailang Yangdf694da2005-12-05 19:42:22 +0100301 const struct hda_verb *init_verbs[5];
302 unsigned int num_dacs;
303 hda_nid_t *dac_nids;
304 hda_nid_t dig_out_nid; /* optional */
305 hda_nid_t hp_nid; /* optional */
306 unsigned int num_adc_nids;
307 hda_nid_t *adc_nids;
Takashi Iwaie1406342008-02-11 18:32:32 +0100308 hda_nid_t *capsrc_nids;
Kailang Yangdf694da2005-12-05 19:42:22 +0100309 hda_nid_t dig_in_nid;
310 unsigned int num_channel_mode;
311 const struct hda_channel_mode *channel_mode;
Takashi Iwai4e195a72006-07-28 14:47:34 +0200312 int need_dac_fix;
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200313 unsigned int num_mux_defs;
Kailang Yangdf694da2005-12-05 19:42:22 +0100314 const struct hda_input_mux *input_mux;
Takashi Iwaiae6b8132006-03-03 16:47:17 +0100315 void (*unsol_event)(struct hda_codec *, unsigned int);
316 void (*init_hook)(struct hda_codec *);
Takashi Iwaicb53c622007-08-10 17:21:45 +0200317#ifdef CONFIG_SND_HDA_POWER_SAVE
318 struct hda_amp_list *loopbacks;
319#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700320};
321
Linus Torvalds1da177e2005-04-16 15:20:36 -0700322
323/*
324 * input MUX handling
325 */
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200326static int alc_mux_enum_info(struct snd_kcontrol *kcontrol,
327 struct snd_ctl_elem_info *uinfo)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700328{
329 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
330 struct alc_spec *spec = codec->spec;
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200331 unsigned int mux_idx = snd_ctl_get_ioffidx(kcontrol, &uinfo->id);
332 if (mux_idx >= spec->num_mux_defs)
333 mux_idx = 0;
334 return snd_hda_input_mux_info(&spec->input_mux[mux_idx], uinfo);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700335}
336
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200337static int alc_mux_enum_get(struct snd_kcontrol *kcontrol,
338 struct snd_ctl_elem_value *ucontrol)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700339{
340 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
341 struct alc_spec *spec = codec->spec;
342 unsigned int adc_idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
343
344 ucontrol->value.enumerated.item[0] = spec->cur_mux[adc_idx];
345 return 0;
346}
347
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200348static int alc_mux_enum_put(struct snd_kcontrol *kcontrol,
349 struct snd_ctl_elem_value *ucontrol)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700350{
351 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
352 struct alc_spec *spec = codec->spec;
353 unsigned int adc_idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200354 unsigned int mux_idx = adc_idx >= spec->num_mux_defs ? 0 : adc_idx;
Takashi Iwaie1406342008-02-11 18:32:32 +0100355 hda_nid_t nid = spec->capsrc_nids ?
356 spec->capsrc_nids[adc_idx] : spec->adc_nids[adc_idx];
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200357 return snd_hda_input_mux_put(codec, &spec->input_mux[mux_idx], ucontrol,
Takashi Iwaie1406342008-02-11 18:32:32 +0100358 nid, &spec->cur_mux[adc_idx]);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700359}
360
Takashi Iwaie9edcee2005-06-13 14:16:38 +0200361
Linus Torvalds1da177e2005-04-16 15:20:36 -0700362/*
363 * channel mode setting
364 */
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200365static int alc_ch_mode_info(struct snd_kcontrol *kcontrol,
366 struct snd_ctl_elem_info *uinfo)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700367{
368 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
369 struct alc_spec *spec = codec->spec;
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +0100370 return snd_hda_ch_mode_info(codec, uinfo, spec->channel_mode,
371 spec->num_channel_mode);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700372}
373
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200374static int alc_ch_mode_get(struct snd_kcontrol *kcontrol,
375 struct snd_ctl_elem_value *ucontrol)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700376{
377 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
378 struct alc_spec *spec = codec->spec;
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +0100379 return snd_hda_ch_mode_get(codec, ucontrol, spec->channel_mode,
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200380 spec->num_channel_mode,
381 spec->multiout.max_channels);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700382}
383
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200384static int alc_ch_mode_put(struct snd_kcontrol *kcontrol,
385 struct snd_ctl_elem_value *ucontrol)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700386{
387 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
388 struct alc_spec *spec = codec->spec;
Takashi Iwai4e195a72006-07-28 14:47:34 +0200389 int err = snd_hda_ch_mode_put(codec, ucontrol, spec->channel_mode,
390 spec->num_channel_mode,
391 &spec->multiout.max_channels);
Takashi Iwaibd2033f2006-10-10 19:49:31 +0200392 if (err >= 0 && spec->need_dac_fix)
Takashi Iwai4e195a72006-07-28 14:47:34 +0200393 spec->multiout.num_dacs = spec->multiout.max_channels / 2;
394 return err;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700395}
396
Linus Torvalds1da177e2005-04-16 15:20:36 -0700397/*
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100398 * Control the mode of pin widget settings via the mixer. "pc" is used
399 * instead of "%" to avoid consequences of accidently treating the % as
400 * being part of a format specifier. Maximum allowed length of a value is
401 * 63 characters plus NULL terminator.
Jonathan Woithe7cf51e42006-02-09 12:01:26 +0100402 *
403 * Note: some retasking pin complexes seem to ignore requests for input
404 * states other than HiZ (eg: PIN_VREFxx) and revert to HiZ if any of these
405 * are requested. Therefore order this list so that this behaviour will not
406 * cause problems when mixer clients move through the enum sequentially.
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200407 * NIDs 0x0f and 0x10 have been observed to have this behaviour as of
408 * March 2006.
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200409 */
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100410static char *alc_pin_mode_names[] = {
Jonathan Woithe7cf51e42006-02-09 12:01:26 +0100411 "Mic 50pc bias", "Mic 80pc bias",
412 "Line in", "Line out", "Headphone out",
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100413};
414static unsigned char alc_pin_mode_values[] = {
Jonathan Woithe7cf51e42006-02-09 12:01:26 +0100415 PIN_VREF50, PIN_VREF80, PIN_IN, PIN_OUT, PIN_HP,
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100416};
417/* The control can present all 5 options, or it can limit the options based
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200418 * in the pin being assumed to be exclusively an input or an output pin. In
419 * addition, "input" pins may or may not process the mic bias option
420 * depending on actual widget capability (NIDs 0x0f and 0x10 don't seem to
421 * accept requests for bias as of chip versions up to March 2006) and/or
422 * wiring in the computer.
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100423 */
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200424#define ALC_PIN_DIR_IN 0x00
425#define ALC_PIN_DIR_OUT 0x01
426#define ALC_PIN_DIR_INOUT 0x02
427#define ALC_PIN_DIR_IN_NOMICBIAS 0x03
428#define ALC_PIN_DIR_INOUT_NOMICBIAS 0x04
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100429
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200430/* Info about the pin modes supported by the different pin direction modes.
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100431 * For each direction the minimum and maximum values are given.
432 */
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200433static signed char alc_pin_mode_dir_info[5][2] = {
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100434 { 0, 2 }, /* ALC_PIN_DIR_IN */
435 { 3, 4 }, /* ALC_PIN_DIR_OUT */
436 { 0, 4 }, /* ALC_PIN_DIR_INOUT */
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200437 { 2, 2 }, /* ALC_PIN_DIR_IN_NOMICBIAS */
438 { 2, 4 }, /* ALC_PIN_DIR_INOUT_NOMICBIAS */
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100439};
440#define alc_pin_mode_min(_dir) (alc_pin_mode_dir_info[_dir][0])
441#define alc_pin_mode_max(_dir) (alc_pin_mode_dir_info[_dir][1])
442#define alc_pin_mode_n_items(_dir) \
443 (alc_pin_mode_max(_dir)-alc_pin_mode_min(_dir)+1)
444
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200445static int alc_pin_mode_info(struct snd_kcontrol *kcontrol,
446 struct snd_ctl_elem_info *uinfo)
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200447{
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100448 unsigned int item_num = uinfo->value.enumerated.item;
449 unsigned char dir = (kcontrol->private_value >> 16) & 0xff;
450
451 uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200452 uinfo->count = 1;
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100453 uinfo->value.enumerated.items = alc_pin_mode_n_items(dir);
454
455 if (item_num<alc_pin_mode_min(dir) || item_num>alc_pin_mode_max(dir))
456 item_num = alc_pin_mode_min(dir);
457 strcpy(uinfo->value.enumerated.name, alc_pin_mode_names[item_num]);
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200458 return 0;
459}
460
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200461static int alc_pin_mode_get(struct snd_kcontrol *kcontrol,
462 struct snd_ctl_elem_value *ucontrol)
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200463{
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100464 unsigned int i;
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200465 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
466 hda_nid_t nid = kcontrol->private_value & 0xffff;
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100467 unsigned char dir = (kcontrol->private_value >> 16) & 0xff;
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200468 long *valp = ucontrol->value.integer.value;
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200469 unsigned int pinctl = snd_hda_codec_read(codec, nid, 0,
470 AC_VERB_GET_PIN_WIDGET_CONTROL,
471 0x00);
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200472
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100473 /* Find enumerated value for current pinctl setting */
474 i = alc_pin_mode_min(dir);
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200475 while (alc_pin_mode_values[i] != pinctl && i <= alc_pin_mode_max(dir))
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100476 i++;
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200477 *valp = i <= alc_pin_mode_max(dir) ? i: alc_pin_mode_min(dir);
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100478 return 0;
479}
480
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200481static int alc_pin_mode_put(struct snd_kcontrol *kcontrol,
482 struct snd_ctl_elem_value *ucontrol)
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100483{
484 signed int change;
485 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
486 hda_nid_t nid = kcontrol->private_value & 0xffff;
487 unsigned char dir = (kcontrol->private_value >> 16) & 0xff;
488 long val = *ucontrol->value.integer.value;
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200489 unsigned int pinctl = snd_hda_codec_read(codec, nid, 0,
490 AC_VERB_GET_PIN_WIDGET_CONTROL,
491 0x00);
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100492
Takashi Iwaif12ab1e2007-04-12 15:51:47 +0200493 if (val < alc_pin_mode_min(dir) || val > alc_pin_mode_max(dir))
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100494 val = alc_pin_mode_min(dir);
495
496 change = pinctl != alc_pin_mode_values[val];
Jonathan Woithecdcd9262006-02-28 11:36:42 +0100497 if (change) {
498 /* Set pin mode to that requested */
Takashi Iwai82beb8f2007-08-10 17:09:26 +0200499 snd_hda_codec_write_cache(codec, nid, 0,
500 AC_VERB_SET_PIN_WIDGET_CONTROL,
501 alc_pin_mode_values[val]);
Jonathan Woithecdcd9262006-02-28 11:36:42 +0100502
503 /* Also enable the retasking pin's input/output as required
504 * for the requested pin mode. Enum values of 2 or less are
505 * input modes.
506 *
507 * Dynamically switching the input/output buffers probably
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200508 * reduces noise slightly (particularly on input) so we'll
509 * do it. However, having both input and output buffers
510 * enabled simultaneously doesn't seem to be problematic if
511 * this turns out to be necessary in the future.
Jonathan Woithecdcd9262006-02-28 11:36:42 +0100512 */
513 if (val <= 2) {
Takashi Iwai47fd8302007-08-10 17:11:07 +0200514 snd_hda_codec_amp_stereo(codec, nid, HDA_OUTPUT, 0,
515 HDA_AMP_MUTE, HDA_AMP_MUTE);
516 snd_hda_codec_amp_stereo(codec, nid, HDA_INPUT, 0,
517 HDA_AMP_MUTE, 0);
Jonathan Woithecdcd9262006-02-28 11:36:42 +0100518 } else {
Takashi Iwai47fd8302007-08-10 17:11:07 +0200519 snd_hda_codec_amp_stereo(codec, nid, HDA_INPUT, 0,
520 HDA_AMP_MUTE, HDA_AMP_MUTE);
521 snd_hda_codec_amp_stereo(codec, nid, HDA_OUTPUT, 0,
522 HDA_AMP_MUTE, 0);
Jonathan Woithecdcd9262006-02-28 11:36:42 +0100523 }
524 }
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200525 return change;
526}
527
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100528#define ALC_PIN_MODE(xname, nid, dir) \
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200529 { .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, .index = 0, \
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100530 .info = alc_pin_mode_info, \
531 .get = alc_pin_mode_get, \
532 .put = alc_pin_mode_put, \
533 .private_value = nid | (dir<<16) }
Kailang Yangdf694da2005-12-05 19:42:22 +0100534
Jonathan Woithe5c8f8582006-02-28 11:43:27 +0100535/* A switch control for ALC260 GPIO pins. Multiple GPIOs can be ganged
536 * together using a mask with more than one bit set. This control is
537 * currently used only by the ALC260 test model. At this stage they are not
538 * needed for any "production" models.
539 */
540#ifdef CONFIG_SND_DEBUG
Takashi Iwaia5ce8892007-07-23 15:42:26 +0200541#define alc_gpio_data_info snd_ctl_boolean_mono_info
Takashi Iwaif12ab1e2007-04-12 15:51:47 +0200542
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200543static int alc_gpio_data_get(struct snd_kcontrol *kcontrol,
544 struct snd_ctl_elem_value *ucontrol)
Jonathan Woithe5c8f8582006-02-28 11:43:27 +0100545{
546 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
547 hda_nid_t nid = kcontrol->private_value & 0xffff;
548 unsigned char mask = (kcontrol->private_value >> 16) & 0xff;
549 long *valp = ucontrol->value.integer.value;
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200550 unsigned int val = snd_hda_codec_read(codec, nid, 0,
551 AC_VERB_GET_GPIO_DATA, 0x00);
Jonathan Woithe5c8f8582006-02-28 11:43:27 +0100552
553 *valp = (val & mask) != 0;
554 return 0;
555}
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200556static int alc_gpio_data_put(struct snd_kcontrol *kcontrol,
557 struct snd_ctl_elem_value *ucontrol)
Jonathan Woithe5c8f8582006-02-28 11:43:27 +0100558{
559 signed int change;
560 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
561 hda_nid_t nid = kcontrol->private_value & 0xffff;
562 unsigned char mask = (kcontrol->private_value >> 16) & 0xff;
563 long val = *ucontrol->value.integer.value;
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200564 unsigned int gpio_data = snd_hda_codec_read(codec, nid, 0,
565 AC_VERB_GET_GPIO_DATA,
566 0x00);
Jonathan Woithe5c8f8582006-02-28 11:43:27 +0100567
568 /* Set/unset the masked GPIO bit(s) as needed */
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200569 change = (val == 0 ? 0 : mask) != (gpio_data & mask);
570 if (val == 0)
Jonathan Woithe5c8f8582006-02-28 11:43:27 +0100571 gpio_data &= ~mask;
572 else
573 gpio_data |= mask;
Takashi Iwai82beb8f2007-08-10 17:09:26 +0200574 snd_hda_codec_write_cache(codec, nid, 0,
575 AC_VERB_SET_GPIO_DATA, gpio_data);
Jonathan Woithe5c8f8582006-02-28 11:43:27 +0100576
577 return change;
578}
579#define ALC_GPIO_DATA_SWITCH(xname, nid, mask) \
580 { .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, .index = 0, \
581 .info = alc_gpio_data_info, \
582 .get = alc_gpio_data_get, \
583 .put = alc_gpio_data_put, \
584 .private_value = nid | (mask<<16) }
585#endif /* CONFIG_SND_DEBUG */
586
Jonathan Woithe92621f12006-02-28 11:47:47 +0100587/* A switch control to allow the enabling of the digital IO pins on the
588 * ALC260. This is incredibly simplistic; the intention of this control is
589 * to provide something in the test model allowing digital outputs to be
590 * identified if present. If models are found which can utilise these
591 * outputs a more complete mixer control can be devised for those models if
592 * necessary.
593 */
594#ifdef CONFIG_SND_DEBUG
Takashi Iwaia5ce8892007-07-23 15:42:26 +0200595#define alc_spdif_ctrl_info snd_ctl_boolean_mono_info
Takashi Iwaif12ab1e2007-04-12 15:51:47 +0200596
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200597static int alc_spdif_ctrl_get(struct snd_kcontrol *kcontrol,
598 struct snd_ctl_elem_value *ucontrol)
Jonathan Woithe92621f12006-02-28 11:47:47 +0100599{
600 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
601 hda_nid_t nid = kcontrol->private_value & 0xffff;
602 unsigned char mask = (kcontrol->private_value >> 16) & 0xff;
603 long *valp = ucontrol->value.integer.value;
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200604 unsigned int val = snd_hda_codec_read(codec, nid, 0,
Andrew Paprocki3982d172007-12-19 12:13:44 +0100605 AC_VERB_GET_DIGI_CONVERT_1, 0x00);
Jonathan Woithe92621f12006-02-28 11:47:47 +0100606
607 *valp = (val & mask) != 0;
608 return 0;
609}
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200610static int alc_spdif_ctrl_put(struct snd_kcontrol *kcontrol,
611 struct snd_ctl_elem_value *ucontrol)
Jonathan Woithe92621f12006-02-28 11:47:47 +0100612{
613 signed int change;
614 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
615 hda_nid_t nid = kcontrol->private_value & 0xffff;
616 unsigned char mask = (kcontrol->private_value >> 16) & 0xff;
617 long val = *ucontrol->value.integer.value;
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200618 unsigned int ctrl_data = snd_hda_codec_read(codec, nid, 0,
Andrew Paprocki3982d172007-12-19 12:13:44 +0100619 AC_VERB_GET_DIGI_CONVERT_1,
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200620 0x00);
Jonathan Woithe92621f12006-02-28 11:47:47 +0100621
622 /* Set/unset the masked control bit(s) as needed */
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200623 change = (val == 0 ? 0 : mask) != (ctrl_data & mask);
Jonathan Woithe92621f12006-02-28 11:47:47 +0100624 if (val==0)
625 ctrl_data &= ~mask;
626 else
627 ctrl_data |= mask;
Takashi Iwai82beb8f2007-08-10 17:09:26 +0200628 snd_hda_codec_write_cache(codec, nid, 0, AC_VERB_SET_DIGI_CONVERT_1,
629 ctrl_data);
Jonathan Woithe92621f12006-02-28 11:47:47 +0100630
631 return change;
632}
633#define ALC_SPDIF_CTRL_SWITCH(xname, nid, mask) \
634 { .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, .index = 0, \
635 .info = alc_spdif_ctrl_info, \
636 .get = alc_spdif_ctrl_get, \
637 .put = alc_spdif_ctrl_put, \
638 .private_value = nid | (mask<<16) }
639#endif /* CONFIG_SND_DEBUG */
640
Jonathan Woithef8225f62008-01-08 12:16:54 +0100641/* A switch control to allow the enabling EAPD digital outputs on the ALC26x.
642 * Again, this is only used in the ALC26x test models to help identify when
643 * the EAPD line must be asserted for features to work.
644 */
645#ifdef CONFIG_SND_DEBUG
646#define alc_eapd_ctrl_info snd_ctl_boolean_mono_info
647
648static int alc_eapd_ctrl_get(struct snd_kcontrol *kcontrol,
649 struct snd_ctl_elem_value *ucontrol)
650{
651 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
652 hda_nid_t nid = kcontrol->private_value & 0xffff;
653 unsigned char mask = (kcontrol->private_value >> 16) & 0xff;
654 long *valp = ucontrol->value.integer.value;
655 unsigned int val = snd_hda_codec_read(codec, nid, 0,
656 AC_VERB_GET_EAPD_BTLENABLE, 0x00);
657
658 *valp = (val & mask) != 0;
659 return 0;
660}
661
662static int alc_eapd_ctrl_put(struct snd_kcontrol *kcontrol,
663 struct snd_ctl_elem_value *ucontrol)
664{
665 int change;
666 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
667 hda_nid_t nid = kcontrol->private_value & 0xffff;
668 unsigned char mask = (kcontrol->private_value >> 16) & 0xff;
669 long val = *ucontrol->value.integer.value;
670 unsigned int ctrl_data = snd_hda_codec_read(codec, nid, 0,
671 AC_VERB_GET_EAPD_BTLENABLE,
672 0x00);
673
674 /* Set/unset the masked control bit(s) as needed */
675 change = (!val ? 0 : mask) != (ctrl_data & mask);
676 if (!val)
677 ctrl_data &= ~mask;
678 else
679 ctrl_data |= mask;
680 snd_hda_codec_write_cache(codec, nid, 0, AC_VERB_SET_EAPD_BTLENABLE,
681 ctrl_data);
682
683 return change;
684}
685
686#define ALC_EAPD_CTRL_SWITCH(xname, nid, mask) \
687 { .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, .index = 0, \
688 .info = alc_eapd_ctrl_info, \
689 .get = alc_eapd_ctrl_get, \
690 .put = alc_eapd_ctrl_put, \
691 .private_value = nid | (mask<<16) }
692#endif /* CONFIG_SND_DEBUG */
693
Kailang Yangdf694da2005-12-05 19:42:22 +0100694/*
695 * set up from the preset table
696 */
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200697static void setup_preset(struct alc_spec *spec,
698 const struct alc_config_preset *preset)
Kailang Yangdf694da2005-12-05 19:42:22 +0100699{
700 int i;
701
702 for (i = 0; i < ARRAY_SIZE(preset->mixers) && preset->mixers[i]; i++)
703 spec->mixers[spec->num_mixers++] = preset->mixers[i];
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200704 for (i = 0; i < ARRAY_SIZE(preset->init_verbs) && preset->init_verbs[i];
705 i++)
706 spec->init_verbs[spec->num_init_verbs++] =
707 preset->init_verbs[i];
Kailang Yangdf694da2005-12-05 19:42:22 +0100708
709 spec->channel_mode = preset->channel_mode;
710 spec->num_channel_mode = preset->num_channel_mode;
Takashi Iwai4e195a72006-07-28 14:47:34 +0200711 spec->need_dac_fix = preset->need_dac_fix;
Kailang Yangdf694da2005-12-05 19:42:22 +0100712
713 spec->multiout.max_channels = spec->channel_mode[0].channels;
714
715 spec->multiout.num_dacs = preset->num_dacs;
716 spec->multiout.dac_nids = preset->dac_nids;
717 spec->multiout.dig_out_nid = preset->dig_out_nid;
718 spec->multiout.hp_nid = preset->hp_nid;
719
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200720 spec->num_mux_defs = preset->num_mux_defs;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +0200721 if (!spec->num_mux_defs)
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200722 spec->num_mux_defs = 1;
Kailang Yangdf694da2005-12-05 19:42:22 +0100723 spec->input_mux = preset->input_mux;
724
725 spec->num_adc_nids = preset->num_adc_nids;
726 spec->adc_nids = preset->adc_nids;
Takashi Iwaie1406342008-02-11 18:32:32 +0100727 spec->capsrc_nids = preset->capsrc_nids;
Kailang Yangdf694da2005-12-05 19:42:22 +0100728 spec->dig_in_nid = preset->dig_in_nid;
Takashi Iwaiae6b8132006-03-03 16:47:17 +0100729
730 spec->unsol_event = preset->unsol_event;
731 spec->init_hook = preset->init_hook;
Takashi Iwaicb53c622007-08-10 17:21:45 +0200732#ifdef CONFIG_SND_HDA_POWER_SAVE
733 spec->loopback.amplist = preset->loopbacks;
734#endif
Kailang Yangdf694da2005-12-05 19:42:22 +0100735}
736
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200737/* Enable GPIO mask and set output */
738static struct hda_verb alc_gpio1_init_verbs[] = {
739 {0x01, AC_VERB_SET_GPIO_MASK, 0x01},
740 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x01},
741 {0x01, AC_VERB_SET_GPIO_DATA, 0x01},
742 { }
743};
744
745static struct hda_verb alc_gpio2_init_verbs[] = {
746 {0x01, AC_VERB_SET_GPIO_MASK, 0x02},
747 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x02},
748 {0x01, AC_VERB_SET_GPIO_DATA, 0x02},
749 { }
750};
751
Kailang Yangbdd148a2007-05-08 15:19:08 +0200752static struct hda_verb alc_gpio3_init_verbs[] = {
753 {0x01, AC_VERB_SET_GPIO_MASK, 0x03},
754 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x03},
755 {0x01, AC_VERB_SET_GPIO_DATA, 0x03},
756 { }
757};
758
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +0200759/*
760 * Fix hardware PLL issue
761 * On some codecs, the analog PLL gating control must be off while
762 * the default value is 1.
763 */
764static void alc_fix_pll(struct hda_codec *codec)
765{
766 struct alc_spec *spec = codec->spec;
767 unsigned int val;
768
769 if (!spec->pll_nid)
770 return;
771 snd_hda_codec_write(codec, spec->pll_nid, 0, AC_VERB_SET_COEF_INDEX,
772 spec->pll_coef_idx);
773 val = snd_hda_codec_read(codec, spec->pll_nid, 0,
774 AC_VERB_GET_PROC_COEF, 0);
775 snd_hda_codec_write(codec, spec->pll_nid, 0, AC_VERB_SET_COEF_INDEX,
776 spec->pll_coef_idx);
777 snd_hda_codec_write(codec, spec->pll_nid, 0, AC_VERB_SET_PROC_COEF,
778 val & ~(1 << spec->pll_coef_bit));
779}
780
781static void alc_fix_pll_init(struct hda_codec *codec, hda_nid_t nid,
782 unsigned int coef_idx, unsigned int coef_bit)
783{
784 struct alc_spec *spec = codec->spec;
785 spec->pll_nid = nid;
786 spec->pll_coef_idx = coef_idx;
787 spec->pll_coef_bit = coef_bit;
788 alc_fix_pll(codec);
789}
790
Kailang Yangc9b58002007-10-16 14:30:01 +0200791static void alc_sku_automute(struct hda_codec *codec)
792{
793 struct alc_spec *spec = codec->spec;
Kailang Yangc9b58002007-10-16 14:30:01 +0200794 unsigned int present;
795 unsigned int hp_nid = spec->autocfg.hp_pins[0];
796 unsigned int sp_nid = spec->autocfg.speaker_pins[0];
797
798 /* need to execute and sync at first */
799 snd_hda_codec_read(codec, hp_nid, 0, AC_VERB_SET_PIN_SENSE, 0);
800 present = snd_hda_codec_read(codec, hp_nid, 0,
801 AC_VERB_GET_PIN_SENSE, 0);
802 spec->jack_present = (present & 0x80000000) != 0;
Takashi Iwaif6c7e542008-02-12 18:32:23 +0100803 snd_hda_codec_write(codec, sp_nid, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
804 spec->jack_present ? 0 : PIN_OUT);
Kailang Yangc9b58002007-10-16 14:30:01 +0200805}
806
807/* unsolicited event for HP jack sensing */
808static void alc_sku_unsol_event(struct hda_codec *codec, unsigned int res)
809{
810 if (codec->vendor_id == 0x10ec0880)
811 res >>= 28;
812 else
813 res >>= 26;
814 if (res != ALC880_HP_EVENT)
815 return;
816
817 alc_sku_automute(codec);
818}
819
Kailang Yangf9423e72008-05-27 12:32:25 +0200820/* additional initialization for ALC888 variants */
821static void alc888_coef_init(struct hda_codec *codec)
822{
823 unsigned int tmp;
824
825 snd_hda_codec_write(codec, 0x20, 0, AC_VERB_SET_COEF_INDEX, 0);
826 tmp = snd_hda_codec_read(codec, 0x20, 0, AC_VERB_GET_PROC_COEF, 0);
827 snd_hda_codec_write(codec, 0x20, 0, AC_VERB_SET_COEF_INDEX, 7);
828 if ((tmp & 0xf0) == 2)
829 /* alc888S-VC */
830 snd_hda_codec_read(codec, 0x20, 0,
831 AC_VERB_SET_PROC_COEF, 0x830);
832 else
833 /* alc888-VB */
834 snd_hda_codec_read(codec, 0x20, 0,
835 AC_VERB_SET_PROC_COEF, 0x3030);
836}
837
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200838/* 32-bit subsystem ID for BIOS loading in HD Audio codec.
839 * 31 ~ 16 : Manufacture ID
840 * 15 ~ 8 : SKU ID
841 * 7 ~ 0 : Assembly ID
842 * port-A --> pin 39/41, port-E --> pin 14/15, port-D --> pin 35/36
843 */
844static void alc_subsystem_id(struct hda_codec *codec,
845 unsigned int porta, unsigned int porte,
846 unsigned int portd)
847{
Kailang Yangc9b58002007-10-16 14:30:01 +0200848 unsigned int ass, tmp, i;
849 unsigned nid;
850 struct alc_spec *spec = codec->spec;
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200851
Kailang Yangc9b58002007-10-16 14:30:01 +0200852 ass = codec->subsystem_id & 0xffff;
853 if ((ass != codec->bus->pci->subsystem_device) && (ass & 1))
854 goto do_sku;
855
856 /*
857 * 31~30 : port conetcivity
858 * 29~21 : reserve
859 * 20 : PCBEEP input
860 * 19~16 : Check sum (15:1)
861 * 15~1 : Custom
862 * 0 : override
863 */
864 nid = 0x1d;
865 if (codec->vendor_id == 0x10ec0260)
866 nid = 0x17;
867 ass = snd_hda_codec_read(codec, nid, 0,
868 AC_VERB_GET_CONFIG_DEFAULT, 0);
869 if (!(ass & 1) && !(ass & 0x100000))
870 return;
871 if ((ass >> 30) != 1) /* no physical connection */
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200872 return;
873
Kailang Yangc9b58002007-10-16 14:30:01 +0200874 /* check sum */
875 tmp = 0;
876 for (i = 1; i < 16; i++) {
Kailang Yang8c427222008-01-10 13:03:59 +0100877 if ((ass >> i) & 1)
Kailang Yangc9b58002007-10-16 14:30:01 +0200878 tmp++;
879 }
880 if (((ass >> 16) & 0xf) != tmp)
881 return;
882do_sku:
883 /*
884 * 0 : override
885 * 1 : Swap Jack
886 * 2 : 0 --> Desktop, 1 --> Laptop
887 * 3~5 : External Amplifier control
888 * 7~6 : Reserved
889 */
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200890 tmp = (ass & 0x38) >> 3; /* external Amp control */
891 switch (tmp) {
892 case 1:
893 snd_hda_sequence_write(codec, alc_gpio1_init_verbs);
894 break;
895 case 3:
896 snd_hda_sequence_write(codec, alc_gpio2_init_verbs);
897 break;
Kailang Yangbdd148a2007-05-08 15:19:08 +0200898 case 7:
899 snd_hda_sequence_write(codec, alc_gpio3_init_verbs);
900 break;
Kailang Yangc9b58002007-10-16 14:30:01 +0200901 case 5: /* set EAPD output high */
Kailang Yangbdd148a2007-05-08 15:19:08 +0200902 switch (codec->vendor_id) {
Kailang Yangc9b58002007-10-16 14:30:01 +0200903 case 0x10ec0260:
904 snd_hda_codec_write(codec, 0x0f, 0,
905 AC_VERB_SET_EAPD_BTLENABLE, 2);
906 snd_hda_codec_write(codec, 0x10, 0,
907 AC_VERB_SET_EAPD_BTLENABLE, 2);
908 break;
909 case 0x10ec0262:
Kailang Yangbdd148a2007-05-08 15:19:08 +0200910 case 0x10ec0267:
911 case 0x10ec0268:
Kailang Yangc9b58002007-10-16 14:30:01 +0200912 case 0x10ec0269:
Kailang Yangf9423e72008-05-27 12:32:25 +0200913 case 0x10ec0660:
914 case 0x10ec0662:
915 case 0x10ec0663:
Kailang Yangc9b58002007-10-16 14:30:01 +0200916 case 0x10ec0862:
Takashi Iwai20a3a052008-05-23 17:52:53 +0200917 case 0x10ec0889:
Kailang Yangbdd148a2007-05-08 15:19:08 +0200918 snd_hda_codec_write(codec, 0x14, 0,
919 AC_VERB_SET_EAPD_BTLENABLE, 2);
920 snd_hda_codec_write(codec, 0x15, 0,
921 AC_VERB_SET_EAPD_BTLENABLE, 2);
Kailang Yangc9b58002007-10-16 14:30:01 +0200922 break;
Kailang Yangbdd148a2007-05-08 15:19:08 +0200923 }
Kailang Yangc9b58002007-10-16 14:30:01 +0200924 switch (codec->vendor_id) {
925 case 0x10ec0260:
926 snd_hda_codec_write(codec, 0x1a, 0,
927 AC_VERB_SET_COEF_INDEX, 7);
928 tmp = snd_hda_codec_read(codec, 0x1a, 0,
929 AC_VERB_GET_PROC_COEF, 0);
930 snd_hda_codec_write(codec, 0x1a, 0,
931 AC_VERB_SET_COEF_INDEX, 7);
932 snd_hda_codec_write(codec, 0x1a, 0,
933 AC_VERB_SET_PROC_COEF,
934 tmp | 0x2010);
935 break;
936 case 0x10ec0262:
937 case 0x10ec0880:
938 case 0x10ec0882:
939 case 0x10ec0883:
940 case 0x10ec0885:
Takashi Iwai20a3a052008-05-23 17:52:53 +0200941 case 0x10ec0889:
Kailang Yangc9b58002007-10-16 14:30:01 +0200942 snd_hda_codec_write(codec, 0x20, 0,
943 AC_VERB_SET_COEF_INDEX, 7);
944 tmp = snd_hda_codec_read(codec, 0x20, 0,
945 AC_VERB_GET_PROC_COEF, 0);
946 snd_hda_codec_write(codec, 0x20, 0,
947 AC_VERB_SET_COEF_INDEX, 7);
948 snd_hda_codec_write(codec, 0x20, 0,
949 AC_VERB_SET_PROC_COEF,
950 tmp | 0x2010);
951 break;
Kailang Yangf9423e72008-05-27 12:32:25 +0200952 case 0x10ec0888:
953 alc888_coef_init(codec);
954 break;
Kailang Yangc9b58002007-10-16 14:30:01 +0200955 case 0x10ec0267:
956 case 0x10ec0268:
957 snd_hda_codec_write(codec, 0x20, 0,
958 AC_VERB_SET_COEF_INDEX, 7);
959 tmp = snd_hda_codec_read(codec, 0x20, 0,
960 AC_VERB_GET_PROC_COEF, 0);
961 snd_hda_codec_write(codec, 0x20, 0,
962 AC_VERB_SET_COEF_INDEX, 7);
963 snd_hda_codec_write(codec, 0x20, 0,
964 AC_VERB_SET_PROC_COEF,
965 tmp | 0x3000);
966 break;
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200967 }
Kailang Yangc9b58002007-10-16 14:30:01 +0200968 default:
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200969 break;
970 }
Kailang Yangc9b58002007-10-16 14:30:01 +0200971
Kailang Yang8c427222008-01-10 13:03:59 +0100972 /* is laptop or Desktop and enable the function "Mute internal speaker
Kailang Yangc9b58002007-10-16 14:30:01 +0200973 * when the external headphone out jack is plugged"
974 */
Kailang Yang8c427222008-01-10 13:03:59 +0100975 if (!(ass & 0x8000))
Kailang Yangc9b58002007-10-16 14:30:01 +0200976 return;
977 /*
978 * 10~8 : Jack location
979 * 12~11: Headphone out -> 00: PortA, 01: PortE, 02: PortD, 03: Resvered
980 * 14~13: Resvered
981 * 15 : 1 --> enable the function "Mute internal speaker
982 * when the external headphone out jack is plugged"
983 */
984 if (!spec->autocfg.speaker_pins[0]) {
Kailang Yang8c427222008-01-10 13:03:59 +0100985 if (spec->autocfg.line_out_pins[0])
Kailang Yangc9b58002007-10-16 14:30:01 +0200986 spec->autocfg.speaker_pins[0] =
Kailang Yang8c427222008-01-10 13:03:59 +0100987 spec->autocfg.line_out_pins[0];
Kailang Yangc9b58002007-10-16 14:30:01 +0200988 else
989 return;
990 }
991
992 if (!spec->autocfg.hp_pins[0]) {
993 tmp = (ass >> 11) & 0x3; /* HP to chassis */
994 if (tmp == 0)
995 spec->autocfg.hp_pins[0] = porta;
996 else if (tmp == 1)
997 spec->autocfg.hp_pins[0] = porte;
998 else if (tmp == 2)
999 spec->autocfg.hp_pins[0] = portd;
1000 else
1001 return;
1002 }
1003
1004 snd_hda_codec_write(codec, spec->autocfg.hp_pins[0], 0,
1005 AC_VERB_SET_UNSOLICITED_ENABLE,
1006 AC_USRSP_EN | ALC880_HP_EVENT);
1007 spec->unsol_event = alc_sku_unsol_event;
Kailang Yangbc9f98a2007-04-12 13:06:07 +02001008}
1009
Takashi Iwai41e41f12005-06-08 14:48:49 +02001010/*
Takashi Iwaif95474e2007-07-10 00:47:43 +02001011 * Fix-up pin default configurations
1012 */
1013
1014struct alc_pincfg {
1015 hda_nid_t nid;
1016 u32 val;
1017};
1018
1019static void alc_fix_pincfg(struct hda_codec *codec,
1020 const struct snd_pci_quirk *quirk,
1021 const struct alc_pincfg **pinfix)
1022{
1023 const struct alc_pincfg *cfg;
1024
1025 quirk = snd_pci_quirk_lookup(codec->bus->pci, quirk);
1026 if (!quirk)
1027 return;
1028
1029 cfg = pinfix[quirk->value];
1030 for (; cfg->nid; cfg++) {
1031 int i;
1032 u32 val = cfg->val;
1033 for (i = 0; i < 4; i++) {
1034 snd_hda_codec_write(codec, cfg->nid, 0,
1035 AC_VERB_SET_CONFIG_DEFAULT_BYTES_0 + i,
1036 val & 0xff);
1037 val >>= 8;
1038 }
1039 }
1040}
1041
1042/*
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001043 * ALC880 3-stack model
1044 *
1045 * DAC: Front = 0x02 (0x0c), Surr = 0x05 (0x0f), CLFE = 0x04 (0x0e)
Takashi Iwai9c7f8522006-06-28 15:08:22 +02001046 * Pin assignment: Front = 0x14, Line-In/Surr = 0x1a, Mic/CLFE = 0x18,
1047 * F-Mic = 0x1b, HP = 0x19
Linus Torvalds1da177e2005-04-16 15:20:36 -07001048 */
1049
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001050static hda_nid_t alc880_dac_nids[4] = {
1051 /* front, rear, clfe, rear_surr */
1052 0x02, 0x05, 0x04, 0x03
1053};
1054
1055static hda_nid_t alc880_adc_nids[3] = {
1056 /* ADC0-2 */
1057 0x07, 0x08, 0x09,
1058};
1059
1060/* The datasheet says the node 0x07 is connected from inputs,
1061 * but it shows zero connection in the real implementation on some devices.
Kailang Yangdf694da2005-12-05 19:42:22 +01001062 * Note: this is a 915GAV bug, fixed on 915GLV
Linus Torvalds1da177e2005-04-16 15:20:36 -07001063 */
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001064static hda_nid_t alc880_adc_nids_alt[2] = {
1065 /* ADC1-2 */
1066 0x08, 0x09,
1067};
1068
1069#define ALC880_DIGOUT_NID 0x06
1070#define ALC880_DIGIN_NID 0x0a
1071
1072static struct hda_input_mux alc880_capture_source = {
1073 .num_items = 4,
1074 .items = {
1075 { "Mic", 0x0 },
1076 { "Front Mic", 0x3 },
1077 { "Line", 0x2 },
1078 { "CD", 0x4 },
Linus Torvalds1da177e2005-04-16 15:20:36 -07001079 },
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001080};
1081
1082/* channel source setting (2/6 channel selection for 3-stack) */
1083/* 2ch mode */
1084static struct hda_verb alc880_threestack_ch2_init[] = {
1085 /* set line-in to input, mute it */
1086 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
1087 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
1088 /* set mic-in to input vref 80%, mute it */
1089 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
1090 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
Linus Torvalds1da177e2005-04-16 15:20:36 -07001091 { } /* end */
1092};
1093
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001094/* 6ch mode */
1095static struct hda_verb alc880_threestack_ch6_init[] = {
1096 /* set line-in to output, unmute it */
1097 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
1098 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
1099 /* set mic-in to output, unmute it */
1100 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
1101 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
1102 { } /* end */
1103};
1104
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +01001105static struct hda_channel_mode alc880_threestack_modes[2] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001106 { 2, alc880_threestack_ch2_init },
1107 { 6, alc880_threestack_ch6_init },
1108};
1109
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01001110static struct snd_kcontrol_new alc880_three_stack_mixer[] = {
Takashi Iwai05acb862005-06-10 19:50:25 +02001111 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001112 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Takashi Iwai05acb862005-06-10 19:50:25 +02001113 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0f, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001114 HDA_BIND_MUTE("Surround Playback Switch", 0x0f, 2, HDA_INPUT),
Takashi Iwai05acb862005-06-10 19:50:25 +02001115 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
1116 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001117 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
1118 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
Linus Torvalds1da177e2005-04-16 15:20:36 -07001119 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
1120 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
1121 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
1122 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
1123 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
1124 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
1125 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x3, HDA_INPUT),
1126 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x3, HDA_INPUT),
1127 HDA_CODEC_VOLUME("PC Speaker Playback Volume", 0x0b, 0x05, HDA_INPUT),
1128 HDA_CODEC_MUTE("PC Speaker Playback Switch", 0x0b, 0x05, HDA_INPUT),
Linus Torvalds1da177e2005-04-16 15:20:36 -07001129 HDA_CODEC_MUTE("Headphone Playback Switch", 0x19, 0x0, HDA_OUTPUT),
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001130 {
1131 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
1132 .name = "Channel Mode",
Kailang Yangdf694da2005-12-05 19:42:22 +01001133 .info = alc_ch_mode_info,
1134 .get = alc_ch_mode_get,
1135 .put = alc_ch_mode_put,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001136 },
1137 { } /* end */
1138};
1139
1140/* capture mixer elements */
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01001141static struct snd_kcontrol_new alc880_capture_mixer[] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001142 HDA_CODEC_VOLUME("Capture Volume", 0x07, 0x0, HDA_INPUT),
1143 HDA_CODEC_MUTE("Capture Switch", 0x07, 0x0, HDA_INPUT),
1144 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x08, 0x0, HDA_INPUT),
1145 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x08, 0x0, HDA_INPUT),
1146 HDA_CODEC_VOLUME_IDX("Capture Volume", 2, 0x09, 0x0, HDA_INPUT),
1147 HDA_CODEC_MUTE_IDX("Capture Switch", 2, 0x09, 0x0, HDA_INPUT),
1148 {
1149 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
1150 /* The multiple "Capture Source" controls confuse alsamixer
1151 * So call somewhat different..
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001152 */
1153 /* .name = "Capture Source", */
1154 .name = "Input Source",
1155 .count = 3,
1156 .info = alc_mux_enum_info,
1157 .get = alc_mux_enum_get,
1158 .put = alc_mux_enum_put,
1159 },
1160 { } /* end */
1161};
1162
1163/* capture mixer elements (in case NID 0x07 not available) */
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01001164static struct snd_kcontrol_new alc880_capture_alt_mixer[] = {
Takashi Iwai71fe7b82005-05-25 18:11:40 +02001165 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
1166 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
1167 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x09, 0x0, HDA_INPUT),
1168 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x09, 0x0, HDA_INPUT),
Linus Torvalds1da177e2005-04-16 15:20:36 -07001169 {
1170 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
1171 /* The multiple "Capture Source" controls confuse alsamixer
1172 * So call somewhat different..
Linus Torvalds1da177e2005-04-16 15:20:36 -07001173 */
1174 /* .name = "Capture Source", */
1175 .name = "Input Source",
1176 .count = 2,
1177 .info = alc_mux_enum_info,
1178 .get = alc_mux_enum_get,
1179 .put = alc_mux_enum_put,
1180 },
Linus Torvalds1da177e2005-04-16 15:20:36 -07001181 { } /* end */
1182};
1183
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001184
1185
1186/*
1187 * ALC880 5-stack model
1188 *
Takashi Iwai9c7f8522006-06-28 15:08:22 +02001189 * DAC: Front = 0x02 (0x0c), Surr = 0x05 (0x0f), CLFE = 0x04 (0x0d),
1190 * Side = 0x02 (0xd)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001191 * Pin assignment: Front = 0x14, Surr = 0x17, CLFE = 0x16
1192 * Line-In/Side = 0x1a, Mic = 0x18, F-Mic = 0x1b, HP = 0x19
1193 */
1194
1195/* additional mixers to alc880_three_stack_mixer */
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01001196static struct snd_kcontrol_new alc880_five_stack_mixer[] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001197 HDA_CODEC_VOLUME("Side Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001198 HDA_BIND_MUTE("Side Playback Switch", 0x0d, 2, HDA_INPUT),
Linus Torvalds1da177e2005-04-16 15:20:36 -07001199 { } /* end */
1200};
1201
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001202/* channel source setting (6/8 channel selection for 5-stack) */
1203/* 6ch mode */
1204static struct hda_verb alc880_fivestack_ch6_init[] = {
1205 /* set line-in to input, mute it */
1206 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
1207 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
Takashi Iwaidfc0ff62005-05-12 14:31:49 +02001208 { } /* end */
1209};
1210
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001211/* 8ch mode */
1212static struct hda_verb alc880_fivestack_ch8_init[] = {
1213 /* set line-in to output, unmute it */
1214 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
1215 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
1216 { } /* end */
1217};
1218
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +01001219static struct hda_channel_mode alc880_fivestack_modes[2] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001220 { 6, alc880_fivestack_ch6_init },
1221 { 8, alc880_fivestack_ch8_init },
1222};
1223
1224
1225/*
1226 * ALC880 6-stack model
1227 *
Takashi Iwai9c7f8522006-06-28 15:08:22 +02001228 * DAC: Front = 0x02 (0x0c), Surr = 0x03 (0x0d), CLFE = 0x04 (0x0e),
1229 * Side = 0x05 (0x0f)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001230 * Pin assignment: Front = 0x14, Surr = 0x15, CLFE = 0x16, Side = 0x17,
1231 * Mic = 0x18, F-Mic = 0x19, Line = 0x1a, HP = 0x1b
1232 */
1233
1234static hda_nid_t alc880_6st_dac_nids[4] = {
1235 /* front, rear, clfe, rear_surr */
1236 0x02, 0x03, 0x04, 0x05
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02001237};
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001238
1239static struct hda_input_mux alc880_6stack_capture_source = {
1240 .num_items = 4,
1241 .items = {
1242 { "Mic", 0x0 },
1243 { "Front Mic", 0x1 },
1244 { "Line", 0x2 },
1245 { "CD", 0x4 },
1246 },
1247};
1248
1249/* fixed 8-channels */
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +01001250static struct hda_channel_mode alc880_sixstack_modes[1] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001251 { 8, NULL },
1252};
1253
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01001254static struct snd_kcontrol_new alc880_six_stack_mixer[] = {
Takashi Iwai16ded522005-06-10 19:58:24 +02001255 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001256 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02001257 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001258 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02001259 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
1260 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001261 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
1262 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02001263 HDA_CODEC_VOLUME("Side Playback Volume", 0x0f, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001264 HDA_BIND_MUTE("Side Playback Switch", 0x0f, 2, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02001265 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
1266 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
1267 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
1268 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
1269 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
1270 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
1271 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
1272 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
1273 HDA_CODEC_VOLUME("PC Speaker Playback Volume", 0x0b, 0x05, HDA_INPUT),
1274 HDA_CODEC_MUTE("PC Speaker Playback Switch", 0x0b, 0x05, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02001275 {
1276 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
1277 .name = "Channel Mode",
Kailang Yangdf694da2005-12-05 19:42:22 +01001278 .info = alc_ch_mode_info,
1279 .get = alc_ch_mode_get,
1280 .put = alc_ch_mode_put,
Takashi Iwai16ded522005-06-10 19:58:24 +02001281 },
1282 { } /* end */
1283};
1284
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001285
1286/*
1287 * ALC880 W810 model
1288 *
1289 * W810 has rear IO for:
1290 * Front (DAC 02)
1291 * Surround (DAC 03)
1292 * Center/LFE (DAC 04)
1293 * Digital out (06)
1294 *
1295 * The system also has a pair of internal speakers, and a headphone jack.
1296 * These are both connected to Line2 on the codec, hence to DAC 02.
1297 *
1298 * There is a variable resistor to control the speaker or headphone
1299 * volume. This is a hardware-only device without a software API.
1300 *
1301 * Plugging headphones in will disable the internal speakers. This is
1302 * implemented in hardware, not via the driver using jack sense. In
1303 * a similar fashion, plugging into the rear socket marked "front" will
1304 * disable both the speakers and headphones.
1305 *
1306 * For input, there's a microphone jack, and an "audio in" jack.
1307 * These may not do anything useful with this driver yet, because I
1308 * haven't setup any initialization verbs for these yet...
1309 */
1310
1311static hda_nid_t alc880_w810_dac_nids[3] = {
1312 /* front, rear/surround, clfe */
1313 0x02, 0x03, 0x04
1314};
1315
1316/* fixed 6 channels */
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +01001317static struct hda_channel_mode alc880_w810_modes[1] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001318 { 6, NULL }
1319};
1320
1321/* Pin assignment: Front = 0x14, Surr = 0x15, CLFE = 0x16, HP = 0x1b */
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01001322static struct snd_kcontrol_new alc880_w810_base_mixer[] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001323 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001324 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001325 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001326 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001327 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
1328 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001329 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
1330 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001331 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
1332 { } /* end */
1333};
1334
1335
1336/*
1337 * Z710V model
1338 *
1339 * DAC: Front = 0x02 (0x0c), HP = 0x03 (0x0d)
Takashi Iwai9c7f8522006-06-28 15:08:22 +02001340 * Pin assignment: Front = 0x14, HP = 0x15, Mic = 0x18, Mic2 = 0x19(?),
1341 * Line = 0x1a
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001342 */
1343
1344static hda_nid_t alc880_z71v_dac_nids[1] = {
1345 0x02
1346};
1347#define ALC880_Z71V_HP_DAC 0x03
1348
1349/* fixed 2 channels */
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +01001350static struct hda_channel_mode alc880_2_jack_modes[1] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001351 { 2, NULL }
1352};
1353
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01001354static struct snd_kcontrol_new alc880_z71v_mixer[] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001355 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001356 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001357 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001358 HDA_BIND_MUTE("Headphone Playback Switch", 0x0d, 2, HDA_INPUT),
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001359 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
1360 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
1361 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
1362 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
1363 { } /* end */
1364};
1365
1366
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001367/*
1368 * ALC880 F1734 model
1369 *
1370 * DAC: HP = 0x02 (0x0c), Front = 0x03 (0x0d)
1371 * Pin assignment: HP = 0x14, Front = 0x15, Mic = 0x18
1372 */
1373
1374static hda_nid_t alc880_f1734_dac_nids[1] = {
1375 0x03
1376};
1377#define ALC880_F1734_HP_DAC 0x02
1378
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01001379static struct snd_kcontrol_new alc880_f1734_mixer[] = {
Takashi Iwai16ded522005-06-10 19:58:24 +02001380 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001381 HDA_BIND_MUTE("Headphone Playback Switch", 0x0c, 2, HDA_INPUT),
Takashi Iwai2134ea42008-01-10 16:53:55 +01001382 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
1383 HDA_BIND_MUTE("Speaker Playback Switch", 0x0d, 2, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02001384 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
1385 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
Takashi Iwai937b4162008-02-11 14:52:36 +01001386 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
1387 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02001388 { } /* end */
1389};
1390
Takashi Iwai937b4162008-02-11 14:52:36 +01001391static struct hda_input_mux alc880_f1734_capture_source = {
1392 .num_items = 2,
1393 .items = {
1394 { "Mic", 0x1 },
1395 { "CD", 0x4 },
1396 },
1397};
1398
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001399
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001400/*
1401 * ALC880 ASUS model
1402 *
1403 * DAC: HP/Front = 0x02 (0x0c), Surr = 0x03 (0x0d), CLFE = 0x04 (0x0e)
1404 * Pin assignment: HP/Front = 0x14, Surr = 0x15, CLFE = 0x16,
1405 * Mic = 0x18, Line = 0x1a
1406 */
1407
1408#define alc880_asus_dac_nids alc880_w810_dac_nids /* identical with w810 */
1409#define alc880_asus_modes alc880_threestack_modes /* 2/6 channel mode */
1410
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01001411static struct snd_kcontrol_new alc880_asus_mixer[] = {
Takashi Iwai16ded522005-06-10 19:58:24 +02001412 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001413 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02001414 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001415 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02001416 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
1417 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001418 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
1419 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02001420 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
1421 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
1422 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
1423 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
1424 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
1425 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02001426 {
1427 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
1428 .name = "Channel Mode",
Kailang Yangdf694da2005-12-05 19:42:22 +01001429 .info = alc_ch_mode_info,
1430 .get = alc_ch_mode_get,
1431 .put = alc_ch_mode_put,
Takashi Iwai16ded522005-06-10 19:58:24 +02001432 },
1433 { } /* end */
1434};
1435
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001436/*
1437 * ALC880 ASUS W1V model
1438 *
1439 * DAC: HP/Front = 0x02 (0x0c), Surr = 0x03 (0x0d), CLFE = 0x04 (0x0e)
1440 * Pin assignment: HP/Front = 0x14, Surr = 0x15, CLFE = 0x16,
1441 * Mic = 0x18, Line = 0x1a, Line2 = 0x1b
1442 */
1443
1444/* additional mixers to alc880_asus_mixer */
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01001445static struct snd_kcontrol_new alc880_asus_w1v_mixer[] = {
Takashi Iwai16ded522005-06-10 19:58:24 +02001446 HDA_CODEC_VOLUME("Line2 Playback Volume", 0x0b, 0x03, HDA_INPUT),
1447 HDA_CODEC_MUTE("Line2 Playback Switch", 0x0b, 0x03, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02001448 { } /* end */
1449};
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001450
Takashi Iwai3c10a9d2005-08-23 20:02:27 +02001451/* additional mixers to alc880_asus_mixer */
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01001452static struct snd_kcontrol_new alc880_pcbeep_mixer[] = {
Takashi Iwai3c10a9d2005-08-23 20:02:27 +02001453 HDA_CODEC_VOLUME("PC Speaker Playback Volume", 0x0b, 0x05, HDA_INPUT),
1454 HDA_CODEC_MUTE("PC Speaker Playback Switch", 0x0b, 0x05, HDA_INPUT),
1455 { } /* end */
1456};
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001457
Kailang Yangdf694da2005-12-05 19:42:22 +01001458/* TCL S700 */
1459static struct snd_kcontrol_new alc880_tcl_s700_mixer[] = {
1460 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
1461 HDA_CODEC_MUTE("Front Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
1462 HDA_CODEC_MUTE("Headphone Playback Switch", 0x14, 0x0, HDA_OUTPUT),
1463 HDA_CODEC_VOLUME("CD Playback Volume", 0x0B, 0x04, HDA_INPUT),
1464 HDA_CODEC_MUTE("CD Playback Switch", 0x0B, 0x04, HDA_INPUT),
1465 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0B, 0x0, HDA_INPUT),
1466 HDA_CODEC_MUTE("Mic Playback Switch", 0x0B, 0x0, HDA_INPUT),
1467 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
1468 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
1469 {
1470 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
1471 /* The multiple "Capture Source" controls confuse alsamixer
1472 * So call somewhat different..
Kailang Yangdf694da2005-12-05 19:42:22 +01001473 */
1474 /* .name = "Capture Source", */
1475 .name = "Input Source",
1476 .count = 1,
1477 .info = alc_mux_enum_info,
1478 .get = alc_mux_enum_get,
1479 .put = alc_mux_enum_put,
1480 },
1481 { } /* end */
1482};
1483
Kailang Yangccc656c2006-10-17 12:32:26 +02001484/* Uniwill */
1485static struct snd_kcontrol_new alc880_uniwill_mixer[] = {
Takashi Iwai2134ea42008-01-10 16:53:55 +01001486 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
1487 HDA_BIND_MUTE("Headphone Playback Switch", 0x0c, 2, HDA_INPUT),
1488 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
1489 HDA_BIND_MUTE("Speaker Playback Switch", 0x0d, 2, HDA_INPUT),
Kailang Yangccc656c2006-10-17 12:32:26 +02001490 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
1491 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
1492 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
1493 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
1494 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
1495 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
1496 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
1497 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
1498 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
1499 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
1500 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
1501 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
1502 HDA_CODEC_VOLUME("PC Speaker Playback Volume", 0x0b, 0x05, HDA_INPUT),
1503 HDA_CODEC_MUTE("PC Speaker Playback Switch", 0x0b, 0x05, HDA_INPUT),
1504 {
1505 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
1506 .name = "Channel Mode",
1507 .info = alc_ch_mode_info,
1508 .get = alc_ch_mode_get,
1509 .put = alc_ch_mode_put,
1510 },
1511 { } /* end */
1512};
1513
Tobin Davis2cf9f0f2007-02-07 16:04:25 +01001514static struct snd_kcontrol_new alc880_fujitsu_mixer[] = {
1515 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
1516 HDA_BIND_MUTE("Headphone Playback Switch", 0x0c, 2, HDA_INPUT),
1517 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
1518 HDA_BIND_MUTE("Speaker Playback Switch", 0x0d, 2, HDA_INPUT),
1519 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
1520 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
1521 HDA_CODEC_VOLUME("Ext Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
1522 HDA_CODEC_MUTE("Ext Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
1523 HDA_CODEC_VOLUME("Int Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
1524 HDA_CODEC_MUTE("Int Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
1525 { } /* end */
1526};
1527
Kailang Yangccc656c2006-10-17 12:32:26 +02001528static struct snd_kcontrol_new alc880_uniwill_p53_mixer[] = {
Takashi Iwai2134ea42008-01-10 16:53:55 +01001529 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
1530 HDA_BIND_MUTE("Headphone Playback Switch", 0x0c, 2, HDA_INPUT),
1531 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
1532 HDA_BIND_MUTE("Speaker Playback Switch", 0x0d, 2, HDA_INPUT),
Kailang Yangccc656c2006-10-17 12:32:26 +02001533 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
1534 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
1535 { } /* end */
1536};
1537
Linus Torvalds1da177e2005-04-16 15:20:36 -07001538/*
Takashi Iwai2134ea42008-01-10 16:53:55 +01001539 * virtual master controls
1540 */
1541
1542/*
1543 * slave controls for virtual master
1544 */
1545static const char *alc_slave_vols[] = {
1546 "Front Playback Volume",
1547 "Surround Playback Volume",
1548 "Center Playback Volume",
1549 "LFE Playback Volume",
1550 "Side Playback Volume",
1551 "Headphone Playback Volume",
1552 "Speaker Playback Volume",
1553 "Mono Playback Volume",
Takashi Iwai2134ea42008-01-10 16:53:55 +01001554 "Line-Out Playback Volume",
1555 NULL,
1556};
1557
1558static const char *alc_slave_sws[] = {
1559 "Front Playback Switch",
1560 "Surround Playback Switch",
1561 "Center Playback Switch",
1562 "LFE Playback Switch",
1563 "Side Playback Switch",
1564 "Headphone Playback Switch",
1565 "Speaker Playback Switch",
1566 "Mono Playback Switch",
Takashi Iwaiedb54a52008-01-29 12:47:02 +01001567 "IEC958 Playback Switch",
Takashi Iwai2134ea42008-01-10 16:53:55 +01001568 NULL,
1569};
1570
1571/*
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001572 * build control elements
Linus Torvalds1da177e2005-04-16 15:20:36 -07001573 */
1574static int alc_build_controls(struct hda_codec *codec)
1575{
1576 struct alc_spec *spec = codec->spec;
1577 int err;
1578 int i;
1579
1580 for (i = 0; i < spec->num_mixers; i++) {
1581 err = snd_hda_add_new_ctls(codec, spec->mixers[i]);
1582 if (err < 0)
1583 return err;
1584 }
1585
1586 if (spec->multiout.dig_out_nid) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +02001587 err = snd_hda_create_spdif_out_ctls(codec,
1588 spec->multiout.dig_out_nid);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001589 if (err < 0)
1590 return err;
Takashi Iwai9a081602008-02-12 18:37:26 +01001591 err = snd_hda_create_spdif_share_sw(codec,
1592 &spec->multiout);
1593 if (err < 0)
1594 return err;
1595 spec->multiout.share_spdif = 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001596 }
1597 if (spec->dig_in_nid) {
1598 err = snd_hda_create_spdif_in_ctls(codec, spec->dig_in_nid);
1599 if (err < 0)
1600 return err;
1601 }
Takashi Iwai2134ea42008-01-10 16:53:55 +01001602
1603 /* if we have no master control, let's create it */
1604 if (!snd_hda_find_mixer_ctl(codec, "Master Playback Volume")) {
Takashi Iwai1c82ed12008-02-18 13:05:50 +01001605 unsigned int vmaster_tlv[4];
Takashi Iwai2134ea42008-01-10 16:53:55 +01001606 snd_hda_set_vmaster_tlv(codec, spec->vmaster_nid,
Takashi Iwai1c82ed12008-02-18 13:05:50 +01001607 HDA_OUTPUT, vmaster_tlv);
Takashi Iwai2134ea42008-01-10 16:53:55 +01001608 err = snd_hda_add_vmaster(codec, "Master Playback Volume",
Takashi Iwai1c82ed12008-02-18 13:05:50 +01001609 vmaster_tlv, alc_slave_vols);
Takashi Iwai2134ea42008-01-10 16:53:55 +01001610 if (err < 0)
1611 return err;
1612 }
1613 if (!snd_hda_find_mixer_ctl(codec, "Master Playback Switch")) {
1614 err = snd_hda_add_vmaster(codec, "Master Playback Switch",
1615 NULL, alc_slave_sws);
1616 if (err < 0)
1617 return err;
1618 }
1619
Linus Torvalds1da177e2005-04-16 15:20:36 -07001620 return 0;
1621}
1622
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001623
Linus Torvalds1da177e2005-04-16 15:20:36 -07001624/*
1625 * initialize the codec volumes, etc
1626 */
1627
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001628/*
1629 * generic initialization of ADC, input mixers and output mixers
1630 */
1631static struct hda_verb alc880_volume_init_verbs[] = {
1632 /*
1633 * Unmute ADC0-2 and set the default input to mic-in
1634 */
Takashi Iwai71fe7b82005-05-25 18:11:40 +02001635 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001636 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Takashi Iwai71fe7b82005-05-25 18:11:40 +02001637 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001638 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Takashi Iwai71fe7b82005-05-25 18:11:40 +02001639 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001640 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07001641
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001642 /* Unmute input amps (CD, Line In, Mic 1 & Mic 2) of the analog-loopback
1643 * mixer widget
Takashi Iwai9c7f8522006-06-28 15:08:22 +02001644 * Note: PASD motherboards uses the Line In 2 as the input for front
1645 * panel mic (mic 2)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001646 */
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001647 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +02001648 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
1649 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
1650 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
1651 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
1652 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
1653 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(6)},
1654 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(7)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07001655
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001656 /*
1657 * Set up output mixers (0x0c - 0x0f)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001658 */
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001659 /* set vol=0 to output mixers */
1660 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
1661 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
1662 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
1663 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
1664 /* set up input amps for analog loopback */
1665 /* Amp Indices: DAC = 0, mixer = 1 */
Takashi Iwai05acb862005-06-10 19:50:25 +02001666 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
1667 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Takashi Iwai05acb862005-06-10 19:50:25 +02001668 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
1669 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Takashi Iwai05acb862005-06-10 19:50:25 +02001670 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
1671 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Takashi Iwai05acb862005-06-10 19:50:25 +02001672 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
1673 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07001674
1675 { }
1676};
1677
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001678/*
1679 * 3-stack pin configuration:
1680 * front = 0x14, mic/clfe = 0x18, HP = 0x19, line/surr = 0x1a, f-mic = 0x1b
1681 */
1682static struct hda_verb alc880_pin_3stack_init_verbs[] = {
1683 /*
1684 * preset connection lists of input pins
1685 * 0 = front, 1 = rear_surr, 2 = CLFE, 3 = surround
1686 */
1687 {0x10, AC_VERB_SET_CONNECT_SEL, 0x02}, /* mic/clfe */
1688 {0x11, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
1689 {0x12, AC_VERB_SET_CONNECT_SEL, 0x03}, /* line/surround */
1690
1691 /*
1692 * Set pin mode and muting
1693 */
1694 /* set front pin widgets 0x14 for output */
1695 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1696 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1697 /* Mic1 (rear panel) pin widget for input and vref at 80% */
1698 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
1699 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
1700 /* Mic2 (as headphone out) for HP output */
1701 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
1702 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1703 /* Line In pin widget for input */
1704 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
1705 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
1706 /* Line2 (as front mic) pin widget for input and vref at 80% */
1707 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
1708 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
1709 /* CD pin widget for input */
1710 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
1711
1712 { }
1713};
1714
1715/*
1716 * 5-stack pin configuration:
1717 * front = 0x14, surround = 0x17, clfe = 0x16, mic = 0x18, HP = 0x19,
1718 * line-in/side = 0x1a, f-mic = 0x1b
1719 */
1720static struct hda_verb alc880_pin_5stack_init_verbs[] = {
1721 /*
1722 * preset connection lists of input pins
1723 * 0 = front, 1 = rear_surr, 2 = CLFE, 3 = surround
1724 */
1725 {0x11, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
1726 {0x12, AC_VERB_SET_CONNECT_SEL, 0x01}, /* line/side */
1727
1728 /*
1729 * Set pin mode and muting
1730 */
1731 /* set pin widgets 0x14-0x17 for output */
Takashi Iwai05acb862005-06-10 19:50:25 +02001732 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1733 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1734 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1735 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001736 /* unmute pins for output (no gain on this amp) */
1737 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1738 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1739 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1740 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1741
Linus Torvalds1da177e2005-04-16 15:20:36 -07001742 /* Mic1 (rear panel) pin widget for input and vref at 80% */
Takashi Iwai16ded522005-06-10 19:58:24 +02001743 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001744 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
1745 /* Mic2 (as headphone out) for HP output */
1746 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Takashi Iwai05acb862005-06-10 19:50:25 +02001747 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001748 /* Line In pin widget for input */
1749 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
1750 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
1751 /* Line2 (as front mic) pin widget for input and vref at 80% */
1752 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
1753 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
1754 /* CD pin widget for input */
1755 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Linus Torvalds1da177e2005-04-16 15:20:36 -07001756
1757 { }
1758};
1759
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001760/*
1761 * W810 pin configuration:
1762 * front = 0x14, surround = 0x15, clfe = 0x16, HP = 0x1b
1763 */
1764static struct hda_verb alc880_pin_w810_init_verbs[] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001765 /* hphone/speaker input selector: front DAC */
1766 {0x13, AC_VERB_SET_CONNECT_SEL, 0x0},
1767
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001768 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1769 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1770 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1771 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1772 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1773 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1774
1775 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Takashi Iwai05acb862005-06-10 19:50:25 +02001776 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
Linus Torvalds1da177e2005-04-16 15:20:36 -07001777
Linus Torvalds1da177e2005-04-16 15:20:36 -07001778 { }
1779};
1780
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001781/*
1782 * Z71V pin configuration:
1783 * Speaker-out = 0x14, HP = 0x15, Mic = 0x18, Line-in = 0x1a, Mic2 = 0x1b (?)
1784 */
1785static struct hda_verb alc880_pin_z71v_init_verbs[] = {
Takashi Iwai05acb862005-06-10 19:50:25 +02001786 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001787 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Takashi Iwai05acb862005-06-10 19:50:25 +02001788 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001789 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Takashi Iwaidfc0ff62005-05-12 14:31:49 +02001790
Takashi Iwai16ded522005-06-10 19:58:24 +02001791 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001792 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Takashi Iwai16ded522005-06-10 19:58:24 +02001793 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001794 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Takashi Iwaidfc0ff62005-05-12 14:31:49 +02001795
1796 { }
1797};
1798
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001799/*
1800 * 6-stack pin configuration:
Takashi Iwai9c7f8522006-06-28 15:08:22 +02001801 * front = 0x14, surr = 0x15, clfe = 0x16, side = 0x17, mic = 0x18,
1802 * f-mic = 0x19, line = 0x1a, HP = 0x1b
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001803 */
1804static struct hda_verb alc880_pin_6stack_init_verbs[] = {
1805 {0x13, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
1806
Takashi Iwai16ded522005-06-10 19:58:24 +02001807 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001808 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Takashi Iwai16ded522005-06-10 19:58:24 +02001809 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001810 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Takashi Iwai16ded522005-06-10 19:58:24 +02001811 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001812 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Takashi Iwai16ded522005-06-10 19:58:24 +02001813 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001814 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1815
Takashi Iwai16ded522005-06-10 19:58:24 +02001816 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001817 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
Takashi Iwai16ded522005-06-10 19:58:24 +02001818 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001819 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
Takashi Iwai16ded522005-06-10 19:58:24 +02001820 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001821 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
Takashi Iwai16ded522005-06-10 19:58:24 +02001822 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Takashi Iwai16ded522005-06-10 19:58:24 +02001823 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Takashi Iwai16ded522005-06-10 19:58:24 +02001824 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
1825
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001826 { }
1827};
Takashi Iwai16ded522005-06-10 19:58:24 +02001828
Kailang Yangccc656c2006-10-17 12:32:26 +02001829/*
1830 * Uniwill pin configuration:
1831 * HP = 0x14, InternalSpeaker = 0x15, mic = 0x18, internal mic = 0x19,
1832 * line = 0x1a
1833 */
1834static struct hda_verb alc880_uniwill_init_verbs[] = {
1835 {0x13, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
1836
1837 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
1838 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1839 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
1840 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1841 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
1842 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1843 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1844 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1845 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
1846 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
1847 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
1848 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
1849 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
1850 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
1851
1852 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
1853 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
1854 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
1855 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
1856 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
1857 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
1858 /* {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP}, */
1859 /* {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE}, */
1860 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
1861
1862 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
1863 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
1864
1865 { }
1866};
1867
1868/*
1869* Uniwill P53
1870* HP = 0x14, InternalSpeaker = 0x15, mic = 0x19,
1871 */
1872static struct hda_verb alc880_uniwill_p53_init_verbs[] = {
1873 {0x13, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
1874
1875 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
1876 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1877 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
1878 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1879 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
1880 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1881 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
1882 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
1883 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
1884 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
1885 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
1886 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
1887
1888 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
1889 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
1890 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
1891 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
1892 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
1893 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
1894
1895 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
1896 {0x21, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_DCVOL_EVENT},
1897
1898 { }
1899};
1900
Tobin Davis2cf9f0f2007-02-07 16:04:25 +01001901static struct hda_verb alc880_beep_init_verbs[] = {
1902 { 0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(5) },
1903 { }
1904};
1905
Kailang Yangccc656c2006-10-17 12:32:26 +02001906/* toggle speaker-output according to the hp-jack state */
Takashi Iwai458a4fa2007-05-05 12:18:40 +02001907static void alc880_uniwill_hp_automute(struct hda_codec *codec)
Kailang Yangccc656c2006-10-17 12:32:26 +02001908{
1909 unsigned int present;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02001910 unsigned char bits;
Kailang Yangccc656c2006-10-17 12:32:26 +02001911
1912 present = snd_hda_codec_read(codec, 0x14, 0,
1913 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +02001914 bits = present ? HDA_AMP_MUTE : 0;
1915 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
1916 HDA_AMP_MUTE, bits);
1917 snd_hda_codec_amp_stereo(codec, 0x16, HDA_OUTPUT, 0,
1918 HDA_AMP_MUTE, bits);
Takashi Iwai458a4fa2007-05-05 12:18:40 +02001919}
1920
1921/* auto-toggle front mic */
1922static void alc880_uniwill_mic_automute(struct hda_codec *codec)
1923{
1924 unsigned int present;
1925 unsigned char bits;
Kailang Yangccc656c2006-10-17 12:32:26 +02001926
1927 present = snd_hda_codec_read(codec, 0x18, 0,
1928 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +02001929 bits = present ? HDA_AMP_MUTE : 0;
1930 snd_hda_codec_amp_stereo(codec, 0x0b, HDA_INPUT, 1, HDA_AMP_MUTE, bits);
Takashi Iwai458a4fa2007-05-05 12:18:40 +02001931}
1932
1933static void alc880_uniwill_automute(struct hda_codec *codec)
1934{
1935 alc880_uniwill_hp_automute(codec);
1936 alc880_uniwill_mic_automute(codec);
Kailang Yangccc656c2006-10-17 12:32:26 +02001937}
1938
1939static void alc880_uniwill_unsol_event(struct hda_codec *codec,
1940 unsigned int res)
1941{
1942 /* Looks like the unsol event is incompatible with the standard
1943 * definition. 4bit tag is placed at 28 bit!
1944 */
Takashi Iwai458a4fa2007-05-05 12:18:40 +02001945 switch (res >> 28) {
1946 case ALC880_HP_EVENT:
1947 alc880_uniwill_hp_automute(codec);
1948 break;
1949 case ALC880_MIC_EVENT:
1950 alc880_uniwill_mic_automute(codec);
1951 break;
1952 }
Kailang Yangccc656c2006-10-17 12:32:26 +02001953}
1954
1955static void alc880_uniwill_p53_hp_automute(struct hda_codec *codec)
1956{
1957 unsigned int present;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02001958 unsigned char bits;
Kailang Yangccc656c2006-10-17 12:32:26 +02001959
1960 present = snd_hda_codec_read(codec, 0x14, 0,
1961 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +02001962 bits = present ? HDA_AMP_MUTE : 0;
Jiang zhe64654c22008-04-14 13:26:21 +02001963 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0, HDA_AMP_MUTE, bits);
Kailang Yangccc656c2006-10-17 12:32:26 +02001964}
1965
1966static void alc880_uniwill_p53_dcvol_automute(struct hda_codec *codec)
1967{
1968 unsigned int present;
1969
1970 present = snd_hda_codec_read(codec, 0x21, 0,
Takashi Iwai47fd8302007-08-10 17:11:07 +02001971 AC_VERB_GET_VOLUME_KNOB_CONTROL, 0);
1972 present &= HDA_AMP_VOLMASK;
1973 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_OUTPUT, 0,
1974 HDA_AMP_VOLMASK, present);
1975 snd_hda_codec_amp_stereo(codec, 0x0d, HDA_OUTPUT, 0,
1976 HDA_AMP_VOLMASK, present);
Kailang Yangccc656c2006-10-17 12:32:26 +02001977}
Takashi Iwai47fd8302007-08-10 17:11:07 +02001978
Kailang Yangccc656c2006-10-17 12:32:26 +02001979static void alc880_uniwill_p53_unsol_event(struct hda_codec *codec,
1980 unsigned int res)
1981{
1982 /* Looks like the unsol event is incompatible with the standard
1983 * definition. 4bit tag is placed at 28 bit!
1984 */
1985 if ((res >> 28) == ALC880_HP_EVENT)
1986 alc880_uniwill_p53_hp_automute(codec);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02001987 if ((res >> 28) == ALC880_DCVOL_EVENT)
Kailang Yangccc656c2006-10-17 12:32:26 +02001988 alc880_uniwill_p53_dcvol_automute(codec);
1989}
1990
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001991/*
1992 * F1734 pin configuration:
1993 * HP = 0x14, speaker-out = 0x15, mic = 0x18
1994 */
1995static struct hda_verb alc880_pin_f1734_init_verbs[] = {
Michael Gruberee7a9c72008-03-10 11:30:59 +01001996 {0x07, AC_VERB_SET_CONNECT_SEL, 0x01},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001997 {0x10, AC_VERB_SET_CONNECT_SEL, 0x02},
1998 {0x11, AC_VERB_SET_CONNECT_SEL, 0x00},
1999 {0x12, AC_VERB_SET_CONNECT_SEL, 0x01},
2000 {0x13, AC_VERB_SET_CONNECT_SEL, 0x00},
2001
2002 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2003 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2004 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2005 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2006
2007 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2008 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
Michael Gruberee7a9c72008-03-10 11:30:59 +01002009 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF50},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002010 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2011 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2012 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2013 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2014 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2015 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Takashi Iwai16ded522005-06-10 19:58:24 +02002016
Takashi Iwai937b4162008-02-11 14:52:36 +01002017 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|ALC880_HP_EVENT},
2018 {0x21, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|ALC880_DCVOL_EVENT},
2019
Takashi Iwai16ded522005-06-10 19:58:24 +02002020 { }
2021};
2022
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002023/*
2024 * ASUS pin configuration:
2025 * HP/front = 0x14, surr = 0x15, clfe = 0x16, mic = 0x18, line = 0x1a
2026 */
2027static struct hda_verb alc880_pin_asus_init_verbs[] = {
2028 {0x10, AC_VERB_SET_CONNECT_SEL, 0x02},
2029 {0x11, AC_VERB_SET_CONNECT_SEL, 0x00},
2030 {0x12, AC_VERB_SET_CONNECT_SEL, 0x01},
2031 {0x13, AC_VERB_SET_CONNECT_SEL, 0x00},
2032
2033 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2034 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2035 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2036 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2037 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2038 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2039 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2040 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2041
2042 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2043 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2044 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2045 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2046 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2047 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2048 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2049 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2050 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2051
2052 { }
2053};
2054
2055/* Enable GPIO mask and set output */
Kailang Yangbc9f98a2007-04-12 13:06:07 +02002056#define alc880_gpio1_init_verbs alc_gpio1_init_verbs
2057#define alc880_gpio2_init_verbs alc_gpio2_init_verbs
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002058
Kailang Yangdf694da2005-12-05 19:42:22 +01002059/* Clevo m520g init */
2060static struct hda_verb alc880_pin_clevo_init_verbs[] = {
2061 /* headphone output */
2062 {0x11, AC_VERB_SET_CONNECT_SEL, 0x01},
2063 /* line-out */
2064 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2065 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2066 /* Line-in */
2067 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2068 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2069 /* CD */
2070 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2071 {0x1c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2072 /* Mic1 (rear panel) */
2073 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2074 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2075 /* Mic2 (front panel) */
2076 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2077 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2078 /* headphone */
2079 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2080 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2081 /* change to EAPD mode */
2082 {0x20, AC_VERB_SET_COEF_INDEX, 0x07},
2083 {0x20, AC_VERB_SET_PROC_COEF, 0x3060},
2084
2085 { }
2086};
2087
2088static struct hda_verb alc880_pin_tcl_S700_init_verbs[] = {
Takashi Iwai4b146cb2006-07-28 14:42:36 +02002089 /* change to EAPD mode */
2090 {0x20, AC_VERB_SET_COEF_INDEX, 0x07},
2091 {0x20, AC_VERB_SET_PROC_COEF, 0x3060},
2092
Kailang Yangdf694da2005-12-05 19:42:22 +01002093 /* Headphone output */
2094 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2095 /* Front output*/
2096 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2097 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
2098
2099 /* Line In pin widget for input */
2100 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2101 /* CD pin widget for input */
2102 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2103 /* Mic1 (rear panel) pin widget for input and vref at 80% */
2104 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2105
2106 /* change to EAPD mode */
2107 {0x20, AC_VERB_SET_COEF_INDEX, 0x07},
2108 {0x20, AC_VERB_SET_PROC_COEF, 0x3070},
2109
2110 { }
2111};
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002112
2113/*
Takashi Iwaiae6b8132006-03-03 16:47:17 +01002114 * LG m1 express dual
2115 *
2116 * Pin assignment:
2117 * Rear Line-In/Out (blue): 0x14
2118 * Build-in Mic-In: 0x15
2119 * Speaker-out: 0x17
2120 * HP-Out (green): 0x1b
2121 * Mic-In/Out (red): 0x19
2122 * SPDIF-Out: 0x1e
2123 */
2124
2125/* To make 5.1 output working (green=Front, blue=Surr, red=CLFE) */
2126static hda_nid_t alc880_lg_dac_nids[3] = {
2127 0x05, 0x02, 0x03
2128};
2129
2130/* seems analog CD is not working */
2131static struct hda_input_mux alc880_lg_capture_source = {
2132 .num_items = 3,
2133 .items = {
2134 { "Mic", 0x1 },
2135 { "Line", 0x5 },
2136 { "Internal Mic", 0x6 },
2137 },
2138};
2139
2140/* 2,4,6 channel modes */
2141static struct hda_verb alc880_lg_ch2_init[] = {
2142 /* set line-in and mic-in to input */
2143 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
2144 { 0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
2145 { }
2146};
2147
2148static struct hda_verb alc880_lg_ch4_init[] = {
2149 /* set line-in to out and mic-in to input */
2150 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP },
2151 { 0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
2152 { }
2153};
2154
2155static struct hda_verb alc880_lg_ch6_init[] = {
2156 /* set line-in and mic-in to output */
2157 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP },
2158 { 0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP },
2159 { }
2160};
2161
2162static struct hda_channel_mode alc880_lg_ch_modes[3] = {
2163 { 2, alc880_lg_ch2_init },
2164 { 4, alc880_lg_ch4_init },
2165 { 6, alc880_lg_ch6_init },
2166};
2167
2168static struct snd_kcontrol_new alc880_lg_mixer[] = {
Takashi Iwai2134ea42008-01-10 16:53:55 +01002169 HDA_CODEC_VOLUME("Front Playback Volume", 0x0f, 0x0, HDA_OUTPUT),
2170 HDA_BIND_MUTE("Front Playback Switch", 0x0f, 2, HDA_INPUT),
Takashi Iwaiae6b8132006-03-03 16:47:17 +01002171 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
2172 HDA_BIND_MUTE("Surround Playback Switch", 0x0c, 2, HDA_INPUT),
2173 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0d, 1, 0x0, HDA_OUTPUT),
2174 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0d, 2, 0x0, HDA_OUTPUT),
2175 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0d, 1, 2, HDA_INPUT),
2176 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0d, 2, 2, HDA_INPUT),
2177 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
2178 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
2179 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x06, HDA_INPUT),
2180 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x06, HDA_INPUT),
2181 HDA_CODEC_VOLUME("Internal Mic Playback Volume", 0x0b, 0x07, HDA_INPUT),
2182 HDA_CODEC_MUTE("Internal Mic Playback Switch", 0x0b, 0x07, HDA_INPUT),
2183 {
2184 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2185 .name = "Channel Mode",
2186 .info = alc_ch_mode_info,
2187 .get = alc_ch_mode_get,
2188 .put = alc_ch_mode_put,
2189 },
2190 { } /* end */
2191};
2192
2193static struct hda_verb alc880_lg_init_verbs[] = {
2194 /* set capture source to mic-in */
2195 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
2196 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
2197 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
2198 /* mute all amp mixer inputs */
2199 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(5)},
Takashi Iwaicb53c622007-08-10 17:21:45 +02002200 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(6)},
2201 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(7)},
Takashi Iwaiae6b8132006-03-03 16:47:17 +01002202 /* line-in to input */
2203 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2204 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2205 /* built-in mic */
2206 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2207 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2208 /* speaker-out */
2209 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2210 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2211 /* mic-in to input */
2212 {0x11, AC_VERB_SET_CONNECT_SEL, 0x01},
2213 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2214 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2215 /* HP-out */
2216 {0x13, AC_VERB_SET_CONNECT_SEL, 0x03},
2217 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2218 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2219 /* jack sense */
2220 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | 0x1},
2221 { }
2222};
2223
2224/* toggle speaker-output according to the hp-jack state */
2225static void alc880_lg_automute(struct hda_codec *codec)
2226{
2227 unsigned int present;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02002228 unsigned char bits;
Takashi Iwaiae6b8132006-03-03 16:47:17 +01002229
2230 present = snd_hda_codec_read(codec, 0x1b, 0,
2231 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +02002232 bits = present ? HDA_AMP_MUTE : 0;
2233 snd_hda_codec_amp_stereo(codec, 0x17, HDA_OUTPUT, 0,
2234 HDA_AMP_MUTE, bits);
Takashi Iwaiae6b8132006-03-03 16:47:17 +01002235}
2236
2237static void alc880_lg_unsol_event(struct hda_codec *codec, unsigned int res)
2238{
2239 /* Looks like the unsol event is incompatible with the standard
2240 * definition. 4bit tag is placed at 28 bit!
2241 */
2242 if ((res >> 28) == 0x01)
2243 alc880_lg_automute(codec);
2244}
2245
2246/*
Takashi Iwaid6815182006-03-23 16:06:23 +01002247 * LG LW20
2248 *
2249 * Pin assignment:
2250 * Speaker-out: 0x14
2251 * Mic-In: 0x18
Claudio Matsuokae4f41da2007-07-13 11:51:06 +02002252 * Built-in Mic-In: 0x19
2253 * Line-In: 0x1b
2254 * HP-Out: 0x1a
Takashi Iwaid6815182006-03-23 16:06:23 +01002255 * SPDIF-Out: 0x1e
2256 */
2257
Takashi Iwaid6815182006-03-23 16:06:23 +01002258static struct hda_input_mux alc880_lg_lw_capture_source = {
Claudio Matsuokae4f41da2007-07-13 11:51:06 +02002259 .num_items = 3,
Takashi Iwaid6815182006-03-23 16:06:23 +01002260 .items = {
2261 { "Mic", 0x0 },
2262 { "Internal Mic", 0x1 },
Claudio Matsuokae4f41da2007-07-13 11:51:06 +02002263 { "Line In", 0x2 },
Takashi Iwaid6815182006-03-23 16:06:23 +01002264 },
2265};
2266
Claudio Matsuoka0a8c5da2007-07-04 15:17:38 +02002267#define alc880_lg_lw_modes alc880_threestack_modes
2268
Takashi Iwaid6815182006-03-23 16:06:23 +01002269static struct snd_kcontrol_new alc880_lg_lw_mixer[] = {
Claudio Matsuoka0a8c5da2007-07-04 15:17:38 +02002270 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
2271 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
2272 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0f, 0x0, HDA_OUTPUT),
2273 HDA_BIND_MUTE("Surround Playback Switch", 0x0f, 2, HDA_INPUT),
2274 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
2275 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
2276 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
2277 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
2278 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
2279 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
Takashi Iwaid6815182006-03-23 16:06:23 +01002280 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
2281 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
2282 HDA_CODEC_VOLUME("Internal Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
2283 HDA_CODEC_MUTE("Internal Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
Claudio Matsuoka0a8c5da2007-07-04 15:17:38 +02002284 {
2285 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2286 .name = "Channel Mode",
2287 .info = alc_ch_mode_info,
2288 .get = alc_ch_mode_get,
2289 .put = alc_ch_mode_put,
2290 },
Takashi Iwaid6815182006-03-23 16:06:23 +01002291 { } /* end */
2292};
2293
2294static struct hda_verb alc880_lg_lw_init_verbs[] = {
Claudio Matsuoka0a8c5da2007-07-04 15:17:38 +02002295 {0x13, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
2296 {0x10, AC_VERB_SET_CONNECT_SEL, 0x02}, /* mic/clfe */
2297 {0x12, AC_VERB_SET_CONNECT_SEL, 0x03}, /* line/surround */
2298
Takashi Iwaid6815182006-03-23 16:06:23 +01002299 /* set capture source to mic-in */
2300 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
2301 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
2302 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Takashi Iwaicb53c622007-08-10 17:21:45 +02002303 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(7)},
Takashi Iwaid6815182006-03-23 16:06:23 +01002304 /* speaker-out */
2305 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2306 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2307 /* HP-out */
Takashi Iwaid6815182006-03-23 16:06:23 +01002308 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2309 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2310 /* mic-in to input */
2311 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2312 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2313 /* built-in mic */
2314 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2315 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2316 /* jack sense */
2317 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | 0x1},
2318 { }
2319};
2320
2321/* toggle speaker-output according to the hp-jack state */
2322static void alc880_lg_lw_automute(struct hda_codec *codec)
2323{
2324 unsigned int present;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02002325 unsigned char bits;
Takashi Iwaid6815182006-03-23 16:06:23 +01002326
2327 present = snd_hda_codec_read(codec, 0x1b, 0,
2328 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +02002329 bits = present ? HDA_AMP_MUTE : 0;
2330 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
2331 HDA_AMP_MUTE, bits);
Takashi Iwaid6815182006-03-23 16:06:23 +01002332}
2333
2334static void alc880_lg_lw_unsol_event(struct hda_codec *codec, unsigned int res)
2335{
2336 /* Looks like the unsol event is incompatible with the standard
2337 * definition. 4bit tag is placed at 28 bit!
2338 */
2339 if ((res >> 28) == 0x01)
2340 alc880_lg_lw_automute(codec);
2341}
2342
Takashi Iwaidf99cd32008-04-25 15:25:04 +02002343static struct snd_kcontrol_new alc880_medion_rim_mixer[] = {
2344 HDA_CODEC_VOLUME("Master Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
2345 HDA_BIND_MUTE("Master Playback Switch", 0x0c, 2, HDA_INPUT),
2346 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
2347 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
2348 HDA_CODEC_VOLUME("Internal Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
2349 HDA_CODEC_MUTE("Internal Playback Switch", 0x0b, 0x1, HDA_INPUT),
2350 { } /* end */
2351};
2352
2353static struct hda_input_mux alc880_medion_rim_capture_source = {
2354 .num_items = 2,
2355 .items = {
2356 { "Mic", 0x0 },
2357 { "Internal Mic", 0x1 },
2358 },
2359};
2360
2361static struct hda_verb alc880_medion_rim_init_verbs[] = {
2362 {0x13, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
2363
2364 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2365 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2366
2367 /* Mic1 (rear panel) pin widget for input and vref at 80% */
2368 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2369 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2370 /* Mic2 (as headphone out) for HP output */
2371 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2372 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2373 /* Internal Speaker */
2374 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2375 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2376
2377 {0x20, AC_VERB_SET_COEF_INDEX, 0x07},
2378 {0x20, AC_VERB_SET_PROC_COEF, 0x3060},
2379
2380 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
2381 { }
2382};
2383
2384/* toggle speaker-output according to the hp-jack state */
2385static void alc880_medion_rim_automute(struct hda_codec *codec)
2386{
2387 unsigned int present;
2388 unsigned char bits;
2389
2390 present = snd_hda_codec_read(codec, 0x14, 0,
2391 AC_VERB_GET_PIN_SENSE, 0)
2392 & AC_PINSENSE_PRESENCE;
2393 bits = present ? HDA_AMP_MUTE : 0;
2394 snd_hda_codec_amp_stereo(codec, 0x1b, HDA_OUTPUT, 0,
2395 HDA_AMP_MUTE, bits);
2396 if (present)
2397 snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DATA, 0);
2398 else
2399 snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DATA, 2);
2400}
2401
2402static void alc880_medion_rim_unsol_event(struct hda_codec *codec,
2403 unsigned int res)
2404{
2405 /* Looks like the unsol event is incompatible with the standard
2406 * definition. 4bit tag is placed at 28 bit!
2407 */
2408 if ((res >> 28) == ALC880_HP_EVENT)
2409 alc880_medion_rim_automute(codec);
2410}
2411
Takashi Iwaicb53c622007-08-10 17:21:45 +02002412#ifdef CONFIG_SND_HDA_POWER_SAVE
2413static struct hda_amp_list alc880_loopbacks[] = {
2414 { 0x0b, HDA_INPUT, 0 },
2415 { 0x0b, HDA_INPUT, 1 },
2416 { 0x0b, HDA_INPUT, 2 },
2417 { 0x0b, HDA_INPUT, 3 },
2418 { 0x0b, HDA_INPUT, 4 },
2419 { } /* end */
2420};
2421
2422static struct hda_amp_list alc880_lg_loopbacks[] = {
2423 { 0x0b, HDA_INPUT, 1 },
2424 { 0x0b, HDA_INPUT, 6 },
2425 { 0x0b, HDA_INPUT, 7 },
2426 { } /* end */
2427};
2428#endif
2429
Takashi Iwaid6815182006-03-23 16:06:23 +01002430/*
Takashi Iwaiae6b8132006-03-03 16:47:17 +01002431 * Common callbacks
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002432 */
Takashi Iwai16ded522005-06-10 19:58:24 +02002433
Linus Torvalds1da177e2005-04-16 15:20:36 -07002434static int alc_init(struct hda_codec *codec)
2435{
2436 struct alc_spec *spec = codec->spec;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002437 unsigned int i;
2438
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +02002439 alc_fix_pll(codec);
2440
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002441 for (i = 0; i < spec->num_init_verbs; i++)
2442 snd_hda_sequence_write(codec, spec->init_verbs[i]);
Takashi Iwaiae6b8132006-03-03 16:47:17 +01002443
2444 if (spec->init_hook)
2445 spec->init_hook(codec);
2446
Linus Torvalds1da177e2005-04-16 15:20:36 -07002447 return 0;
2448}
2449
Takashi Iwaiae6b8132006-03-03 16:47:17 +01002450static void alc_unsol_event(struct hda_codec *codec, unsigned int res)
2451{
2452 struct alc_spec *spec = codec->spec;
2453
2454 if (spec->unsol_event)
2455 spec->unsol_event(codec, res);
2456}
2457
Takashi Iwaicb53c622007-08-10 17:21:45 +02002458#ifdef CONFIG_SND_HDA_POWER_SAVE
2459static int alc_check_power_status(struct hda_codec *codec, hda_nid_t nid)
2460{
2461 struct alc_spec *spec = codec->spec;
2462 return snd_hda_check_amp_list_power(codec, &spec->loopback, nid);
2463}
2464#endif
2465
Linus Torvalds1da177e2005-04-16 15:20:36 -07002466/*
2467 * Analog playback callbacks
2468 */
2469static int alc880_playback_pcm_open(struct hda_pcm_stream *hinfo,
2470 struct hda_codec *codec,
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01002471 struct snd_pcm_substream *substream)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002472{
2473 struct alc_spec *spec = codec->spec;
Takashi Iwai9a081602008-02-12 18:37:26 +01002474 return snd_hda_multi_out_analog_open(codec, &spec->multiout, substream,
2475 hinfo);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002476}
2477
2478static int alc880_playback_pcm_prepare(struct hda_pcm_stream *hinfo,
2479 struct hda_codec *codec,
2480 unsigned int stream_tag,
2481 unsigned int format,
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01002482 struct snd_pcm_substream *substream)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002483{
2484 struct alc_spec *spec = codec->spec;
Takashi Iwai9c7f8522006-06-28 15:08:22 +02002485 return snd_hda_multi_out_analog_prepare(codec, &spec->multiout,
2486 stream_tag, format, substream);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002487}
2488
2489static int alc880_playback_pcm_cleanup(struct hda_pcm_stream *hinfo,
2490 struct hda_codec *codec,
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01002491 struct snd_pcm_substream *substream)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002492{
2493 struct alc_spec *spec = codec->spec;
2494 return snd_hda_multi_out_analog_cleanup(codec, &spec->multiout);
2495}
2496
2497/*
2498 * Digital out
2499 */
2500static int alc880_dig_playback_pcm_open(struct hda_pcm_stream *hinfo,
2501 struct hda_codec *codec,
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01002502 struct snd_pcm_substream *substream)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002503{
2504 struct alc_spec *spec = codec->spec;
2505 return snd_hda_multi_out_dig_open(codec, &spec->multiout);
2506}
2507
Takashi Iwai6b97eb42007-04-05 14:51:48 +02002508static int alc880_dig_playback_pcm_prepare(struct hda_pcm_stream *hinfo,
2509 struct hda_codec *codec,
2510 unsigned int stream_tag,
2511 unsigned int format,
2512 struct snd_pcm_substream *substream)
2513{
2514 struct alc_spec *spec = codec->spec;
2515 return snd_hda_multi_out_dig_prepare(codec, &spec->multiout,
2516 stream_tag, format, substream);
2517}
2518
Linus Torvalds1da177e2005-04-16 15:20:36 -07002519static int alc880_dig_playback_pcm_close(struct hda_pcm_stream *hinfo,
2520 struct hda_codec *codec,
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01002521 struct snd_pcm_substream *substream)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002522{
2523 struct alc_spec *spec = codec->spec;
2524 return snd_hda_multi_out_dig_close(codec, &spec->multiout);
2525}
2526
2527/*
2528 * Analog capture
2529 */
Takashi Iwai63300792008-01-24 15:31:36 +01002530static int alc880_alt_capture_pcm_prepare(struct hda_pcm_stream *hinfo,
Linus Torvalds1da177e2005-04-16 15:20:36 -07002531 struct hda_codec *codec,
2532 unsigned int stream_tag,
2533 unsigned int format,
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01002534 struct snd_pcm_substream *substream)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002535{
2536 struct alc_spec *spec = codec->spec;
2537
Takashi Iwai63300792008-01-24 15:31:36 +01002538 snd_hda_codec_setup_stream(codec, spec->adc_nids[substream->number + 1],
Linus Torvalds1da177e2005-04-16 15:20:36 -07002539 stream_tag, 0, format);
2540 return 0;
2541}
2542
Takashi Iwai63300792008-01-24 15:31:36 +01002543static int alc880_alt_capture_pcm_cleanup(struct hda_pcm_stream *hinfo,
Linus Torvalds1da177e2005-04-16 15:20:36 -07002544 struct hda_codec *codec,
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01002545 struct snd_pcm_substream *substream)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002546{
2547 struct alc_spec *spec = codec->spec;
2548
Takashi Iwai888afa12008-03-18 09:57:50 +01002549 snd_hda_codec_cleanup_stream(codec,
2550 spec->adc_nids[substream->number + 1]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002551 return 0;
2552}
2553
2554
2555/*
2556 */
2557static struct hda_pcm_stream alc880_pcm_analog_playback = {
2558 .substreams = 1,
2559 .channels_min = 2,
2560 .channels_max = 8,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002561 /* NID is set in alc_build_pcms */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002562 .ops = {
2563 .open = alc880_playback_pcm_open,
2564 .prepare = alc880_playback_pcm_prepare,
2565 .cleanup = alc880_playback_pcm_cleanup
2566 },
2567};
2568
2569static struct hda_pcm_stream alc880_pcm_analog_capture = {
Takashi Iwai63300792008-01-24 15:31:36 +01002570 .substreams = 1,
2571 .channels_min = 2,
2572 .channels_max = 2,
2573 /* NID is set in alc_build_pcms */
2574};
2575
2576static struct hda_pcm_stream alc880_pcm_analog_alt_playback = {
2577 .substreams = 1,
2578 .channels_min = 2,
2579 .channels_max = 2,
2580 /* NID is set in alc_build_pcms */
2581};
2582
2583static struct hda_pcm_stream alc880_pcm_analog_alt_capture = {
2584 .substreams = 2, /* can be overridden */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002585 .channels_min = 2,
2586 .channels_max = 2,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002587 /* NID is set in alc_build_pcms */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002588 .ops = {
Takashi Iwai63300792008-01-24 15:31:36 +01002589 .prepare = alc880_alt_capture_pcm_prepare,
2590 .cleanup = alc880_alt_capture_pcm_cleanup
Linus Torvalds1da177e2005-04-16 15:20:36 -07002591 },
2592};
2593
2594static struct hda_pcm_stream alc880_pcm_digital_playback = {
2595 .substreams = 1,
2596 .channels_min = 2,
2597 .channels_max = 2,
2598 /* NID is set in alc_build_pcms */
2599 .ops = {
2600 .open = alc880_dig_playback_pcm_open,
Takashi Iwai6b97eb42007-04-05 14:51:48 +02002601 .close = alc880_dig_playback_pcm_close,
2602 .prepare = alc880_dig_playback_pcm_prepare
Linus Torvalds1da177e2005-04-16 15:20:36 -07002603 },
2604};
2605
2606static struct hda_pcm_stream alc880_pcm_digital_capture = {
2607 .substreams = 1,
2608 .channels_min = 2,
2609 .channels_max = 2,
2610 /* NID is set in alc_build_pcms */
2611};
2612
Jonathan Woithe4c5186e2006-02-09 11:53:48 +01002613/* Used by alc_build_pcms to flag that a PCM has no playback stream */
Takashi Iwai63300792008-01-24 15:31:36 +01002614static struct hda_pcm_stream alc_pcm_null_stream = {
Jonathan Woithe4c5186e2006-02-09 11:53:48 +01002615 .substreams = 0,
2616 .channels_min = 0,
2617 .channels_max = 0,
2618};
2619
Linus Torvalds1da177e2005-04-16 15:20:36 -07002620static int alc_build_pcms(struct hda_codec *codec)
2621{
2622 struct alc_spec *spec = codec->spec;
2623 struct hda_pcm *info = spec->pcm_rec;
2624 int i;
2625
2626 codec->num_pcms = 1;
2627 codec->pcm_info = info;
2628
2629 info->name = spec->stream_name_analog;
Takashi Iwai4a471b72005-12-07 13:56:29 +01002630 if (spec->stream_analog_playback) {
2631 snd_assert(spec->multiout.dac_nids, return -EINVAL);
2632 info->stream[SNDRV_PCM_STREAM_PLAYBACK] = *(spec->stream_analog_playback);
2633 info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid = spec->multiout.dac_nids[0];
2634 }
2635 if (spec->stream_analog_capture) {
2636 snd_assert(spec->adc_nids, return -EINVAL);
2637 info->stream[SNDRV_PCM_STREAM_CAPTURE] = *(spec->stream_analog_capture);
2638 info->stream[SNDRV_PCM_STREAM_CAPTURE].nid = spec->adc_nids[0];
2639 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002640
Takashi Iwai4a471b72005-12-07 13:56:29 +01002641 if (spec->channel_mode) {
2642 info->stream[SNDRV_PCM_STREAM_PLAYBACK].channels_max = 0;
2643 for (i = 0; i < spec->num_channel_mode; i++) {
2644 if (spec->channel_mode[i].channels > info->stream[SNDRV_PCM_STREAM_PLAYBACK].channels_max) {
2645 info->stream[SNDRV_PCM_STREAM_PLAYBACK].channels_max = spec->channel_mode[i].channels;
2646 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002647 }
2648 }
2649
Takashi Iwaie08a0072006-09-07 17:52:14 +02002650 /* SPDIF for stream index #1 */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002651 if (spec->multiout.dig_out_nid || spec->dig_in_nid) {
Takashi Iwaie08a0072006-09-07 17:52:14 +02002652 codec->num_pcms = 2;
Takashi Iwaic06134d2006-10-11 18:49:13 +02002653 info = spec->pcm_rec + 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002654 info->name = spec->stream_name_digital;
Takashi Iwai7ba72ba2008-02-06 14:03:20 +01002655 info->pcm_type = HDA_PCM_TYPE_SPDIF;
Takashi Iwai4a471b72005-12-07 13:56:29 +01002656 if (spec->multiout.dig_out_nid &&
2657 spec->stream_digital_playback) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002658 info->stream[SNDRV_PCM_STREAM_PLAYBACK] = *(spec->stream_digital_playback);
2659 info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid = spec->multiout.dig_out_nid;
2660 }
Takashi Iwai4a471b72005-12-07 13:56:29 +01002661 if (spec->dig_in_nid &&
2662 spec->stream_digital_capture) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002663 info->stream[SNDRV_PCM_STREAM_CAPTURE] = *(spec->stream_digital_capture);
2664 info->stream[SNDRV_PCM_STREAM_CAPTURE].nid = spec->dig_in_nid;
2665 }
2666 }
2667
Takashi Iwaie08a0072006-09-07 17:52:14 +02002668 /* If the use of more than one ADC is requested for the current
2669 * model, configure a second analog capture-only PCM.
2670 */
2671 /* Additional Analaog capture for index #2 */
Takashi Iwai63300792008-01-24 15:31:36 +01002672 if ((spec->alt_dac_nid && spec->stream_analog_alt_playback) ||
2673 (spec->num_adc_nids > 1 && spec->stream_analog_alt_capture)) {
Takashi Iwaie08a0072006-09-07 17:52:14 +02002674 codec->num_pcms = 3;
Takashi Iwaic06134d2006-10-11 18:49:13 +02002675 info = spec->pcm_rec + 2;
Takashi Iwaie08a0072006-09-07 17:52:14 +02002676 info->name = spec->stream_name_analog;
Takashi Iwai63300792008-01-24 15:31:36 +01002677 if (spec->alt_dac_nid) {
2678 info->stream[SNDRV_PCM_STREAM_PLAYBACK] =
2679 *spec->stream_analog_alt_playback;
2680 info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid =
2681 spec->alt_dac_nid;
2682 } else {
2683 info->stream[SNDRV_PCM_STREAM_PLAYBACK] =
2684 alc_pcm_null_stream;
2685 info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid = 0;
2686 }
2687 if (spec->num_adc_nids > 1) {
2688 info->stream[SNDRV_PCM_STREAM_CAPTURE] =
2689 *spec->stream_analog_alt_capture;
2690 info->stream[SNDRV_PCM_STREAM_CAPTURE].nid =
2691 spec->adc_nids[1];
2692 info->stream[SNDRV_PCM_STREAM_CAPTURE].substreams =
2693 spec->num_adc_nids - 1;
2694 } else {
2695 info->stream[SNDRV_PCM_STREAM_CAPTURE] =
2696 alc_pcm_null_stream;
2697 info->stream[SNDRV_PCM_STREAM_CAPTURE].nid = 0;
Takashi Iwaie08a0072006-09-07 17:52:14 +02002698 }
2699 }
2700
Linus Torvalds1da177e2005-04-16 15:20:36 -07002701 return 0;
2702}
2703
2704static void alc_free(struct hda_codec *codec)
2705{
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002706 struct alc_spec *spec = codec->spec;
2707 unsigned int i;
2708
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02002709 if (!spec)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002710 return;
2711
2712 if (spec->kctl_alloc) {
2713 for (i = 0; i < spec->num_kctl_used; i++)
2714 kfree(spec->kctl_alloc[i].name);
2715 kfree(spec->kctl_alloc);
2716 }
2717 kfree(spec);
Takashi Iwai7943a8a2008-04-16 17:29:09 +02002718 codec->spec = NULL; /* to be sure */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002719}
2720
2721/*
2722 */
2723static struct hda_codec_ops alc_patch_ops = {
2724 .build_controls = alc_build_controls,
2725 .build_pcms = alc_build_pcms,
2726 .init = alc_init,
2727 .free = alc_free,
Takashi Iwaiae6b8132006-03-03 16:47:17 +01002728 .unsol_event = alc_unsol_event,
Takashi Iwaicb53c622007-08-10 17:21:45 +02002729#ifdef CONFIG_SND_HDA_POWER_SAVE
2730 .check_power_status = alc_check_power_status,
2731#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07002732};
2733
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002734
2735/*
2736 * Test configuration for debugging
2737 *
2738 * Almost all inputs/outputs are enabled. I/O pins can be configured via
2739 * enum controls.
2740 */
2741#ifdef CONFIG_SND_DEBUG
2742static hda_nid_t alc880_test_dac_nids[4] = {
2743 0x02, 0x03, 0x04, 0x05
2744};
2745
2746static struct hda_input_mux alc880_test_capture_source = {
Takashi Iwaiae6b8132006-03-03 16:47:17 +01002747 .num_items = 7,
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002748 .items = {
2749 { "In-1", 0x0 },
2750 { "In-2", 0x1 },
2751 { "In-3", 0x2 },
2752 { "In-4", 0x3 },
2753 { "CD", 0x4 },
Takashi Iwaiae6b8132006-03-03 16:47:17 +01002754 { "Front", 0x5 },
2755 { "Surround", 0x6 },
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002756 },
2757};
2758
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +01002759static struct hda_channel_mode alc880_test_modes[4] = {
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002760 { 2, NULL },
Takashi Iwaifd2c3262005-05-13 17:18:42 +02002761 { 4, NULL },
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002762 { 6, NULL },
Takashi Iwaifd2c3262005-05-13 17:18:42 +02002763 { 8, NULL },
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002764};
2765
Takashi Iwai9c7f8522006-06-28 15:08:22 +02002766static int alc_test_pin_ctl_info(struct snd_kcontrol *kcontrol,
2767 struct snd_ctl_elem_info *uinfo)
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002768{
2769 static char *texts[] = {
2770 "N/A", "Line Out", "HP Out",
2771 "In Hi-Z", "In 50%", "In Grd", "In 80%", "In 100%"
2772 };
2773 uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
2774 uinfo->count = 1;
2775 uinfo->value.enumerated.items = 8;
2776 if (uinfo->value.enumerated.item >= 8)
2777 uinfo->value.enumerated.item = 7;
2778 strcpy(uinfo->value.enumerated.name, texts[uinfo->value.enumerated.item]);
2779 return 0;
2780}
2781
Takashi Iwai9c7f8522006-06-28 15:08:22 +02002782static int alc_test_pin_ctl_get(struct snd_kcontrol *kcontrol,
2783 struct snd_ctl_elem_value *ucontrol)
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002784{
2785 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2786 hda_nid_t nid = (hda_nid_t)kcontrol->private_value;
2787 unsigned int pin_ctl, item = 0;
2788
2789 pin_ctl = snd_hda_codec_read(codec, nid, 0,
2790 AC_VERB_GET_PIN_WIDGET_CONTROL, 0);
2791 if (pin_ctl & AC_PINCTL_OUT_EN) {
2792 if (pin_ctl & AC_PINCTL_HP_EN)
2793 item = 2;
2794 else
2795 item = 1;
2796 } else if (pin_ctl & AC_PINCTL_IN_EN) {
2797 switch (pin_ctl & AC_PINCTL_VREFEN) {
2798 case AC_PINCTL_VREF_HIZ: item = 3; break;
2799 case AC_PINCTL_VREF_50: item = 4; break;
2800 case AC_PINCTL_VREF_GRD: item = 5; break;
2801 case AC_PINCTL_VREF_80: item = 6; break;
2802 case AC_PINCTL_VREF_100: item = 7; break;
2803 }
2804 }
2805 ucontrol->value.enumerated.item[0] = item;
2806 return 0;
2807}
2808
Takashi Iwai9c7f8522006-06-28 15:08:22 +02002809static int alc_test_pin_ctl_put(struct snd_kcontrol *kcontrol,
2810 struct snd_ctl_elem_value *ucontrol)
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002811{
2812 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2813 hda_nid_t nid = (hda_nid_t)kcontrol->private_value;
2814 static unsigned int ctls[] = {
2815 0, AC_PINCTL_OUT_EN, AC_PINCTL_OUT_EN | AC_PINCTL_HP_EN,
2816 AC_PINCTL_IN_EN | AC_PINCTL_VREF_HIZ,
2817 AC_PINCTL_IN_EN | AC_PINCTL_VREF_50,
2818 AC_PINCTL_IN_EN | AC_PINCTL_VREF_GRD,
2819 AC_PINCTL_IN_EN | AC_PINCTL_VREF_80,
2820 AC_PINCTL_IN_EN | AC_PINCTL_VREF_100,
2821 };
2822 unsigned int old_ctl, new_ctl;
2823
2824 old_ctl = snd_hda_codec_read(codec, nid, 0,
2825 AC_VERB_GET_PIN_WIDGET_CONTROL, 0);
2826 new_ctl = ctls[ucontrol->value.enumerated.item[0]];
2827 if (old_ctl != new_ctl) {
Takashi Iwai82beb8f2007-08-10 17:09:26 +02002828 int val;
2829 snd_hda_codec_write_cache(codec, nid, 0,
2830 AC_VERB_SET_PIN_WIDGET_CONTROL,
2831 new_ctl);
Takashi Iwai47fd8302007-08-10 17:11:07 +02002832 val = ucontrol->value.enumerated.item[0] >= 3 ?
2833 HDA_AMP_MUTE : 0;
2834 snd_hda_codec_amp_stereo(codec, nid, HDA_OUTPUT, 0,
2835 HDA_AMP_MUTE, val);
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002836 return 1;
2837 }
2838 return 0;
2839}
2840
Takashi Iwai9c7f8522006-06-28 15:08:22 +02002841static int alc_test_pin_src_info(struct snd_kcontrol *kcontrol,
2842 struct snd_ctl_elem_info *uinfo)
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002843{
2844 static char *texts[] = {
2845 "Front", "Surround", "CLFE", "Side"
2846 };
2847 uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
2848 uinfo->count = 1;
2849 uinfo->value.enumerated.items = 4;
2850 if (uinfo->value.enumerated.item >= 4)
2851 uinfo->value.enumerated.item = 3;
2852 strcpy(uinfo->value.enumerated.name, texts[uinfo->value.enumerated.item]);
2853 return 0;
2854}
2855
Takashi Iwai9c7f8522006-06-28 15:08:22 +02002856static int alc_test_pin_src_get(struct snd_kcontrol *kcontrol,
2857 struct snd_ctl_elem_value *ucontrol)
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002858{
2859 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2860 hda_nid_t nid = (hda_nid_t)kcontrol->private_value;
2861 unsigned int sel;
2862
2863 sel = snd_hda_codec_read(codec, nid, 0, AC_VERB_GET_CONNECT_SEL, 0);
2864 ucontrol->value.enumerated.item[0] = sel & 3;
2865 return 0;
2866}
2867
Takashi Iwai9c7f8522006-06-28 15:08:22 +02002868static int alc_test_pin_src_put(struct snd_kcontrol *kcontrol,
2869 struct snd_ctl_elem_value *ucontrol)
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002870{
2871 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2872 hda_nid_t nid = (hda_nid_t)kcontrol->private_value;
2873 unsigned int sel;
2874
2875 sel = snd_hda_codec_read(codec, nid, 0, AC_VERB_GET_CONNECT_SEL, 0) & 3;
2876 if (ucontrol->value.enumerated.item[0] != sel) {
2877 sel = ucontrol->value.enumerated.item[0] & 3;
Takashi Iwai82beb8f2007-08-10 17:09:26 +02002878 snd_hda_codec_write_cache(codec, nid, 0,
2879 AC_VERB_SET_CONNECT_SEL, sel);
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002880 return 1;
2881 }
2882 return 0;
2883}
2884
2885#define PIN_CTL_TEST(xname,nid) { \
2886 .iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
2887 .name = xname, \
2888 .info = alc_test_pin_ctl_info, \
2889 .get = alc_test_pin_ctl_get, \
2890 .put = alc_test_pin_ctl_put, \
2891 .private_value = nid \
2892 }
2893
2894#define PIN_SRC_TEST(xname,nid) { \
2895 .iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
2896 .name = xname, \
2897 .info = alc_test_pin_src_info, \
2898 .get = alc_test_pin_src_get, \
2899 .put = alc_test_pin_src_put, \
2900 .private_value = nid \
2901 }
2902
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01002903static struct snd_kcontrol_new alc880_test_mixer[] = {
Takashi Iwai05acb862005-06-10 19:50:25 +02002904 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
2905 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
2906 HDA_CODEC_VOLUME("CLFE Playback Volume", 0x0e, 0x0, HDA_OUTPUT),
2907 HDA_CODEC_VOLUME("Side Playback Volume", 0x0f, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01002908 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
2909 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
2910 HDA_BIND_MUTE("CLFE Playback Switch", 0x0e, 2, HDA_INPUT),
2911 HDA_BIND_MUTE("Side Playback Switch", 0x0f, 2, HDA_INPUT),
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002912 PIN_CTL_TEST("Front Pin Mode", 0x14),
2913 PIN_CTL_TEST("Surround Pin Mode", 0x15),
2914 PIN_CTL_TEST("CLFE Pin Mode", 0x16),
2915 PIN_CTL_TEST("Side Pin Mode", 0x17),
2916 PIN_CTL_TEST("In-1 Pin Mode", 0x18),
2917 PIN_CTL_TEST("In-2 Pin Mode", 0x19),
2918 PIN_CTL_TEST("In-3 Pin Mode", 0x1a),
2919 PIN_CTL_TEST("In-4 Pin Mode", 0x1b),
2920 PIN_SRC_TEST("In-1 Pin Source", 0x18),
2921 PIN_SRC_TEST("In-2 Pin Source", 0x19),
2922 PIN_SRC_TEST("In-3 Pin Source", 0x1a),
2923 PIN_SRC_TEST("In-4 Pin Source", 0x1b),
2924 HDA_CODEC_VOLUME("In-1 Playback Volume", 0x0b, 0x0, HDA_INPUT),
2925 HDA_CODEC_MUTE("In-1 Playback Switch", 0x0b, 0x0, HDA_INPUT),
2926 HDA_CODEC_VOLUME("In-2 Playback Volume", 0x0b, 0x1, HDA_INPUT),
2927 HDA_CODEC_MUTE("In-2 Playback Switch", 0x0b, 0x1, HDA_INPUT),
2928 HDA_CODEC_VOLUME("In-3 Playback Volume", 0x0b, 0x2, HDA_INPUT),
2929 HDA_CODEC_MUTE("In-3 Playback Switch", 0x0b, 0x2, HDA_INPUT),
2930 HDA_CODEC_VOLUME("In-4 Playback Volume", 0x0b, 0x3, HDA_INPUT),
2931 HDA_CODEC_MUTE("In-4 Playback Switch", 0x0b, 0x3, HDA_INPUT),
2932 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x4, HDA_INPUT),
2933 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x4, HDA_INPUT),
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002934 {
2935 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2936 .name = "Channel Mode",
Kailang Yangdf694da2005-12-05 19:42:22 +01002937 .info = alc_ch_mode_info,
2938 .get = alc_ch_mode_get,
2939 .put = alc_ch_mode_put,
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002940 },
2941 { } /* end */
2942};
2943
2944static struct hda_verb alc880_test_init_verbs[] = {
2945 /* Unmute inputs of 0x0c - 0x0f */
Takashi Iwai05acb862005-06-10 19:50:25 +02002946 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
2947 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
2948 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
2949 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
2950 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
2951 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
2952 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
2953 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002954 /* Vol output for 0x0c-0x0f */
Takashi Iwai05acb862005-06-10 19:50:25 +02002955 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
2956 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
2957 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
2958 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002959 /* Set output pins 0x14-0x17 */
Takashi Iwai05acb862005-06-10 19:50:25 +02002960 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2961 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2962 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2963 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002964 /* Unmute output pins 0x14-0x17 */
Takashi Iwai05acb862005-06-10 19:50:25 +02002965 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2966 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2967 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2968 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002969 /* Set input pins 0x18-0x1c */
Takashi Iwai16ded522005-06-10 19:58:24 +02002970 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2971 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Takashi Iwai05acb862005-06-10 19:50:25 +02002972 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2973 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2974 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002975 /* Mute input pins 0x18-0x1b */
Takashi Iwai05acb862005-06-10 19:50:25 +02002976 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2977 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2978 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2979 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
Takashi Iwai71fe7b82005-05-25 18:11:40 +02002980 /* ADC set up */
Takashi Iwai05acb862005-06-10 19:50:25 +02002981 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Takashi Iwai71fe7b82005-05-25 18:11:40 +02002982 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
Takashi Iwai05acb862005-06-10 19:50:25 +02002983 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Takashi Iwai71fe7b82005-05-25 18:11:40 +02002984 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
Takashi Iwai05acb862005-06-10 19:50:25 +02002985 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Takashi Iwai71fe7b82005-05-25 18:11:40 +02002986 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
Takashi Iwai05acb862005-06-10 19:50:25 +02002987 /* Analog input/passthru */
2988 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
2989 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
2990 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
2991 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
2992 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Takashi Iwai2fa522b2005-05-12 14:51:12 +02002993 { }
2994};
2995#endif
2996
Linus Torvalds1da177e2005-04-16 15:20:36 -07002997/*
2998 */
2999
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003000static const char *alc880_models[ALC880_MODEL_LAST] = {
3001 [ALC880_3ST] = "3stack",
3002 [ALC880_TCL_S700] = "tcl",
3003 [ALC880_3ST_DIG] = "3stack-digout",
3004 [ALC880_CLEVO] = "clevo",
3005 [ALC880_5ST] = "5stack",
3006 [ALC880_5ST_DIG] = "5stack-digout",
3007 [ALC880_W810] = "w810",
3008 [ALC880_Z71V] = "z71v",
3009 [ALC880_6ST] = "6stack",
3010 [ALC880_6ST_DIG] = "6stack-digout",
3011 [ALC880_ASUS] = "asus",
3012 [ALC880_ASUS_W1V] = "asus-w1v",
3013 [ALC880_ASUS_DIG] = "asus-dig",
3014 [ALC880_ASUS_DIG2] = "asus-dig2",
3015 [ALC880_UNIWILL_DIG] = "uniwill",
Tobin Davis2cf9f0f2007-02-07 16:04:25 +01003016 [ALC880_UNIWILL_P53] = "uniwill-p53",
3017 [ALC880_FUJITSU] = "fujitsu",
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003018 [ALC880_F1734] = "F1734",
3019 [ALC880_LG] = "lg",
3020 [ALC880_LG_LW] = "lg-lw",
Takashi Iwaidf99cd32008-04-25 15:25:04 +02003021 [ALC880_MEDION_RIM] = "medion",
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003022#ifdef CONFIG_SND_DEBUG
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003023 [ALC880_TEST] = "test",
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003024#endif
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003025 [ALC880_AUTO] = "auto",
3026};
3027
3028static struct snd_pci_quirk alc880_cfg_tbl[] = {
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003029 SND_PCI_QUIRK(0x1019, 0x0f69, "Coeus G610P", ALC880_W810),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003030 SND_PCI_QUIRK(0x1019, 0xa880, "ECS", ALC880_5ST_DIG),
3031 SND_PCI_QUIRK(0x1019, 0xa884, "Acer APFV", ALC880_6ST),
3032 SND_PCI_QUIRK(0x1025, 0x0070, "ULI", ALC880_3ST_DIG),
3033 SND_PCI_QUIRK(0x1025, 0x0077, "ULI", ALC880_6ST_DIG),
3034 SND_PCI_QUIRK(0x1025, 0x0078, "ULI", ALC880_6ST_DIG),
3035 SND_PCI_QUIRK(0x1025, 0x0087, "ULI", ALC880_6ST_DIG),
3036 SND_PCI_QUIRK(0x1025, 0xe309, "ULI", ALC880_3ST_DIG),
3037 SND_PCI_QUIRK(0x1025, 0xe310, "ULI", ALC880_3ST),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003038 SND_PCI_QUIRK(0x1039, 0x1234, NULL, ALC880_6ST_DIG),
3039 SND_PCI_QUIRK(0x103c, 0x2a09, "HP", ALC880_5ST),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003040 SND_PCI_QUIRK(0x1043, 0x10b3, "ASUS W1V", ALC880_ASUS_W1V),
3041 SND_PCI_QUIRK(0x1043, 0x10c2, "ASUS W6A", ALC880_ASUS_DIG),
3042 SND_PCI_QUIRK(0x1043, 0x10c3, "ASUS Wxx", ALC880_ASUS_DIG),
3043 SND_PCI_QUIRK(0x1043, 0x1113, "ASUS", ALC880_ASUS_DIG),
3044 SND_PCI_QUIRK(0x1043, 0x1123, "ASUS", ALC880_ASUS_DIG),
3045 SND_PCI_QUIRK(0x1043, 0x1173, "ASUS", ALC880_ASUS_DIG),
3046 SND_PCI_QUIRK(0x1043, 0x1964, "ASUS Z71V", ALC880_Z71V),
3047 /* SND_PCI_QUIRK(0x1043, 0x1964, "ASUS", ALC880_ASUS_DIG), */
3048 SND_PCI_QUIRK(0x1043, 0x1973, "ASUS", ALC880_ASUS_DIG),
3049 SND_PCI_QUIRK(0x1043, 0x19b3, "ASUS", ALC880_ASUS_DIG),
Travis Place186c3112008-05-20 11:54:41 +02003050 SND_PCI_QUIRK(0x1043, 0x814e, "ASUS P5GD1 w/SPDIF", ALC880_6ST_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003051 SND_PCI_QUIRK(0x1043, 0x8181, "ASUS P4GPL", ALC880_ASUS_DIG),
3052 SND_PCI_QUIRK(0x1043, 0x8196, "ASUS P5GD1", ALC880_6ST),
3053 SND_PCI_QUIRK(0x1043, 0x81b4, "ASUS", ALC880_6ST),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003054 SND_PCI_QUIRK(0x1043, 0, "ASUS", ALC880_ASUS), /* default ASUS */
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003055 SND_PCI_QUIRK(0x104d, 0x81a0, "Sony", ALC880_3ST),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003056 SND_PCI_QUIRK(0x104d, 0x81d6, "Sony", ALC880_3ST),
3057 SND_PCI_QUIRK(0x107b, 0x3032, "Gateway", ALC880_5ST),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003058 SND_PCI_QUIRK(0x107b, 0x3033, "Gateway", ALC880_5ST),
3059 SND_PCI_QUIRK(0x107b, 0x4039, "Gateway", ALC880_5ST),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003060 SND_PCI_QUIRK(0x1297, 0xc790, "Shuttle ST20G5", ALC880_6ST_DIG),
3061 SND_PCI_QUIRK(0x1458, 0xa102, "Gigabyte K8", ALC880_6ST_DIG),
3062 SND_PCI_QUIRK(0x1462, 0x1150, "MSI", ALC880_6ST_DIG),
3063 SND_PCI_QUIRK(0x1509, 0x925d, "FIC P4M", ALC880_6ST_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003064 SND_PCI_QUIRK(0x1558, 0x0520, "Clevo m520G", ALC880_CLEVO),
3065 SND_PCI_QUIRK(0x1558, 0x0660, "Clevo m655n", ALC880_CLEVO),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003066 SND_PCI_QUIRK(0x1558, 0x5401, "ASUS", ALC880_ASUS_DIG2),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003067 SND_PCI_QUIRK(0x1565, 0x8202, "Biostar", ALC880_5ST_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003068 SND_PCI_QUIRK(0x1584, 0x9050, "Uniwill", ALC880_UNIWILL_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003069 SND_PCI_QUIRK(0x1584, 0x9054, "Uniwlll", ALC880_F1734),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003070 SND_PCI_QUIRK(0x1584, 0x9070, "Uniwill", ALC880_UNIWILL),
3071 SND_PCI_QUIRK(0x1584, 0x9077, "Uniwill P53", ALC880_UNIWILL_P53),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003072 SND_PCI_QUIRK(0x161f, 0x203d, "W810", ALC880_W810),
Takashi Iwaidf99cd32008-04-25 15:25:04 +02003073 SND_PCI_QUIRK(0x161f, 0x205d, "Medion Rim 2150", ALC880_MEDION_RIM),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003074 SND_PCI_QUIRK(0x1695, 0x400d, "EPoX", ALC880_5ST_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003075 SND_PCI_QUIRK(0x1695, 0x4012, "EPox EP-5LDA", ALC880_5ST_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003076 SND_PCI_QUIRK(0x1734, 0x107c, "FSC F1734", ALC880_F1734),
Travis Place1d116042008-06-23 11:42:30 +02003077 SND_PCI_QUIRK(0x1734, 0x1094, "FSC Amilo M1451G", ALC880_FUJITSU),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003078 SND_PCI_QUIRK(0x1734, 0x10ac, "FSC", ALC880_UNIWILL),
Tobin Davis2cf9f0f2007-02-07 16:04:25 +01003079 SND_PCI_QUIRK(0x1734, 0x10b0, "Fujitsu", ALC880_FUJITSU),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003080 SND_PCI_QUIRK(0x1854, 0x0018, "LG LW20", ALC880_LG_LW),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003081 SND_PCI_QUIRK(0x1854, 0x003b, "LG", ALC880_LG),
3082 SND_PCI_QUIRK(0x1854, 0x0068, "LG w1", ALC880_LG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003083 SND_PCI_QUIRK(0x1854, 0x0077, "LG LW25", ALC880_LG_LW),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003084 SND_PCI_QUIRK(0x19db, 0x4188, "TCL S700", ALC880_TCL_S700),
3085 SND_PCI_QUIRK(0x2668, 0x8086, NULL, ALC880_6ST_DIG), /* broken BIOS */
3086 SND_PCI_QUIRK(0x8086, 0x2668, NULL, ALC880_6ST_DIG),
3087 SND_PCI_QUIRK(0x8086, 0xa100, "Intel mobo", ALC880_5ST_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003088 SND_PCI_QUIRK(0x8086, 0xd400, "Intel mobo", ALC880_5ST_DIG),
3089 SND_PCI_QUIRK(0x8086, 0xd401, "Intel mobo", ALC880_5ST_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003090 SND_PCI_QUIRK(0x8086, 0xd402, "Intel mobo", ALC880_3ST_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003091 SND_PCI_QUIRK(0x8086, 0xe224, "Intel mobo", ALC880_5ST_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003092 SND_PCI_QUIRK(0x8086, 0xe305, "Intel mobo", ALC880_3ST_DIG),
3093 SND_PCI_QUIRK(0x8086, 0xe308, "Intel mobo", ALC880_3ST_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003094 SND_PCI_QUIRK(0x8086, 0xe400, "Intel mobo", ALC880_5ST_DIG),
3095 SND_PCI_QUIRK(0x8086, 0xe401, "Intel mobo", ALC880_5ST_DIG),
3096 SND_PCI_QUIRK(0x8086, 0xe402, "Intel mobo", ALC880_5ST_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003097 SND_PCI_QUIRK(0x8086, 0, "Intel mobo", ALC880_3ST), /* default Intel */
3098 SND_PCI_QUIRK(0xa0a0, 0x0560, "AOpen i915GMm-HFS", ALC880_5ST_DIG),
3099 SND_PCI_QUIRK(0xe803, 0x1019, NULL, ALC880_6ST_DIG),
Linus Torvalds1da177e2005-04-16 15:20:36 -07003100 {}
3101};
3102
Takashi Iwai16ded522005-06-10 19:58:24 +02003103/*
Kailang Yangdf694da2005-12-05 19:42:22 +01003104 * ALC880 codec presets
Takashi Iwai16ded522005-06-10 19:58:24 +02003105 */
Takashi Iwai16ded522005-06-10 19:58:24 +02003106static struct alc_config_preset alc880_presets[] = {
3107 [ALC880_3ST] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003108 .mixers = { alc880_three_stack_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003109 .init_verbs = { alc880_volume_init_verbs,
3110 alc880_pin_3stack_init_verbs },
Takashi Iwai16ded522005-06-10 19:58:24 +02003111 .num_dacs = ARRAY_SIZE(alc880_dac_nids),
Takashi Iwai16ded522005-06-10 19:58:24 +02003112 .dac_nids = alc880_dac_nids,
Takashi Iwai16ded522005-06-10 19:58:24 +02003113 .num_channel_mode = ARRAY_SIZE(alc880_threestack_modes),
3114 .channel_mode = alc880_threestack_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02003115 .need_dac_fix = 1,
Takashi Iwai16ded522005-06-10 19:58:24 +02003116 .input_mux = &alc880_capture_source,
3117 },
3118 [ALC880_3ST_DIG] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003119 .mixers = { alc880_three_stack_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003120 .init_verbs = { alc880_volume_init_verbs,
3121 alc880_pin_3stack_init_verbs },
Takashi Iwai16ded522005-06-10 19:58:24 +02003122 .num_dacs = ARRAY_SIZE(alc880_dac_nids),
Takashi Iwai16ded522005-06-10 19:58:24 +02003123 .dac_nids = alc880_dac_nids,
3124 .dig_out_nid = ALC880_DIGOUT_NID,
Takashi Iwai16ded522005-06-10 19:58:24 +02003125 .num_channel_mode = ARRAY_SIZE(alc880_threestack_modes),
3126 .channel_mode = alc880_threestack_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02003127 .need_dac_fix = 1,
Takashi Iwai16ded522005-06-10 19:58:24 +02003128 .input_mux = &alc880_capture_source,
3129 },
Kailang Yangdf694da2005-12-05 19:42:22 +01003130 [ALC880_TCL_S700] = {
3131 .mixers = { alc880_tcl_s700_mixer },
3132 .init_verbs = { alc880_volume_init_verbs,
3133 alc880_pin_tcl_S700_init_verbs,
3134 alc880_gpio2_init_verbs },
3135 .num_dacs = ARRAY_SIZE(alc880_dac_nids),
3136 .dac_nids = alc880_dac_nids,
3137 .hp_nid = 0x03,
3138 .num_channel_mode = ARRAY_SIZE(alc880_2_jack_modes),
3139 .channel_mode = alc880_2_jack_modes,
3140 .input_mux = &alc880_capture_source,
3141 },
Takashi Iwai16ded522005-06-10 19:58:24 +02003142 [ALC880_5ST] = {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003143 .mixers = { alc880_three_stack_mixer,
3144 alc880_five_stack_mixer},
3145 .init_verbs = { alc880_volume_init_verbs,
3146 alc880_pin_5stack_init_verbs },
Takashi Iwai16ded522005-06-10 19:58:24 +02003147 .num_dacs = ARRAY_SIZE(alc880_dac_nids),
3148 .dac_nids = alc880_dac_nids,
Takashi Iwai16ded522005-06-10 19:58:24 +02003149 .num_channel_mode = ARRAY_SIZE(alc880_fivestack_modes),
3150 .channel_mode = alc880_fivestack_modes,
3151 .input_mux = &alc880_capture_source,
3152 },
3153 [ALC880_5ST_DIG] = {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003154 .mixers = { alc880_three_stack_mixer,
3155 alc880_five_stack_mixer },
3156 .init_verbs = { alc880_volume_init_verbs,
3157 alc880_pin_5stack_init_verbs },
Takashi Iwai16ded522005-06-10 19:58:24 +02003158 .num_dacs = ARRAY_SIZE(alc880_dac_nids),
3159 .dac_nids = alc880_dac_nids,
3160 .dig_out_nid = ALC880_DIGOUT_NID,
Takashi Iwai16ded522005-06-10 19:58:24 +02003161 .num_channel_mode = ARRAY_SIZE(alc880_fivestack_modes),
3162 .channel_mode = alc880_fivestack_modes,
3163 .input_mux = &alc880_capture_source,
3164 },
Takashi Iwaib6482d42005-06-27 15:32:43 +02003165 [ALC880_6ST] = {
3166 .mixers = { alc880_six_stack_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003167 .init_verbs = { alc880_volume_init_verbs,
3168 alc880_pin_6stack_init_verbs },
Takashi Iwaib6482d42005-06-27 15:32:43 +02003169 .num_dacs = ARRAY_SIZE(alc880_6st_dac_nids),
3170 .dac_nids = alc880_6st_dac_nids,
3171 .num_channel_mode = ARRAY_SIZE(alc880_sixstack_modes),
3172 .channel_mode = alc880_sixstack_modes,
3173 .input_mux = &alc880_6stack_capture_source,
3174 },
Takashi Iwai16ded522005-06-10 19:58:24 +02003175 [ALC880_6ST_DIG] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003176 .mixers = { alc880_six_stack_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003177 .init_verbs = { alc880_volume_init_verbs,
3178 alc880_pin_6stack_init_verbs },
Takashi Iwai16ded522005-06-10 19:58:24 +02003179 .num_dacs = ARRAY_SIZE(alc880_6st_dac_nids),
3180 .dac_nids = alc880_6st_dac_nids,
3181 .dig_out_nid = ALC880_DIGOUT_NID,
Takashi Iwai16ded522005-06-10 19:58:24 +02003182 .num_channel_mode = ARRAY_SIZE(alc880_sixstack_modes),
3183 .channel_mode = alc880_sixstack_modes,
3184 .input_mux = &alc880_6stack_capture_source,
3185 },
3186 [ALC880_W810] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003187 .mixers = { alc880_w810_base_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003188 .init_verbs = { alc880_volume_init_verbs,
3189 alc880_pin_w810_init_verbs,
Takashi Iwaib0af0de2005-06-21 14:49:19 +02003190 alc880_gpio2_init_verbs },
Takashi Iwai16ded522005-06-10 19:58:24 +02003191 .num_dacs = ARRAY_SIZE(alc880_w810_dac_nids),
3192 .dac_nids = alc880_w810_dac_nids,
3193 .dig_out_nid = ALC880_DIGOUT_NID,
Takashi Iwai16ded522005-06-10 19:58:24 +02003194 .num_channel_mode = ARRAY_SIZE(alc880_w810_modes),
3195 .channel_mode = alc880_w810_modes,
3196 .input_mux = &alc880_capture_source,
3197 },
3198 [ALC880_Z71V] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003199 .mixers = { alc880_z71v_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003200 .init_verbs = { alc880_volume_init_verbs,
3201 alc880_pin_z71v_init_verbs },
Takashi Iwai16ded522005-06-10 19:58:24 +02003202 .num_dacs = ARRAY_SIZE(alc880_z71v_dac_nids),
3203 .dac_nids = alc880_z71v_dac_nids,
3204 .dig_out_nid = ALC880_DIGOUT_NID,
3205 .hp_nid = 0x03,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003206 .num_channel_mode = ARRAY_SIZE(alc880_2_jack_modes),
3207 .channel_mode = alc880_2_jack_modes,
Takashi Iwai16ded522005-06-10 19:58:24 +02003208 .input_mux = &alc880_capture_source,
3209 },
3210 [ALC880_F1734] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003211 .mixers = { alc880_f1734_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003212 .init_verbs = { alc880_volume_init_verbs,
3213 alc880_pin_f1734_init_verbs },
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003214 .num_dacs = ARRAY_SIZE(alc880_f1734_dac_nids),
3215 .dac_nids = alc880_f1734_dac_nids,
3216 .hp_nid = 0x02,
3217 .num_channel_mode = ARRAY_SIZE(alc880_2_jack_modes),
3218 .channel_mode = alc880_2_jack_modes,
Takashi Iwai937b4162008-02-11 14:52:36 +01003219 .input_mux = &alc880_f1734_capture_source,
3220 .unsol_event = alc880_uniwill_p53_unsol_event,
3221 .init_hook = alc880_uniwill_p53_hp_automute,
Takashi Iwai16ded522005-06-10 19:58:24 +02003222 },
3223 [ALC880_ASUS] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003224 .mixers = { alc880_asus_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003225 .init_verbs = { alc880_volume_init_verbs,
3226 alc880_pin_asus_init_verbs,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003227 alc880_gpio1_init_verbs },
3228 .num_dacs = ARRAY_SIZE(alc880_asus_dac_nids),
3229 .dac_nids = alc880_asus_dac_nids,
3230 .num_channel_mode = ARRAY_SIZE(alc880_asus_modes),
3231 .channel_mode = alc880_asus_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02003232 .need_dac_fix = 1,
Takashi Iwai16ded522005-06-10 19:58:24 +02003233 .input_mux = &alc880_capture_source,
3234 },
3235 [ALC880_ASUS_DIG] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003236 .mixers = { alc880_asus_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003237 .init_verbs = { alc880_volume_init_verbs,
3238 alc880_pin_asus_init_verbs,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003239 alc880_gpio1_init_verbs },
3240 .num_dacs = ARRAY_SIZE(alc880_asus_dac_nids),
3241 .dac_nids = alc880_asus_dac_nids,
Takashi Iwai16ded522005-06-10 19:58:24 +02003242 .dig_out_nid = ALC880_DIGOUT_NID,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003243 .num_channel_mode = ARRAY_SIZE(alc880_asus_modes),
3244 .channel_mode = alc880_asus_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02003245 .need_dac_fix = 1,
Takashi Iwai16ded522005-06-10 19:58:24 +02003246 .input_mux = &alc880_capture_source,
3247 },
Kailang Yangdf694da2005-12-05 19:42:22 +01003248 [ALC880_ASUS_DIG2] = {
3249 .mixers = { alc880_asus_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003250 .init_verbs = { alc880_volume_init_verbs,
3251 alc880_pin_asus_init_verbs,
Kailang Yangdf694da2005-12-05 19:42:22 +01003252 alc880_gpio2_init_verbs }, /* use GPIO2 */
3253 .num_dacs = ARRAY_SIZE(alc880_asus_dac_nids),
3254 .dac_nids = alc880_asus_dac_nids,
3255 .dig_out_nid = ALC880_DIGOUT_NID,
3256 .num_channel_mode = ARRAY_SIZE(alc880_asus_modes),
3257 .channel_mode = alc880_asus_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02003258 .need_dac_fix = 1,
Kailang Yangdf694da2005-12-05 19:42:22 +01003259 .input_mux = &alc880_capture_source,
3260 },
Takashi Iwai16ded522005-06-10 19:58:24 +02003261 [ALC880_ASUS_W1V] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003262 .mixers = { alc880_asus_mixer, alc880_asus_w1v_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003263 .init_verbs = { alc880_volume_init_verbs,
3264 alc880_pin_asus_init_verbs,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003265 alc880_gpio1_init_verbs },
3266 .num_dacs = ARRAY_SIZE(alc880_asus_dac_nids),
3267 .dac_nids = alc880_asus_dac_nids,
Takashi Iwai16ded522005-06-10 19:58:24 +02003268 .dig_out_nid = ALC880_DIGOUT_NID,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003269 .num_channel_mode = ARRAY_SIZE(alc880_asus_modes),
3270 .channel_mode = alc880_asus_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02003271 .need_dac_fix = 1,
Takashi Iwai16ded522005-06-10 19:58:24 +02003272 .input_mux = &alc880_capture_source,
3273 },
3274 [ALC880_UNIWILL_DIG] = {
Takashi Iwai3c10a9d2005-08-23 20:02:27 +02003275 .mixers = { alc880_asus_mixer, alc880_pcbeep_mixer },
Kailang Yangccc656c2006-10-17 12:32:26 +02003276 .init_verbs = { alc880_volume_init_verbs,
3277 alc880_pin_asus_init_verbs },
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003278 .num_dacs = ARRAY_SIZE(alc880_asus_dac_nids),
3279 .dac_nids = alc880_asus_dac_nids,
Takashi Iwai16ded522005-06-10 19:58:24 +02003280 .dig_out_nid = ALC880_DIGOUT_NID,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003281 .num_channel_mode = ARRAY_SIZE(alc880_asus_modes),
3282 .channel_mode = alc880_asus_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02003283 .need_dac_fix = 1,
Takashi Iwai16ded522005-06-10 19:58:24 +02003284 .input_mux = &alc880_capture_source,
3285 },
Kailang Yangccc656c2006-10-17 12:32:26 +02003286 [ALC880_UNIWILL] = {
3287 .mixers = { alc880_uniwill_mixer },
3288 .init_verbs = { alc880_volume_init_verbs,
3289 alc880_uniwill_init_verbs },
3290 .num_dacs = ARRAY_SIZE(alc880_asus_dac_nids),
3291 .dac_nids = alc880_asus_dac_nids,
3292 .dig_out_nid = ALC880_DIGOUT_NID,
3293 .num_channel_mode = ARRAY_SIZE(alc880_threestack_modes),
3294 .channel_mode = alc880_threestack_modes,
3295 .need_dac_fix = 1,
3296 .input_mux = &alc880_capture_source,
3297 .unsol_event = alc880_uniwill_unsol_event,
3298 .init_hook = alc880_uniwill_automute,
3299 },
3300 [ALC880_UNIWILL_P53] = {
3301 .mixers = { alc880_uniwill_p53_mixer },
3302 .init_verbs = { alc880_volume_init_verbs,
3303 alc880_uniwill_p53_init_verbs },
3304 .num_dacs = ARRAY_SIZE(alc880_asus_dac_nids),
3305 .dac_nids = alc880_asus_dac_nids,
3306 .num_channel_mode = ARRAY_SIZE(alc880_w810_modes),
Tobin Davis2cf9f0f2007-02-07 16:04:25 +01003307 .channel_mode = alc880_threestack_modes,
3308 .input_mux = &alc880_capture_source,
3309 .unsol_event = alc880_uniwill_p53_unsol_event,
3310 .init_hook = alc880_uniwill_p53_hp_automute,
3311 },
3312 [ALC880_FUJITSU] = {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003313 .mixers = { alc880_fujitsu_mixer,
Tobin Davis2cf9f0f2007-02-07 16:04:25 +01003314 alc880_pcbeep_mixer, },
3315 .init_verbs = { alc880_volume_init_verbs,
3316 alc880_uniwill_p53_init_verbs,
3317 alc880_beep_init_verbs },
3318 .num_dacs = ARRAY_SIZE(alc880_dac_nids),
3319 .dac_nids = alc880_dac_nids,
Takashi Iwaid53d7d92007-08-20 15:20:02 +02003320 .dig_out_nid = ALC880_DIGOUT_NID,
Tobin Davis2cf9f0f2007-02-07 16:04:25 +01003321 .num_channel_mode = ARRAY_SIZE(alc880_2_jack_modes),
3322 .channel_mode = alc880_2_jack_modes,
Kailang Yangccc656c2006-10-17 12:32:26 +02003323 .input_mux = &alc880_capture_source,
3324 .unsol_event = alc880_uniwill_p53_unsol_event,
3325 .init_hook = alc880_uniwill_p53_hp_automute,
3326 },
Kailang Yangdf694da2005-12-05 19:42:22 +01003327 [ALC880_CLEVO] = {
3328 .mixers = { alc880_three_stack_mixer },
3329 .init_verbs = { alc880_volume_init_verbs,
3330 alc880_pin_clevo_init_verbs },
3331 .num_dacs = ARRAY_SIZE(alc880_dac_nids),
3332 .dac_nids = alc880_dac_nids,
3333 .hp_nid = 0x03,
3334 .num_channel_mode = ARRAY_SIZE(alc880_threestack_modes),
3335 .channel_mode = alc880_threestack_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02003336 .need_dac_fix = 1,
Kailang Yangdf694da2005-12-05 19:42:22 +01003337 .input_mux = &alc880_capture_source,
3338 },
Takashi Iwaiae6b8132006-03-03 16:47:17 +01003339 [ALC880_LG] = {
3340 .mixers = { alc880_lg_mixer },
3341 .init_verbs = { alc880_volume_init_verbs,
3342 alc880_lg_init_verbs },
3343 .num_dacs = ARRAY_SIZE(alc880_lg_dac_nids),
3344 .dac_nids = alc880_lg_dac_nids,
3345 .dig_out_nid = ALC880_DIGOUT_NID,
3346 .num_channel_mode = ARRAY_SIZE(alc880_lg_ch_modes),
3347 .channel_mode = alc880_lg_ch_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02003348 .need_dac_fix = 1,
Takashi Iwaiae6b8132006-03-03 16:47:17 +01003349 .input_mux = &alc880_lg_capture_source,
3350 .unsol_event = alc880_lg_unsol_event,
3351 .init_hook = alc880_lg_automute,
Takashi Iwaicb53c622007-08-10 17:21:45 +02003352#ifdef CONFIG_SND_HDA_POWER_SAVE
3353 .loopbacks = alc880_lg_loopbacks,
3354#endif
Takashi Iwaiae6b8132006-03-03 16:47:17 +01003355 },
Takashi Iwaid6815182006-03-23 16:06:23 +01003356 [ALC880_LG_LW] = {
3357 .mixers = { alc880_lg_lw_mixer },
3358 .init_verbs = { alc880_volume_init_verbs,
3359 alc880_lg_lw_init_verbs },
Claudio Matsuoka0a8c5da2007-07-04 15:17:38 +02003360 .num_dacs = ARRAY_SIZE(alc880_dac_nids),
Takashi Iwaid6815182006-03-23 16:06:23 +01003361 .dac_nids = alc880_dac_nids,
3362 .dig_out_nid = ALC880_DIGOUT_NID,
Claudio Matsuoka0a8c5da2007-07-04 15:17:38 +02003363 .num_channel_mode = ARRAY_SIZE(alc880_lg_lw_modes),
3364 .channel_mode = alc880_lg_lw_modes,
Takashi Iwaid6815182006-03-23 16:06:23 +01003365 .input_mux = &alc880_lg_lw_capture_source,
3366 .unsol_event = alc880_lg_lw_unsol_event,
3367 .init_hook = alc880_lg_lw_automute,
3368 },
Takashi Iwaidf99cd32008-04-25 15:25:04 +02003369 [ALC880_MEDION_RIM] = {
3370 .mixers = { alc880_medion_rim_mixer },
3371 .init_verbs = { alc880_volume_init_verbs,
3372 alc880_medion_rim_init_verbs,
3373 alc_gpio2_init_verbs },
3374 .num_dacs = ARRAY_SIZE(alc880_dac_nids),
3375 .dac_nids = alc880_dac_nids,
3376 .dig_out_nid = ALC880_DIGOUT_NID,
3377 .num_channel_mode = ARRAY_SIZE(alc880_2_jack_modes),
3378 .channel_mode = alc880_2_jack_modes,
3379 .input_mux = &alc880_medion_rim_capture_source,
3380 .unsol_event = alc880_medion_rim_unsol_event,
3381 .init_hook = alc880_medion_rim_automute,
3382 },
Takashi Iwai16ded522005-06-10 19:58:24 +02003383#ifdef CONFIG_SND_DEBUG
3384 [ALC880_TEST] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003385 .mixers = { alc880_test_mixer },
3386 .init_verbs = { alc880_test_init_verbs },
Takashi Iwai16ded522005-06-10 19:58:24 +02003387 .num_dacs = ARRAY_SIZE(alc880_test_dac_nids),
3388 .dac_nids = alc880_test_dac_nids,
3389 .dig_out_nid = ALC880_DIGOUT_NID,
Takashi Iwai16ded522005-06-10 19:58:24 +02003390 .num_channel_mode = ARRAY_SIZE(alc880_test_modes),
3391 .channel_mode = alc880_test_modes,
3392 .input_mux = &alc880_test_capture_source,
3393 },
3394#endif
3395};
3396
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003397/*
3398 * Automatic parse of I/O pins from the BIOS configuration
3399 */
3400
3401#define NUM_CONTROL_ALLOC 32
3402#define NUM_VERB_ALLOC 32
3403
3404enum {
3405 ALC_CTL_WIDGET_VOL,
3406 ALC_CTL_WIDGET_MUTE,
3407 ALC_CTL_BIND_MUTE,
3408};
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01003409static struct snd_kcontrol_new alc880_control_templates[] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003410 HDA_CODEC_VOLUME(NULL, 0, 0, 0),
3411 HDA_CODEC_MUTE(NULL, 0, 0, 0),
Takashi Iwai985be542005-11-02 18:26:49 +01003412 HDA_BIND_MUTE(NULL, 0, 0, 0),
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003413};
3414
3415/* add dynamic controls */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003416static int add_control(struct alc_spec *spec, int type, const char *name,
3417 unsigned long val)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003418{
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01003419 struct snd_kcontrol_new *knew;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003420
3421 if (spec->num_kctl_used >= spec->num_kctl_alloc) {
3422 int num = spec->num_kctl_alloc + NUM_CONTROL_ALLOC;
3423
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003424 /* array + terminator */
3425 knew = kcalloc(num + 1, sizeof(*knew), GFP_KERNEL);
3426 if (!knew)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003427 return -ENOMEM;
3428 if (spec->kctl_alloc) {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003429 memcpy(knew, spec->kctl_alloc,
3430 sizeof(*knew) * spec->num_kctl_alloc);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003431 kfree(spec->kctl_alloc);
3432 }
3433 spec->kctl_alloc = knew;
3434 spec->num_kctl_alloc = num;
3435 }
3436
3437 knew = &spec->kctl_alloc[spec->num_kctl_used];
3438 *knew = alc880_control_templates[type];
Paulo Marques543537b2005-06-23 00:09:02 -07003439 knew->name = kstrdup(name, GFP_KERNEL);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003440 if (!knew->name)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003441 return -ENOMEM;
3442 knew->private_value = val;
3443 spec->num_kctl_used++;
3444 return 0;
3445}
3446
3447#define alc880_is_fixed_pin(nid) ((nid) >= 0x14 && (nid) <= 0x17)
3448#define alc880_fixed_pin_idx(nid) ((nid) - 0x14)
3449#define alc880_is_multi_pin(nid) ((nid) >= 0x18)
3450#define alc880_multi_pin_idx(nid) ((nid) - 0x18)
3451#define alc880_is_input_pin(nid) ((nid) >= 0x18)
3452#define alc880_input_pin_idx(nid) ((nid) - 0x18)
3453#define alc880_idx_to_dac(nid) ((nid) + 0x02)
3454#define alc880_dac_to_idx(nid) ((nid) - 0x02)
3455#define alc880_idx_to_mixer(nid) ((nid) + 0x0c)
3456#define alc880_idx_to_selector(nid) ((nid) + 0x10)
3457#define ALC880_PIN_CD_NID 0x1c
3458
3459/* fill in the dac_nids table from the parsed pin configuration */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003460static int alc880_auto_fill_dac_nids(struct alc_spec *spec,
3461 const struct auto_pin_cfg *cfg)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003462{
3463 hda_nid_t nid;
3464 int assigned[4];
3465 int i, j;
3466
3467 memset(assigned, 0, sizeof(assigned));
Takashi Iwaib0af0de2005-06-21 14:49:19 +02003468 spec->multiout.dac_nids = spec->private_dac_nids;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003469
3470 /* check the pins hardwired to audio widget */
3471 for (i = 0; i < cfg->line_outs; i++) {
3472 nid = cfg->line_out_pins[i];
3473 if (alc880_is_fixed_pin(nid)) {
3474 int idx = alc880_fixed_pin_idx(nid);
Libin Yang5014f192005-11-23 15:48:36 +01003475 spec->multiout.dac_nids[i] = alc880_idx_to_dac(idx);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003476 assigned[idx] = 1;
3477 }
3478 }
3479 /* left pins can be connect to any audio widget */
3480 for (i = 0; i < cfg->line_outs; i++) {
3481 nid = cfg->line_out_pins[i];
3482 if (alc880_is_fixed_pin(nid))
3483 continue;
3484 /* search for an empty channel */
3485 for (j = 0; j < cfg->line_outs; j++) {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003486 if (!assigned[j]) {
3487 spec->multiout.dac_nids[i] =
3488 alc880_idx_to_dac(j);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003489 assigned[j] = 1;
3490 break;
3491 }
3492 }
3493 }
3494 spec->multiout.num_dacs = cfg->line_outs;
3495 return 0;
3496}
3497
3498/* add playback controls from the parsed DAC table */
Kailang Yangdf694da2005-12-05 19:42:22 +01003499static int alc880_auto_create_multi_out_ctls(struct alc_spec *spec,
3500 const struct auto_pin_cfg *cfg)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003501{
3502 char name[32];
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003503 static const char *chname[4] = {
3504 "Front", "Surround", NULL /*CLFE*/, "Side"
3505 };
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003506 hda_nid_t nid;
3507 int i, err;
3508
3509 for (i = 0; i < cfg->line_outs; i++) {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003510 if (!spec->multiout.dac_nids[i])
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003511 continue;
3512 nid = alc880_idx_to_mixer(alc880_dac_to_idx(spec->multiout.dac_nids[i]));
3513 if (i == 2) {
3514 /* Center/LFE */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003515 err = add_control(spec, ALC_CTL_WIDGET_VOL,
3516 "Center Playback Volume",
3517 HDA_COMPOSE_AMP_VAL(nid, 1, 0,
3518 HDA_OUTPUT));
3519 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003520 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003521 err = add_control(spec, ALC_CTL_WIDGET_VOL,
3522 "LFE Playback Volume",
3523 HDA_COMPOSE_AMP_VAL(nid, 2, 0,
3524 HDA_OUTPUT));
3525 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003526 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003527 err = add_control(spec, ALC_CTL_BIND_MUTE,
3528 "Center Playback Switch",
3529 HDA_COMPOSE_AMP_VAL(nid, 1, 2,
3530 HDA_INPUT));
3531 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003532 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003533 err = add_control(spec, ALC_CTL_BIND_MUTE,
3534 "LFE Playback Switch",
3535 HDA_COMPOSE_AMP_VAL(nid, 2, 2,
3536 HDA_INPUT));
3537 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003538 return err;
3539 } else {
3540 sprintf(name, "%s Playback Volume", chname[i]);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003541 err = add_control(spec, ALC_CTL_WIDGET_VOL, name,
3542 HDA_COMPOSE_AMP_VAL(nid, 3, 0,
3543 HDA_OUTPUT));
3544 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003545 return err;
3546 sprintf(name, "%s Playback Switch", chname[i]);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003547 err = add_control(spec, ALC_CTL_BIND_MUTE, name,
3548 HDA_COMPOSE_AMP_VAL(nid, 3, 2,
3549 HDA_INPUT));
3550 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003551 return err;
3552 }
3553 }
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003554 return 0;
3555}
3556
Takashi Iwai8d88bc32005-11-17 11:09:23 +01003557/* add playback controls for speaker and HP outputs */
3558static int alc880_auto_create_extra_out(struct alc_spec *spec, hda_nid_t pin,
3559 const char *pfx)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003560{
3561 hda_nid_t nid;
3562 int err;
Takashi Iwai8d88bc32005-11-17 11:09:23 +01003563 char name[32];
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003564
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003565 if (!pin)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003566 return 0;
3567
3568 if (alc880_is_fixed_pin(pin)) {
3569 nid = alc880_idx_to_dac(alc880_fixed_pin_idx(pin));
Takashi Iwai82bc9552006-03-21 11:24:42 +01003570 /* specify the DAC as the extra output */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003571 if (!spec->multiout.hp_nid)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003572 spec->multiout.hp_nid = nid;
Takashi Iwai82bc9552006-03-21 11:24:42 +01003573 else
3574 spec->multiout.extra_out_nid[0] = nid;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003575 /* control HP volume/switch on the output mixer amp */
3576 nid = alc880_idx_to_mixer(alc880_fixed_pin_idx(pin));
Takashi Iwai8d88bc32005-11-17 11:09:23 +01003577 sprintf(name, "%s Playback Volume", pfx);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003578 err = add_control(spec, ALC_CTL_WIDGET_VOL, name,
3579 HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_OUTPUT));
3580 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003581 return err;
Takashi Iwai8d88bc32005-11-17 11:09:23 +01003582 sprintf(name, "%s Playback Switch", pfx);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003583 err = add_control(spec, ALC_CTL_BIND_MUTE, name,
3584 HDA_COMPOSE_AMP_VAL(nid, 3, 2, HDA_INPUT));
3585 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003586 return err;
3587 } else if (alc880_is_multi_pin(pin)) {
3588 /* set manual connection */
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003589 /* we have only a switch on HP-out PIN */
Takashi Iwai8d88bc32005-11-17 11:09:23 +01003590 sprintf(name, "%s Playback Switch", pfx);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003591 err = add_control(spec, ALC_CTL_WIDGET_MUTE, name,
3592 HDA_COMPOSE_AMP_VAL(pin, 3, 0, HDA_OUTPUT));
3593 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003594 return err;
3595 }
3596 return 0;
3597}
3598
3599/* create input playback/capture controls for the given pin */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003600static int new_analog_input(struct alc_spec *spec, hda_nid_t pin,
3601 const char *ctlname,
Kailang Yangdf694da2005-12-05 19:42:22 +01003602 int idx, hda_nid_t mix_nid)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003603{
3604 char name[32];
Kailang Yangdf694da2005-12-05 19:42:22 +01003605 int err;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003606
3607 sprintf(name, "%s Playback Volume", ctlname);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003608 err = add_control(spec, ALC_CTL_WIDGET_VOL, name,
3609 HDA_COMPOSE_AMP_VAL(mix_nid, 3, idx, HDA_INPUT));
3610 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003611 return err;
3612 sprintf(name, "%s Playback Switch", ctlname);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003613 err = add_control(spec, ALC_CTL_WIDGET_MUTE, name,
3614 HDA_COMPOSE_AMP_VAL(mix_nid, 3, idx, HDA_INPUT));
3615 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003616 return err;
3617 return 0;
3618}
3619
3620/* create playback/capture controls for input pins */
Kailang Yangdf694da2005-12-05 19:42:22 +01003621static int alc880_auto_create_analog_input_ctls(struct alc_spec *spec,
3622 const struct auto_pin_cfg *cfg)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003623{
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003624 struct hda_input_mux *imux = &spec->private_imux;
Kailang Yangdf694da2005-12-05 19:42:22 +01003625 int i, err, idx;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003626
3627 for (i = 0; i < AUTO_PIN_LAST; i++) {
3628 if (alc880_is_input_pin(cfg->input_pins[i])) {
Kailang Yangdf694da2005-12-05 19:42:22 +01003629 idx = alc880_input_pin_idx(cfg->input_pins[i]);
Takashi Iwai4a471b72005-12-07 13:56:29 +01003630 err = new_analog_input(spec, cfg->input_pins[i],
3631 auto_pin_cfg_labels[i],
Kailang Yangdf694da2005-12-05 19:42:22 +01003632 idx, 0x0b);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003633 if (err < 0)
3634 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003635 imux->items[imux->num_items].label =
3636 auto_pin_cfg_labels[i];
3637 imux->items[imux->num_items].index =
3638 alc880_input_pin_idx(cfg->input_pins[i]);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003639 imux->num_items++;
3640 }
3641 }
3642 return 0;
3643}
3644
Takashi Iwaif6c7e542008-02-12 18:32:23 +01003645static void alc_set_pin_output(struct hda_codec *codec, hda_nid_t nid,
3646 unsigned int pin_type)
3647{
3648 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
3649 pin_type);
3650 /* unmute pin */
Takashi Iwaid260cdf2008-02-13 17:19:35 +01003651 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_AMP_GAIN_MUTE,
3652 AMP_OUT_UNMUTE);
Takashi Iwaif6c7e542008-02-12 18:32:23 +01003653}
3654
Kailang Yangdf694da2005-12-05 19:42:22 +01003655static void alc880_auto_set_output_and_unmute(struct hda_codec *codec,
3656 hda_nid_t nid, int pin_type,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003657 int dac_idx)
3658{
Takashi Iwaif6c7e542008-02-12 18:32:23 +01003659 alc_set_pin_output(codec, nid, pin_type);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003660 /* need the manual connection? */
3661 if (alc880_is_multi_pin(nid)) {
3662 struct alc_spec *spec = codec->spec;
3663 int idx = alc880_multi_pin_idx(nid);
3664 snd_hda_codec_write(codec, alc880_idx_to_selector(idx), 0,
3665 AC_VERB_SET_CONNECT_SEL,
3666 alc880_dac_to_idx(spec->multiout.dac_nids[dac_idx]));
3667 }
3668}
3669
Takashi Iwaibaba8ee2007-04-23 17:17:48 +02003670static int get_pin_type(int line_out_type)
3671{
3672 if (line_out_type == AUTO_PIN_HP_OUT)
3673 return PIN_HP;
3674 else
3675 return PIN_OUT;
3676}
3677
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003678static void alc880_auto_init_multi_out(struct hda_codec *codec)
3679{
3680 struct alc_spec *spec = codec->spec;
3681 int i;
Kailang Yangbc9f98a2007-04-12 13:06:07 +02003682
3683 alc_subsystem_id(codec, 0x15, 0x1b, 0x14);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003684 for (i = 0; i < spec->autocfg.line_outs; i++) {
3685 hda_nid_t nid = spec->autocfg.line_out_pins[i];
Takashi Iwaibaba8ee2007-04-23 17:17:48 +02003686 int pin_type = get_pin_type(spec->autocfg.line_out_type);
3687 alc880_auto_set_output_and_unmute(codec, nid, pin_type, i);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003688 }
3689}
3690
Takashi Iwai8d88bc32005-11-17 11:09:23 +01003691static void alc880_auto_init_extra_out(struct hda_codec *codec)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003692{
3693 struct alc_spec *spec = codec->spec;
3694 hda_nid_t pin;
3695
Takashi Iwai82bc9552006-03-21 11:24:42 +01003696 pin = spec->autocfg.speaker_pins[0];
Takashi Iwai8d88bc32005-11-17 11:09:23 +01003697 if (pin) /* connect to front */
3698 alc880_auto_set_output_and_unmute(codec, pin, PIN_OUT, 0);
Takashi Iwaieb06ed82006-09-20 17:10:27 +02003699 pin = spec->autocfg.hp_pins[0];
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003700 if (pin) /* connect to front */
3701 alc880_auto_set_output_and_unmute(codec, pin, PIN_HP, 0);
3702}
3703
3704static void alc880_auto_init_analog_input(struct hda_codec *codec)
3705{
3706 struct alc_spec *spec = codec->spec;
3707 int i;
3708
3709 for (i = 0; i < AUTO_PIN_LAST; i++) {
3710 hda_nid_t nid = spec->autocfg.input_pins[i];
3711 if (alc880_is_input_pin(nid)) {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003712 snd_hda_codec_write(codec, nid, 0,
3713 AC_VERB_SET_PIN_WIDGET_CONTROL,
3714 i <= AUTO_PIN_FRONT_MIC ?
3715 PIN_VREF80 : PIN_IN);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003716 if (nid != ALC880_PIN_CD_NID)
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003717 snd_hda_codec_write(codec, nid, 0,
3718 AC_VERB_SET_AMP_GAIN_MUTE,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003719 AMP_OUT_MUTE);
3720 }
3721 }
3722}
3723
3724/* parse the BIOS configuration and set up the alc_spec */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003725/* return 1 if successful, 0 if the proper config is not found,
3726 * or a negative error code
3727 */
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003728static int alc880_parse_auto_config(struct hda_codec *codec)
3729{
3730 struct alc_spec *spec = codec->spec;
3731 int err;
Kailang Yangdf694da2005-12-05 19:42:22 +01003732 static hda_nid_t alc880_ignore[] = { 0x1d, 0 };
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003733
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003734 err = snd_hda_parse_pin_def_config(codec, &spec->autocfg,
3735 alc880_ignore);
3736 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003737 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003738 if (!spec->autocfg.line_outs)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003739 return 0; /* can't find valid BIOS pin config */
Kailang Yangdf694da2005-12-05 19:42:22 +01003740
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003741 err = alc880_auto_fill_dac_nids(spec, &spec->autocfg);
3742 if (err < 0)
3743 return err;
3744 err = alc880_auto_create_multi_out_ctls(spec, &spec->autocfg);
3745 if (err < 0)
3746 return err;
3747 err = alc880_auto_create_extra_out(spec,
3748 spec->autocfg.speaker_pins[0],
3749 "Speaker");
3750 if (err < 0)
3751 return err;
3752 err = alc880_auto_create_extra_out(spec, spec->autocfg.hp_pins[0],
3753 "Headphone");
3754 if (err < 0)
3755 return err;
3756 err = alc880_auto_create_analog_input_ctls(spec, &spec->autocfg);
3757 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003758 return err;
3759
3760 spec->multiout.max_channels = spec->multiout.num_dacs * 2;
3761
3762 if (spec->autocfg.dig_out_pin)
3763 spec->multiout.dig_out_nid = ALC880_DIGOUT_NID;
3764 if (spec->autocfg.dig_in_pin)
3765 spec->dig_in_nid = ALC880_DIGIN_NID;
3766
3767 if (spec->kctl_alloc)
3768 spec->mixers[spec->num_mixers++] = spec->kctl_alloc;
3769
3770 spec->init_verbs[spec->num_init_verbs++] = alc880_volume_init_verbs;
3771
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02003772 spec->num_mux_defs = 1;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003773 spec->input_mux = &spec->private_imux;
3774
3775 return 1;
3776}
3777
Takashi Iwaiae6b8132006-03-03 16:47:17 +01003778/* additional initialization for auto-configuration model */
3779static void alc880_auto_init(struct hda_codec *codec)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003780{
Takashi Iwaif6c7e542008-02-12 18:32:23 +01003781 struct alc_spec *spec = codec->spec;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003782 alc880_auto_init_multi_out(codec);
Takashi Iwai8d88bc32005-11-17 11:09:23 +01003783 alc880_auto_init_extra_out(codec);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003784 alc880_auto_init_analog_input(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +01003785 if (spec->unsol_event)
3786 alc_sku_automute(codec);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003787}
3788
3789/*
3790 * OK, here we have finally the patch for ALC880
3791 */
3792
Linus Torvalds1da177e2005-04-16 15:20:36 -07003793static int patch_alc880(struct hda_codec *codec)
3794{
3795 struct alc_spec *spec;
3796 int board_config;
Kailang Yangdf694da2005-12-05 19:42:22 +01003797 int err;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003798
Takashi Iwaie560d8d2005-09-09 14:21:46 +02003799 spec = kzalloc(sizeof(*spec), GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003800 if (spec == NULL)
3801 return -ENOMEM;
3802
3803 codec->spec = spec;
3804
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003805 board_config = snd_hda_check_board_config(codec, ALC880_MODEL_LAST,
3806 alc880_models,
3807 alc880_cfg_tbl);
3808 if (board_config < 0) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +02003809 printk(KERN_INFO "hda_codec: Unknown model for ALC880, "
3810 "trying auto-probe from BIOS...\n");
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003811 board_config = ALC880_AUTO;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003812 }
3813
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003814 if (board_config == ALC880_AUTO) {
3815 /* automatic parse from the BIOS config */
3816 err = alc880_parse_auto_config(codec);
3817 if (err < 0) {
3818 alc_free(codec);
3819 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003820 } else if (!err) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +02003821 printk(KERN_INFO
3822 "hda_codec: Cannot set up configuration "
3823 "from BIOS. Using 3-stack mode...\n");
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003824 board_config = ALC880_3ST;
3825 }
3826 }
3827
Kailang Yangdf694da2005-12-05 19:42:22 +01003828 if (board_config != ALC880_AUTO)
3829 setup_preset(spec, &alc880_presets[board_config]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003830
3831 spec->stream_name_analog = "ALC880 Analog";
3832 spec->stream_analog_playback = &alc880_pcm_analog_playback;
3833 spec->stream_analog_capture = &alc880_pcm_analog_capture;
Takashi Iwai63300792008-01-24 15:31:36 +01003834 spec->stream_analog_alt_capture = &alc880_pcm_analog_alt_capture;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003835
3836 spec->stream_name_digital = "ALC880 Digital";
3837 spec->stream_digital_playback = &alc880_pcm_digital_playback;
3838 spec->stream_digital_capture = &alc880_pcm_digital_capture;
3839
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003840 if (!spec->adc_nids && spec->input_mux) {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003841 /* check whether NID 0x07 is valid */
Takashi Iwai54d17402005-11-21 16:33:22 +01003842 unsigned int wcap = get_wcaps(codec, alc880_adc_nids[0]);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003843 /* get type */
3844 wcap = (wcap & AC_WCAP_TYPE) >> AC_WCAP_TYPE_SHIFT;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003845 if (wcap != AC_WID_AUD_IN) {
3846 spec->adc_nids = alc880_adc_nids_alt;
3847 spec->num_adc_nids = ARRAY_SIZE(alc880_adc_nids_alt);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003848 spec->mixers[spec->num_mixers] =
3849 alc880_capture_alt_mixer;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003850 spec->num_mixers++;
3851 } else {
3852 spec->adc_nids = alc880_adc_nids;
3853 spec->num_adc_nids = ARRAY_SIZE(alc880_adc_nids);
3854 spec->mixers[spec->num_mixers] = alc880_capture_mixer;
3855 spec->num_mixers++;
3856 }
3857 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07003858
Takashi Iwai2134ea42008-01-10 16:53:55 +01003859 spec->vmaster_nid = 0x0c;
3860
Linus Torvalds1da177e2005-04-16 15:20:36 -07003861 codec->patch_ops = alc_patch_ops;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003862 if (board_config == ALC880_AUTO)
Takashi Iwaiae6b8132006-03-03 16:47:17 +01003863 spec->init_hook = alc880_auto_init;
Takashi Iwaicb53c622007-08-10 17:21:45 +02003864#ifdef CONFIG_SND_HDA_POWER_SAVE
3865 if (!spec->loopback.amplist)
3866 spec->loopback.amplist = alc880_loopbacks;
3867#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07003868
3869 return 0;
3870}
3871
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003872
Linus Torvalds1da177e2005-04-16 15:20:36 -07003873/*
3874 * ALC260 support
3875 */
3876
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003877static hda_nid_t alc260_dac_nids[1] = {
3878 /* front */
3879 0x02,
3880};
3881
3882static hda_nid_t alc260_adc_nids[1] = {
3883 /* ADC0 */
3884 0x04,
3885};
3886
Kailang Yangdf694da2005-12-05 19:42:22 +01003887static hda_nid_t alc260_adc_nids_alt[1] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003888 /* ADC1 */
3889 0x05,
3890};
3891
Kailang Yangdf694da2005-12-05 19:42:22 +01003892static hda_nid_t alc260_hp_adc_nids[2] = {
3893 /* ADC1, 0 */
3894 0x05, 0x04
3895};
3896
Jonathan Woithed57fdac2006-02-28 11:38:35 +01003897/* NIDs used when simultaneous access to both ADCs makes sense. Note that
3898 * alc260_capture_mixer assumes ADC0 (nid 0x04) is the first ADC.
3899 */
3900static hda_nid_t alc260_dual_adc_nids[2] = {
Jonathan Woithe4c5186e2006-02-09 11:53:48 +01003901 /* ADC0, ADC1 */
3902 0x04, 0x05
3903};
3904
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003905#define ALC260_DIGOUT_NID 0x03
3906#define ALC260_DIGIN_NID 0x06
3907
3908static struct hda_input_mux alc260_capture_source = {
3909 .num_items = 4,
3910 .items = {
3911 { "Mic", 0x0 },
3912 { "Front Mic", 0x1 },
3913 { "Line", 0x2 },
3914 { "CD", 0x4 },
3915 },
3916};
3917
Jonathan Woithe17e7aec2006-02-28 11:35:18 +01003918/* On Fujitsu S702x laptops capture only makes sense from Mic/LineIn jack,
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02003919 * headphone jack and the internal CD lines since these are the only pins at
3920 * which audio can appear. For flexibility, also allow the option of
3921 * recording the mixer output on the second ADC (ADC0 doesn't have a
3922 * connection to the mixer output).
Jonathan Woithea9430dd2005-09-16 19:12:48 +02003923 */
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02003924static struct hda_input_mux alc260_fujitsu_capture_sources[2] = {
3925 {
3926 .num_items = 3,
3927 .items = {
3928 { "Mic/Line", 0x0 },
3929 { "CD", 0x4 },
3930 { "Headphone", 0x2 },
3931 },
Jonathan Woithea9430dd2005-09-16 19:12:48 +02003932 },
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02003933 {
3934 .num_items = 4,
3935 .items = {
3936 { "Mic/Line", 0x0 },
3937 { "CD", 0x4 },
3938 { "Headphone", 0x2 },
3939 { "Mixer", 0x5 },
3940 },
3941 },
3942
Jonathan Woithea9430dd2005-09-16 19:12:48 +02003943};
3944
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02003945/* Acer TravelMate(/Extensa/Aspire) notebooks have similar configuration to
3946 * the Fujitsu S702x, but jacks are marked differently.
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01003947 */
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02003948static struct hda_input_mux alc260_acer_capture_sources[2] = {
3949 {
3950 .num_items = 4,
3951 .items = {
3952 { "Mic", 0x0 },
3953 { "Line", 0x2 },
3954 { "CD", 0x4 },
3955 { "Headphone", 0x5 },
3956 },
3957 },
3958 {
3959 .num_items = 5,
3960 .items = {
3961 { "Mic", 0x0 },
3962 { "Line", 0x2 },
3963 { "CD", 0x4 },
3964 { "Headphone", 0x6 },
3965 { "Mixer", 0x5 },
3966 },
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01003967 },
3968};
Linus Torvalds1da177e2005-04-16 15:20:36 -07003969/*
3970 * This is just place-holder, so there's something for alc_build_pcms to look
3971 * at when it calculates the maximum number of channels. ALC260 has no mixer
3972 * element which allows changing the channel mode, so the verb list is
3973 * never used.
3974 */
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +01003975static struct hda_channel_mode alc260_modes[1] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003976 { 2, NULL },
3977};
3978
Kailang Yangdf694da2005-12-05 19:42:22 +01003979
3980/* Mixer combinations
3981 *
3982 * basic: base_output + input + pc_beep + capture
3983 * HP: base_output + input + capture_alt
3984 * HP_3013: hp_3013 + input + capture
3985 * fujitsu: fujitsu + capture
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01003986 * acer: acer + capture
Kailang Yangdf694da2005-12-05 19:42:22 +01003987 */
3988
3989static struct snd_kcontrol_new alc260_base_output_mixer[] = {
Takashi Iwai05acb862005-06-10 19:50:25 +02003990 HDA_CODEC_VOLUME("Front Playback Volume", 0x08, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01003991 HDA_BIND_MUTE("Front Playback Switch", 0x08, 2, HDA_INPUT),
Kailang Yangdf694da2005-12-05 19:42:22 +01003992 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x09, 0x0, HDA_OUTPUT),
3993 HDA_BIND_MUTE("Headphone Playback Switch", 0x09, 2, HDA_INPUT),
3994 HDA_CODEC_VOLUME_MONO("Mono Playback Volume", 0x0a, 1, 0x0, HDA_OUTPUT),
3995 HDA_BIND_MUTE_MONO("Mono Playback Switch", 0x0a, 1, 2, HDA_INPUT),
3996 { } /* end */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003997};
Kailang Yangdf694da2005-12-05 19:42:22 +01003998
3999static struct snd_kcontrol_new alc260_input_mixer[] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07004000 HDA_CODEC_VOLUME("CD Playback Volume", 0x07, 0x04, HDA_INPUT),
4001 HDA_CODEC_MUTE("CD Playback Switch", 0x07, 0x04, HDA_INPUT),
4002 HDA_CODEC_VOLUME("Line Playback Volume", 0x07, 0x02, HDA_INPUT),
4003 HDA_CODEC_MUTE("Line Playback Switch", 0x07, 0x02, HDA_INPUT),
4004 HDA_CODEC_VOLUME("Mic Playback Volume", 0x07, 0x0, HDA_INPUT),
4005 HDA_CODEC_MUTE("Mic Playback Switch", 0x07, 0x0, HDA_INPUT),
4006 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x07, 0x01, HDA_INPUT),
4007 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x07, 0x01, HDA_INPUT),
Linus Torvalds1da177e2005-04-16 15:20:36 -07004008 { } /* end */
4009};
4010
Kailang Yangdf694da2005-12-05 19:42:22 +01004011static struct snd_kcontrol_new alc260_pc_beep_mixer[] = {
4012 HDA_CODEC_VOLUME("PC Speaker Playback Volume", 0x07, 0x05, HDA_INPUT),
4013 HDA_CODEC_MUTE("PC Speaker Playback Switch", 0x07, 0x05, HDA_INPUT),
4014 { } /* end */
4015};
4016
Takashi Iwaibec15c32008-01-28 18:16:30 +01004017/* update HP, line and mono out pins according to the master switch */
4018static void alc260_hp_master_update(struct hda_codec *codec,
4019 hda_nid_t hp, hda_nid_t line,
4020 hda_nid_t mono)
4021{
4022 struct alc_spec *spec = codec->spec;
4023 unsigned int val = spec->master_sw ? PIN_HP : 0;
4024 /* change HP and line-out pins */
4025 snd_hda_codec_write(codec, 0x0f, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
4026 val);
4027 snd_hda_codec_write(codec, 0x10, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
4028 val);
4029 /* mono (speaker) depending on the HP jack sense */
4030 val = (val && !spec->jack_present) ? PIN_OUT : 0;
4031 snd_hda_codec_write(codec, 0x11, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
4032 val);
4033}
4034
4035static int alc260_hp_master_sw_get(struct snd_kcontrol *kcontrol,
4036 struct snd_ctl_elem_value *ucontrol)
4037{
4038 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
4039 struct alc_spec *spec = codec->spec;
4040 *ucontrol->value.integer.value = spec->master_sw;
4041 return 0;
4042}
4043
4044static int alc260_hp_master_sw_put(struct snd_kcontrol *kcontrol,
4045 struct snd_ctl_elem_value *ucontrol)
4046{
4047 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
4048 struct alc_spec *spec = codec->spec;
4049 int val = !!*ucontrol->value.integer.value;
4050 hda_nid_t hp, line, mono;
4051
4052 if (val == spec->master_sw)
4053 return 0;
4054 spec->master_sw = val;
4055 hp = (kcontrol->private_value >> 16) & 0xff;
4056 line = (kcontrol->private_value >> 8) & 0xff;
4057 mono = kcontrol->private_value & 0xff;
4058 alc260_hp_master_update(codec, hp, line, mono);
4059 return 1;
4060}
4061
4062static struct snd_kcontrol_new alc260_hp_output_mixer[] = {
4063 {
4064 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
4065 .name = "Master Playback Switch",
4066 .info = snd_ctl_boolean_mono_info,
4067 .get = alc260_hp_master_sw_get,
4068 .put = alc260_hp_master_sw_put,
4069 .private_value = (0x0f << 16) | (0x10 << 8) | 0x11
4070 },
4071 HDA_CODEC_VOLUME("Front Playback Volume", 0x08, 0x0, HDA_OUTPUT),
4072 HDA_BIND_MUTE("Front Playback Switch", 0x08, 2, HDA_INPUT),
4073 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x09, 0x0, HDA_OUTPUT),
4074 HDA_BIND_MUTE("Headphone Playback Switch", 0x09, 2, HDA_INPUT),
4075 HDA_CODEC_VOLUME_MONO("Speaker Playback Volume", 0x0a, 1, 0x0,
4076 HDA_OUTPUT),
4077 HDA_BIND_MUTE_MONO("Speaker Playback Switch", 0x0a, 1, 2, HDA_INPUT),
4078 { } /* end */
4079};
4080
4081static struct hda_verb alc260_hp_unsol_verbs[] = {
4082 {0x10, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
4083 {},
4084};
4085
4086static void alc260_hp_automute(struct hda_codec *codec)
4087{
4088 struct alc_spec *spec = codec->spec;
4089 unsigned int present;
4090
4091 present = snd_hda_codec_read(codec, 0x10, 0,
4092 AC_VERB_GET_PIN_SENSE, 0);
4093 spec->jack_present = (present & AC_PINSENSE_PRESENCE) != 0;
4094 alc260_hp_master_update(codec, 0x0f, 0x10, 0x11);
4095}
4096
4097static void alc260_hp_unsol_event(struct hda_codec *codec, unsigned int res)
4098{
4099 if ((res >> 26) == ALC880_HP_EVENT)
4100 alc260_hp_automute(codec);
4101}
4102
Kailang Yangdf694da2005-12-05 19:42:22 +01004103static struct snd_kcontrol_new alc260_hp_3013_mixer[] = {
Takashi Iwaibec15c32008-01-28 18:16:30 +01004104 {
4105 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
4106 .name = "Master Playback Switch",
4107 .info = snd_ctl_boolean_mono_info,
4108 .get = alc260_hp_master_sw_get,
4109 .put = alc260_hp_master_sw_put,
4110 .private_value = (0x10 << 16) | (0x15 << 8) | 0x11
4111 },
Kailang Yangdf694da2005-12-05 19:42:22 +01004112 HDA_CODEC_VOLUME("Front Playback Volume", 0x09, 0x0, HDA_OUTPUT),
4113 HDA_CODEC_MUTE("Front Playback Switch", 0x10, 0x0, HDA_OUTPUT),
4114 HDA_CODEC_VOLUME("Aux-In Playback Volume", 0x07, 0x06, HDA_INPUT),
4115 HDA_CODEC_MUTE("Aux-In Playback Switch", 0x07, 0x06, HDA_INPUT),
4116 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x08, 0x0, HDA_OUTPUT),
4117 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
Takashi Iwai86cd9292008-01-28 18:09:56 +01004118 HDA_CODEC_VOLUME_MONO("Speaker Playback Volume", 0x0a, 1, 0x0, HDA_OUTPUT),
4119 HDA_CODEC_MUTE_MONO("Speaker Playback Switch", 0x11, 1, 0x0, HDA_OUTPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02004120 { } /* end */
4121};
4122
Takashi Iwaibec15c32008-01-28 18:16:30 +01004123static struct hda_verb alc260_hp_3013_unsol_verbs[] = {
4124 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
4125 {},
4126};
4127
4128static void alc260_hp_3013_automute(struct hda_codec *codec)
4129{
4130 struct alc_spec *spec = codec->spec;
4131 unsigned int present;
4132
4133 present = snd_hda_codec_read(codec, 0x15, 0,
4134 AC_VERB_GET_PIN_SENSE, 0);
4135 spec->jack_present = (present & AC_PINSENSE_PRESENCE) != 0;
4136 alc260_hp_master_update(codec, 0x10, 0x15, 0x11);
4137}
4138
4139static void alc260_hp_3013_unsol_event(struct hda_codec *codec,
4140 unsigned int res)
4141{
4142 if ((res >> 26) == ALC880_HP_EVENT)
4143 alc260_hp_3013_automute(codec);
4144}
4145
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02004146/* Fujitsu S702x series laptops. ALC260 pin usage: Mic/Line jack = 0x12,
4147 * HP jack = 0x14, CD audio = 0x16, internal speaker = 0x10.
4148 */
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01004149static struct snd_kcontrol_new alc260_fujitsu_mixer[] = {
Jonathan Woithea9430dd2005-09-16 19:12:48 +02004150 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x08, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01004151 HDA_BIND_MUTE("Headphone Playback Switch", 0x08, 2, HDA_INPUT),
Jonathan Woithe4c5186e2006-02-09 11:53:48 +01004152 ALC_PIN_MODE("Headphone Jack Mode", 0x14, ALC_PIN_DIR_INOUT),
Jonathan Woithea9430dd2005-09-16 19:12:48 +02004153 HDA_CODEC_VOLUME("CD Playback Volume", 0x07, 0x04, HDA_INPUT),
4154 HDA_CODEC_MUTE("CD Playback Switch", 0x07, 0x04, HDA_INPUT),
4155 HDA_CODEC_VOLUME("Mic/Line Playback Volume", 0x07, 0x0, HDA_INPUT),
4156 HDA_CODEC_MUTE("Mic/Line Playback Switch", 0x07, 0x0, HDA_INPUT),
Jonathan Woithe4c5186e2006-02-09 11:53:48 +01004157 ALC_PIN_MODE("Mic/Line Jack Mode", 0x12, ALC_PIN_DIR_IN),
Jonathan Woithea9430dd2005-09-16 19:12:48 +02004158 HDA_CODEC_VOLUME("Beep Playback Volume", 0x07, 0x05, HDA_INPUT),
4159 HDA_CODEC_MUTE("Beep Playback Switch", 0x07, 0x05, HDA_INPUT),
Takashi Iwai31bffaa2008-02-27 16:10:44 +01004160 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x09, 0x0, HDA_OUTPUT),
4161 HDA_BIND_MUTE("Speaker Playback Switch", 0x09, 2, HDA_INPUT),
Kailang Yangdf694da2005-12-05 19:42:22 +01004162 { } /* end */
4163};
4164
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02004165/* Mixer for Acer TravelMate(/Extensa/Aspire) notebooks. Note that current
4166 * versions of the ALC260 don't act on requests to enable mic bias from NID
4167 * 0x0f (used to drive the headphone jack in these laptops). The ALC260
4168 * datasheet doesn't mention this restriction. At this stage it's not clear
4169 * whether this behaviour is intentional or is a hardware bug in chip
4170 * revisions available in early 2006. Therefore for now allow the
4171 * "Headphone Jack Mode" control to span all choices, but if it turns out
4172 * that the lack of mic bias for this NID is intentional we could change the
4173 * mode from ALC_PIN_DIR_INOUT to ALC_PIN_DIR_INOUT_NOMICBIAS.
4174 *
4175 * In addition, Acer TravelMate(/Extensa/Aspire) notebooks in early 2006
4176 * don't appear to make the mic bias available from the "line" jack, even
4177 * though the NID used for this jack (0x14) can supply it. The theory is
4178 * that perhaps Acer have included blocking capacitors between the ALC260
4179 * and the output jack. If this turns out to be the case for all such
4180 * models the "Line Jack Mode" mode could be changed from ALC_PIN_DIR_INOUT
4181 * to ALC_PIN_DIR_INOUT_NOMICBIAS.
Jonathan Woithebd869482006-11-28 11:35:52 +01004182 *
4183 * The C20x Tablet series have a mono internal speaker which is controlled
4184 * via the chip's Mono sum widget and pin complex, so include the necessary
4185 * controls for such models. On models without a "mono speaker" the control
4186 * won't do anything.
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02004187 */
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01004188static struct snd_kcontrol_new alc260_acer_mixer[] = {
4189 HDA_CODEC_VOLUME("Master Playback Volume", 0x08, 0x0, HDA_OUTPUT),
4190 HDA_BIND_MUTE("Master Playback Switch", 0x08, 2, HDA_INPUT),
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02004191 ALC_PIN_MODE("Headphone Jack Mode", 0x0f, ALC_PIN_DIR_INOUT),
Takashi Iwai31bffaa2008-02-27 16:10:44 +01004192 HDA_CODEC_VOLUME_MONO("Speaker Playback Volume", 0x0a, 1, 0x0,
Jonathan Woithebd869482006-11-28 11:35:52 +01004193 HDA_OUTPUT),
Takashi Iwai31bffaa2008-02-27 16:10:44 +01004194 HDA_BIND_MUTE_MONO("Speaker Playback Switch", 0x0a, 1, 2,
Jonathan Woithebd869482006-11-28 11:35:52 +01004195 HDA_INPUT),
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01004196 HDA_CODEC_VOLUME("CD Playback Volume", 0x07, 0x04, HDA_INPUT),
4197 HDA_CODEC_MUTE("CD Playback Switch", 0x07, 0x04, HDA_INPUT),
4198 HDA_CODEC_VOLUME("Mic Playback Volume", 0x07, 0x0, HDA_INPUT),
4199 HDA_CODEC_MUTE("Mic Playback Switch", 0x07, 0x0, HDA_INPUT),
4200 ALC_PIN_MODE("Mic Jack Mode", 0x12, ALC_PIN_DIR_IN),
4201 HDA_CODEC_VOLUME("Line Playback Volume", 0x07, 0x02, HDA_INPUT),
4202 HDA_CODEC_MUTE("Line Playback Switch", 0x07, 0x02, HDA_INPUT),
4203 ALC_PIN_MODE("Line Jack Mode", 0x14, ALC_PIN_DIR_INOUT),
4204 HDA_CODEC_VOLUME("Beep Playback Volume", 0x07, 0x05, HDA_INPUT),
4205 HDA_CODEC_MUTE("Beep Playback Switch", 0x07, 0x05, HDA_INPUT),
4206 { } /* end */
4207};
4208
Kailang Yangbc9f98a2007-04-12 13:06:07 +02004209/* Packard bell V7900 ALC260 pin usage: HP = 0x0f, Mic jack = 0x12,
4210 * Line In jack = 0x14, CD audio = 0x16, pc beep = 0x17.
4211 */
4212static struct snd_kcontrol_new alc260_will_mixer[] = {
4213 HDA_CODEC_VOLUME("Master Playback Volume", 0x08, 0x0, HDA_OUTPUT),
4214 HDA_BIND_MUTE("Master Playback Switch", 0x08, 0x2, HDA_INPUT),
4215 HDA_CODEC_VOLUME("Mic Playback Volume", 0x07, 0x0, HDA_INPUT),
4216 HDA_CODEC_MUTE("Mic Playback Switch", 0x07, 0x0, HDA_INPUT),
4217 ALC_PIN_MODE("Mic Jack Mode", 0x12, ALC_PIN_DIR_IN),
4218 HDA_CODEC_VOLUME("Line Playback Volume", 0x07, 0x02, HDA_INPUT),
4219 HDA_CODEC_MUTE("Line Playback Switch", 0x07, 0x02, HDA_INPUT),
4220 ALC_PIN_MODE("Line Jack Mode", 0x14, ALC_PIN_DIR_INOUT),
4221 HDA_CODEC_VOLUME("CD Playback Volume", 0x07, 0x04, HDA_INPUT),
4222 HDA_CODEC_MUTE("CD Playback Switch", 0x07, 0x04, HDA_INPUT),
4223 HDA_CODEC_VOLUME("Beep Playback Volume", 0x07, 0x05, HDA_INPUT),
4224 HDA_CODEC_MUTE("Beep Playback Switch", 0x07, 0x05, HDA_INPUT),
4225 { } /* end */
4226};
4227
4228/* Replacer 672V ALC260 pin usage: Mic jack = 0x12,
4229 * Line In jack = 0x14, ATAPI Mic = 0x13, speaker = 0x0f.
4230 */
4231static struct snd_kcontrol_new alc260_replacer_672v_mixer[] = {
4232 HDA_CODEC_VOLUME("Master Playback Volume", 0x08, 0x0, HDA_OUTPUT),
4233 HDA_BIND_MUTE("Master Playback Switch", 0x08, 0x2, HDA_INPUT),
4234 HDA_CODEC_VOLUME("Mic Playback Volume", 0x07, 0x0, HDA_INPUT),
4235 HDA_CODEC_MUTE("Mic Playback Switch", 0x07, 0x0, HDA_INPUT),
4236 ALC_PIN_MODE("Mic Jack Mode", 0x12, ALC_PIN_DIR_IN),
4237 HDA_CODEC_VOLUME("ATAPI Mic Playback Volume", 0x07, 0x1, HDA_INPUT),
4238 HDA_CODEC_MUTE("ATATI Mic Playback Switch", 0x07, 0x1, HDA_INPUT),
4239 HDA_CODEC_VOLUME("Line Playback Volume", 0x07, 0x02, HDA_INPUT),
4240 HDA_CODEC_MUTE("Line Playback Switch", 0x07, 0x02, HDA_INPUT),
4241 ALC_PIN_MODE("Line Jack Mode", 0x14, ALC_PIN_DIR_INOUT),
4242 { } /* end */
4243};
4244
Kailang Yangdf694da2005-12-05 19:42:22 +01004245/* capture mixer elements */
4246static struct snd_kcontrol_new alc260_capture_mixer[] = {
Jonathan Woithea9430dd2005-09-16 19:12:48 +02004247 HDA_CODEC_VOLUME("Capture Volume", 0x04, 0x0, HDA_INPUT),
4248 HDA_CODEC_MUTE("Capture Switch", 0x04, 0x0, HDA_INPUT),
Kailang Yangdf694da2005-12-05 19:42:22 +01004249 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x05, 0x0, HDA_INPUT),
4250 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x05, 0x0, HDA_INPUT),
Jonathan Woithea9430dd2005-09-16 19:12:48 +02004251 {
4252 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
Kailang Yangdf694da2005-12-05 19:42:22 +01004253 /* The multiple "Capture Source" controls confuse alsamixer
4254 * So call somewhat different..
Kailang Yangdf694da2005-12-05 19:42:22 +01004255 */
4256 /* .name = "Capture Source", */
4257 .name = "Input Source",
4258 .count = 2,
Jonathan Woithea9430dd2005-09-16 19:12:48 +02004259 .info = alc_mux_enum_info,
4260 .get = alc_mux_enum_get,
4261 .put = alc_mux_enum_put,
4262 },
4263 { } /* end */
4264};
4265
Kailang Yangdf694da2005-12-05 19:42:22 +01004266static struct snd_kcontrol_new alc260_capture_alt_mixer[] = {
4267 HDA_CODEC_VOLUME("Capture Volume", 0x05, 0x0, HDA_INPUT),
4268 HDA_CODEC_MUTE("Capture Switch", 0x05, 0x0, HDA_INPUT),
4269 {
4270 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
4271 /* The multiple "Capture Source" controls confuse alsamixer
4272 * So call somewhat different..
Kailang Yangdf694da2005-12-05 19:42:22 +01004273 */
4274 /* .name = "Capture Source", */
4275 .name = "Input Source",
4276 .count = 1,
4277 .info = alc_mux_enum_info,
4278 .get = alc_mux_enum_get,
4279 .put = alc_mux_enum_put,
4280 },
4281 { } /* end */
4282};
4283
4284/*
4285 * initialization verbs
4286 */
Linus Torvalds1da177e2005-04-16 15:20:36 -07004287static struct hda_verb alc260_init_verbs[] = {
4288 /* Line In pin widget for input */
Takashi Iwai05acb862005-06-10 19:50:25 +02004289 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Linus Torvalds1da177e2005-04-16 15:20:36 -07004290 /* CD pin widget for input */
Takashi Iwai05acb862005-06-10 19:50:25 +02004291 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Linus Torvalds1da177e2005-04-16 15:20:36 -07004292 /* Mic1 (rear panel) pin widget for input and vref at 80% */
Takashi Iwai16ded522005-06-10 19:58:24 +02004293 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Linus Torvalds1da177e2005-04-16 15:20:36 -07004294 /* Mic2 (front panel) pin widget for input and vref at 80% */
Takashi Iwai16ded522005-06-10 19:58:24 +02004295 {0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Linus Torvalds1da177e2005-04-16 15:20:36 -07004296 /* LINE-2 is used for line-out in rear */
Takashi Iwai05acb862005-06-10 19:50:25 +02004297 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Linus Torvalds1da177e2005-04-16 15:20:36 -07004298 /* select line-out */
Jonathan Woithefd56f2d2006-01-24 10:35:46 +01004299 {0x0e, AC_VERB_SET_CONNECT_SEL, 0x00},
Linus Torvalds1da177e2005-04-16 15:20:36 -07004300 /* LINE-OUT pin */
Takashi Iwai05acb862005-06-10 19:50:25 +02004301 {0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Linus Torvalds1da177e2005-04-16 15:20:36 -07004302 /* enable HP */
Takashi Iwai05acb862005-06-10 19:50:25 +02004303 {0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Linus Torvalds1da177e2005-04-16 15:20:36 -07004304 /* enable Mono */
Takashi Iwai05acb862005-06-10 19:50:25 +02004305 {0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
4306 /* mute capture amp left and right */
Takashi Iwai16ded522005-06-10 19:58:24 +02004307 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07004308 /* set connection select to line in (default select for this ADC) */
4309 {0x04, AC_VERB_SET_CONNECT_SEL, 0x02},
Takashi Iwai16ded522005-06-10 19:58:24 +02004310 /* mute capture amp left and right */
4311 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
4312 /* set connection select to line in (default select for this ADC) */
4313 {0x05, AC_VERB_SET_CONNECT_SEL, 0x02},
Takashi Iwai05acb862005-06-10 19:50:25 +02004314 /* set vol=0 Line-Out mixer amp left and right */
4315 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
4316 /* unmute pin widget amp left and right (no gain on this amp) */
4317 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
4318 /* set vol=0 HP mixer amp left and right */
4319 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
4320 /* unmute pin widget amp left and right (no gain on this amp) */
4321 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
4322 /* set vol=0 Mono mixer amp left and right */
4323 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
4324 /* unmute pin widget amp left and right (no gain on this amp) */
4325 {0x11, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
4326 /* unmute LINE-2 out pin */
4327 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004328 /* Amp Indexes: CD = 0x04, Line In 1 = 0x02, Mic 1 = 0x00 &
4329 * Line In 2 = 0x03
4330 */
Takashi Iwaicb53c622007-08-10 17:21:45 +02004331 /* mute analog inputs */
4332 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4333 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
4334 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
4335 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
4336 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07004337 /* Amp Indexes: DAC = 0x01 & mixer = 0x00 */
Takashi Iwai05acb862005-06-10 19:50:25 +02004338 /* mute Front out path */
4339 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4340 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
4341 /* mute Headphone out path */
4342 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4343 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
4344 /* mute Mono out path */
4345 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4346 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07004347 { }
4348};
4349
Takashi Iwai474167d2006-05-17 17:17:43 +02004350#if 0 /* should be identical with alc260_init_verbs? */
Kailang Yangdf694da2005-12-05 19:42:22 +01004351static struct hda_verb alc260_hp_init_verbs[] = {
4352 /* Headphone and output */
4353 {0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc0},
4354 /* mono output */
4355 {0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40},
4356 /* Mic1 (rear panel) pin widget for input and vref at 80% */
4357 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
4358 /* Mic2 (front panel) pin widget for input and vref at 80% */
4359 {0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
4360 /* Line In pin widget for input */
4361 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
4362 /* Line-2 pin widget for output */
4363 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40},
4364 /* CD pin widget for input */
4365 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
4366 /* unmute amp left and right */
4367 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000},
4368 /* set connection select to line in (default select for this ADC) */
4369 {0x04, AC_VERB_SET_CONNECT_SEL, 0x02},
4370 /* unmute Line-Out mixer amp left and right (volume = 0) */
4371 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, 0xb000},
4372 /* mute pin widget amp left and right (no gain on this amp) */
4373 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
4374 /* unmute HP mixer amp left and right (volume = 0) */
4375 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, 0xb000},
4376 /* mute pin widget amp left and right (no gain on this amp) */
4377 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004378 /* Amp Indexes: CD = 0x04, Line In 1 = 0x02, Mic 1 = 0x00 &
4379 * Line In 2 = 0x03
4380 */
Takashi Iwaicb53c622007-08-10 17:21:45 +02004381 /* mute analog inputs */
4382 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4383 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
4384 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
4385 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
4386 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Kailang Yangdf694da2005-12-05 19:42:22 +01004387 /* Amp Indexes: DAC = 0x01 & mixer = 0x00 */
4388 /* Unmute Front out path */
4389 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
4390 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
4391 /* Unmute Headphone out path */
4392 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
4393 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
4394 /* Unmute Mono out path */
4395 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
4396 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
4397 { }
4398};
Takashi Iwai474167d2006-05-17 17:17:43 +02004399#endif
Kailang Yangdf694da2005-12-05 19:42:22 +01004400
4401static struct hda_verb alc260_hp_3013_init_verbs[] = {
4402 /* Line out and output */
4403 {0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40},
4404 /* mono output */
4405 {0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40},
4406 /* Mic1 (rear panel) pin widget for input and vref at 80% */
4407 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
4408 /* Mic2 (front panel) pin widget for input and vref at 80% */
4409 {0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
4410 /* Line In pin widget for input */
4411 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
4412 /* Headphone pin widget for output */
4413 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc0},
4414 /* CD pin widget for input */
4415 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
4416 /* unmute amp left and right */
4417 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000},
4418 /* set connection select to line in (default select for this ADC) */
4419 {0x04, AC_VERB_SET_CONNECT_SEL, 0x02},
4420 /* unmute Line-Out mixer amp left and right (volume = 0) */
4421 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, 0xb000},
4422 /* mute pin widget amp left and right (no gain on this amp) */
4423 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
4424 /* unmute HP mixer amp left and right (volume = 0) */
4425 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, 0xb000},
4426 /* mute pin widget amp left and right (no gain on this amp) */
4427 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004428 /* Amp Indexes: CD = 0x04, Line In 1 = 0x02, Mic 1 = 0x00 &
4429 * Line In 2 = 0x03
4430 */
Takashi Iwaicb53c622007-08-10 17:21:45 +02004431 /* mute analog inputs */
4432 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4433 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
4434 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
4435 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
4436 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Kailang Yangdf694da2005-12-05 19:42:22 +01004437 /* Amp Indexes: DAC = 0x01 & mixer = 0x00 */
4438 /* Unmute Front out path */
4439 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
4440 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
4441 /* Unmute Headphone out path */
4442 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
4443 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
4444 /* Unmute Mono out path */
4445 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
4446 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
4447 { }
4448};
4449
Jonathan Woithea9430dd2005-09-16 19:12:48 +02004450/* Initialisation sequence for ALC260 as configured in Fujitsu S702x
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02004451 * laptops. ALC260 pin usage: Mic/Line jack = 0x12, HP jack = 0x14, CD
4452 * audio = 0x16, internal speaker = 0x10.
Jonathan Woithea9430dd2005-09-16 19:12:48 +02004453 */
4454static struct hda_verb alc260_fujitsu_init_verbs[] = {
4455 /* Disable all GPIOs */
4456 {0x01, AC_VERB_SET_GPIO_MASK, 0},
4457 /* Internal speaker is connected to headphone pin */
4458 {0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
4459 /* Headphone/Line-out jack connects to Line1 pin; make it an output */
4460 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Jonathan Woithef7ace402006-02-28 11:46:14 +01004461 /* Mic/Line-in jack is connected to mic1 pin, so make it an input */
4462 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
4463 /* Ensure all other unused pins are disabled and muted. */
4464 {0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
4465 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Jonathan Woithea9430dd2005-09-16 19:12:48 +02004466 {0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
Jonathan Woithef7ace402006-02-28 11:46:14 +01004467 {0x11, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Jonathan Woithea9430dd2005-09-16 19:12:48 +02004468 {0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
Jonathan Woithef7ace402006-02-28 11:46:14 +01004469 {0x13, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4470 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
4471 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Jonathan Woithea9430dd2005-09-16 19:12:48 +02004472
Jonathan Woithef7ace402006-02-28 11:46:14 +01004473 /* Disable digital (SPDIF) pins */
4474 {0x03, AC_VERB_SET_DIGI_CONVERT_1, 0},
4475 {0x06, AC_VERB_SET_DIGI_CONVERT_1, 0},
Jonathan Woithe4c5186e2006-02-09 11:53:48 +01004476
Jonathan Woithef7ace402006-02-28 11:46:14 +01004477 /* Ensure Line1 pin widget takes its input from the OUT1 sum bus
4478 * when acting as an output.
4479 */
4480 {0x0d, AC_VERB_SET_CONNECT_SEL, 0},
4481
4482 /* Start with output sum widgets muted and their output gains at min */
Takashi Iwai8b33a5a2006-02-09 11:57:01 +01004483 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4484 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
4485 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
4486 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4487 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
4488 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
4489 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4490 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
4491 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
Jonathan Woithea9430dd2005-09-16 19:12:48 +02004492
Jonathan Woithef7ace402006-02-28 11:46:14 +01004493 /* Unmute HP pin widget amp left and right (no equiv mixer ctrl) */
4494 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
4495 /* Unmute Line1 pin widget output buffer since it starts as an output.
4496 * If the pin mode is changed by the user the pin mode control will
4497 * take care of enabling the pin's input/output buffers as needed.
4498 * Therefore there's no need to enable the input buffer at this
4499 * stage.
Jonathan Woithecdcd9262006-02-28 11:36:42 +01004500 */
Jonathan Woithef7ace402006-02-28 11:46:14 +01004501 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Jonathan Woithecdcd9262006-02-28 11:36:42 +01004502 /* Unmute input buffer of pin widget used for Line-in (no equiv
4503 * mixer ctrl)
4504 */
Jonathan Woithef7ace402006-02-28 11:46:14 +01004505 {0x12, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Jonathan Woithea9430dd2005-09-16 19:12:48 +02004506
Jonathan Woithef7ace402006-02-28 11:46:14 +01004507 /* Mute capture amp left and right */
4508 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4509 /* Set ADC connection select to match default mixer setting - line
4510 * in (on mic1 pin)
4511 */
4512 {0x04, AC_VERB_SET_CONNECT_SEL, 0x00},
Jonathan Woithea9430dd2005-09-16 19:12:48 +02004513
Jonathan Woithef7ace402006-02-28 11:46:14 +01004514 /* Do the same for the second ADC: mute capture input amp and
4515 * set ADC connection to line in (on mic1 pin)
4516 */
4517 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4518 {0x05, AC_VERB_SET_CONNECT_SEL, 0x00},
Jonathan Woithe4c5186e2006-02-09 11:53:48 +01004519
Jonathan Woithef7ace402006-02-28 11:46:14 +01004520 /* Mute all inputs to mixer widget (even unconnected ones) */
4521 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)}, /* mic1 pin */
4522 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)}, /* mic2 pin */
4523 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)}, /* line1 pin */
4524 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)}, /* line2 pin */
4525 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)}, /* CD pin */
4526 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(5)}, /* Beep-gen pin */
4527 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(6)}, /* Line-out pin */
4528 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(7)}, /* HP-pin pin */
Takashi Iwai4a471b72005-12-07 13:56:29 +01004529
4530 { }
Jonathan Woithea9430dd2005-09-16 19:12:48 +02004531};
4532
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01004533/* Initialisation sequence for ALC260 as configured in Acer TravelMate and
4534 * similar laptops (adapted from Fujitsu init verbs).
4535 */
4536static struct hda_verb alc260_acer_init_verbs[] = {
4537 /* On TravelMate laptops, GPIO 0 enables the internal speaker and
4538 * the headphone jack. Turn this on and rely on the standard mute
4539 * methods whenever the user wants to turn these outputs off.
4540 */
4541 {0x01, AC_VERB_SET_GPIO_MASK, 0x01},
4542 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x01},
4543 {0x01, AC_VERB_SET_GPIO_DATA, 0x01},
4544 /* Internal speaker/Headphone jack is connected to Line-out pin */
4545 {0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
4546 /* Internal microphone/Mic jack is connected to Mic1 pin */
4547 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF50},
4548 /* Line In jack is connected to Line1 pin */
4549 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Jonathan Woithebd869482006-11-28 11:35:52 +01004550 /* Some Acers (eg: C20x Tablets) use Mono pin for internal speaker */
4551 {0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01004552 /* Ensure all other unused pins are disabled and muted. */
4553 {0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
4554 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01004555 {0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
4556 {0x13, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4557 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
4558 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4559 /* Disable digital (SPDIF) pins */
4560 {0x03, AC_VERB_SET_DIGI_CONVERT_1, 0},
4561 {0x06, AC_VERB_SET_DIGI_CONVERT_1, 0},
4562
4563 /* Ensure Mic1 and Line1 pin widgets take input from the OUT1 sum
4564 * bus when acting as outputs.
4565 */
4566 {0x0b, AC_VERB_SET_CONNECT_SEL, 0},
4567 {0x0d, AC_VERB_SET_CONNECT_SEL, 0},
4568
4569 /* Start with output sum widgets muted and their output gains at min */
4570 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4571 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
4572 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
4573 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4574 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
4575 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
4576 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4577 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
4578 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
4579
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004580 /* Unmute Line-out pin widget amp left and right
4581 * (no equiv mixer ctrl)
4582 */
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01004583 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Jonathan Woithebd869482006-11-28 11:35:52 +01004584 /* Unmute mono pin widget amp output (no equiv mixer ctrl) */
4585 {0x11, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01004586 /* Unmute Mic1 and Line1 pin widget input buffers since they start as
4587 * inputs. If the pin mode is changed by the user the pin mode control
4588 * will take care of enabling the pin's input/output buffers as needed.
4589 * Therefore there's no need to enable the input buffer at this
4590 * stage.
4591 */
4592 {0x12, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
4593 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
4594
4595 /* Mute capture amp left and right */
4596 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4597 /* Set ADC connection select to match default mixer setting - mic
4598 * (on mic1 pin)
4599 */
4600 {0x04, AC_VERB_SET_CONNECT_SEL, 0x00},
4601
4602 /* Do similar with the second ADC: mute capture input amp and
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02004603 * set ADC connection to mic to match ALSA's default state.
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01004604 */
4605 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02004606 {0x05, AC_VERB_SET_CONNECT_SEL, 0x00},
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01004607
4608 /* Mute all inputs to mixer widget (even unconnected ones) */
4609 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)}, /* mic1 pin */
4610 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)}, /* mic2 pin */
4611 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)}, /* line1 pin */
4612 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)}, /* line2 pin */
4613 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)}, /* CD pin */
4614 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(5)}, /* Beep-gen pin */
4615 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(6)}, /* Line-out pin */
4616 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(7)}, /* HP-pin pin */
4617
4618 { }
4619};
4620
Kailang Yangbc9f98a2007-04-12 13:06:07 +02004621static struct hda_verb alc260_will_verbs[] = {
4622 {0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
4623 {0x0b, AC_VERB_SET_CONNECT_SEL, 0x00},
4624 {0x0d, AC_VERB_SET_CONNECT_SEL, 0x00},
4625 {0x0f, AC_VERB_SET_EAPD_BTLENABLE, 0x02},
4626 {0x1a, AC_VERB_SET_COEF_INDEX, 0x07},
4627 {0x1a, AC_VERB_SET_PROC_COEF, 0x3040},
4628 {}
4629};
4630
4631static struct hda_verb alc260_replacer_672v_verbs[] = {
4632 {0x0f, AC_VERB_SET_EAPD_BTLENABLE, 0x02},
4633 {0x1a, AC_VERB_SET_COEF_INDEX, 0x07},
4634 {0x1a, AC_VERB_SET_PROC_COEF, 0x3050},
4635
4636 {0x01, AC_VERB_SET_GPIO_MASK, 0x01},
4637 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x01},
4638 {0x01, AC_VERB_SET_GPIO_DATA, 0x00},
4639
4640 {0x0f, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
4641 {}
4642};
4643
4644/* toggle speaker-output according to the hp-jack state */
4645static void alc260_replacer_672v_automute(struct hda_codec *codec)
4646{
4647 unsigned int present;
4648
4649 /* speaker --> GPIO Data 0, hp or spdif --> GPIO data 1 */
4650 present = snd_hda_codec_read(codec, 0x0f, 0,
4651 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
4652 if (present) {
Takashi Iwai82beb8f2007-08-10 17:09:26 +02004653 snd_hda_codec_write_cache(codec, 0x01, 0,
4654 AC_VERB_SET_GPIO_DATA, 1);
4655 snd_hda_codec_write_cache(codec, 0x0f, 0,
4656 AC_VERB_SET_PIN_WIDGET_CONTROL,
4657 PIN_HP);
Kailang Yangbc9f98a2007-04-12 13:06:07 +02004658 } else {
Takashi Iwai82beb8f2007-08-10 17:09:26 +02004659 snd_hda_codec_write_cache(codec, 0x01, 0,
4660 AC_VERB_SET_GPIO_DATA, 0);
4661 snd_hda_codec_write_cache(codec, 0x0f, 0,
4662 AC_VERB_SET_PIN_WIDGET_CONTROL,
4663 PIN_OUT);
Kailang Yangbc9f98a2007-04-12 13:06:07 +02004664 }
4665}
4666
4667static void alc260_replacer_672v_unsol_event(struct hda_codec *codec,
4668 unsigned int res)
4669{
4670 if ((res >> 26) == ALC880_HP_EVENT)
4671 alc260_replacer_672v_automute(codec);
4672}
4673
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01004674/* Test configuration for debugging, modelled after the ALC880 test
4675 * configuration.
4676 */
4677#ifdef CONFIG_SND_DEBUG
4678static hda_nid_t alc260_test_dac_nids[1] = {
4679 0x02,
4680};
4681static hda_nid_t alc260_test_adc_nids[2] = {
4682 0x04, 0x05,
4683};
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02004684/* For testing the ALC260, each input MUX needs its own definition since
4685 * the signal assignments are different. This assumes that the first ADC
4686 * is NID 0x04.
Jonathan Woithe17e7aec2006-02-28 11:35:18 +01004687 */
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02004688static struct hda_input_mux alc260_test_capture_sources[2] = {
4689 {
4690 .num_items = 7,
4691 .items = {
4692 { "MIC1 pin", 0x0 },
4693 { "MIC2 pin", 0x1 },
4694 { "LINE1 pin", 0x2 },
4695 { "LINE2 pin", 0x3 },
4696 { "CD pin", 0x4 },
4697 { "LINE-OUT pin", 0x5 },
4698 { "HP-OUT pin", 0x6 },
4699 },
4700 },
4701 {
4702 .num_items = 8,
4703 .items = {
4704 { "MIC1 pin", 0x0 },
4705 { "MIC2 pin", 0x1 },
4706 { "LINE1 pin", 0x2 },
4707 { "LINE2 pin", 0x3 },
4708 { "CD pin", 0x4 },
4709 { "Mixer", 0x5 },
4710 { "LINE-OUT pin", 0x6 },
4711 { "HP-OUT pin", 0x7 },
4712 },
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01004713 },
4714};
4715static struct snd_kcontrol_new alc260_test_mixer[] = {
4716 /* Output driver widgets */
4717 HDA_CODEC_VOLUME_MONO("Mono Playback Volume", 0x0a, 1, 0x0, HDA_OUTPUT),
4718 HDA_BIND_MUTE_MONO("Mono Playback Switch", 0x0a, 1, 2, HDA_INPUT),
4719 HDA_CODEC_VOLUME("LOUT2 Playback Volume", 0x09, 0x0, HDA_OUTPUT),
4720 HDA_BIND_MUTE("LOUT2 Playback Switch", 0x09, 2, HDA_INPUT),
4721 HDA_CODEC_VOLUME("LOUT1 Playback Volume", 0x08, 0x0, HDA_OUTPUT),
4722 HDA_BIND_MUTE("LOUT1 Playback Switch", 0x08, 2, HDA_INPUT),
4723
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02004724 /* Modes for retasking pin widgets
4725 * Note: the ALC260 doesn't seem to act on requests to enable mic
4726 * bias from NIDs 0x0f and 0x10. The ALC260 datasheet doesn't
4727 * mention this restriction. At this stage it's not clear whether
4728 * this behaviour is intentional or is a hardware bug in chip
4729 * revisions available at least up until early 2006. Therefore for
4730 * now allow the "HP-OUT" and "LINE-OUT" Mode controls to span all
4731 * choices, but if it turns out that the lack of mic bias for these
4732 * NIDs is intentional we could change their modes from
4733 * ALC_PIN_DIR_INOUT to ALC_PIN_DIR_INOUT_NOMICBIAS.
4734 */
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01004735 ALC_PIN_MODE("HP-OUT pin mode", 0x10, ALC_PIN_DIR_INOUT),
4736 ALC_PIN_MODE("LINE-OUT pin mode", 0x0f, ALC_PIN_DIR_INOUT),
4737 ALC_PIN_MODE("LINE2 pin mode", 0x15, ALC_PIN_DIR_INOUT),
4738 ALC_PIN_MODE("LINE1 pin mode", 0x14, ALC_PIN_DIR_INOUT),
4739 ALC_PIN_MODE("MIC2 pin mode", 0x13, ALC_PIN_DIR_INOUT),
4740 ALC_PIN_MODE("MIC1 pin mode", 0x12, ALC_PIN_DIR_INOUT),
4741
4742 /* Loopback mixer controls */
4743 HDA_CODEC_VOLUME("MIC1 Playback Volume", 0x07, 0x00, HDA_INPUT),
4744 HDA_CODEC_MUTE("MIC1 Playback Switch", 0x07, 0x00, HDA_INPUT),
4745 HDA_CODEC_VOLUME("MIC2 Playback Volume", 0x07, 0x01, HDA_INPUT),
4746 HDA_CODEC_MUTE("MIC2 Playback Switch", 0x07, 0x01, HDA_INPUT),
4747 HDA_CODEC_VOLUME("LINE1 Playback Volume", 0x07, 0x02, HDA_INPUT),
4748 HDA_CODEC_MUTE("LINE1 Playback Switch", 0x07, 0x02, HDA_INPUT),
4749 HDA_CODEC_VOLUME("LINE2 Playback Volume", 0x07, 0x03, HDA_INPUT),
4750 HDA_CODEC_MUTE("LINE2 Playback Switch", 0x07, 0x03, HDA_INPUT),
4751 HDA_CODEC_VOLUME("CD Playback Volume", 0x07, 0x04, HDA_INPUT),
4752 HDA_CODEC_MUTE("CD Playback Switch", 0x07, 0x04, HDA_INPUT),
4753 HDA_CODEC_VOLUME("Beep Playback Volume", 0x07, 0x05, HDA_INPUT),
4754 HDA_CODEC_MUTE("Beep Playback Switch", 0x07, 0x05, HDA_INPUT),
4755 HDA_CODEC_VOLUME("LINE-OUT loopback Playback Volume", 0x07, 0x06, HDA_INPUT),
4756 HDA_CODEC_MUTE("LINE-OUT loopback Playback Switch", 0x07, 0x06, HDA_INPUT),
4757 HDA_CODEC_VOLUME("HP-OUT loopback Playback Volume", 0x07, 0x7, HDA_INPUT),
4758 HDA_CODEC_MUTE("HP-OUT loopback Playback Switch", 0x07, 0x7, HDA_INPUT),
Jonathan Woithe5c8f8582006-02-28 11:43:27 +01004759
4760 /* Controls for GPIO pins, assuming they are configured as outputs */
4761 ALC_GPIO_DATA_SWITCH("GPIO pin 0", 0x01, 0x01),
4762 ALC_GPIO_DATA_SWITCH("GPIO pin 1", 0x01, 0x02),
4763 ALC_GPIO_DATA_SWITCH("GPIO pin 2", 0x01, 0x04),
4764 ALC_GPIO_DATA_SWITCH("GPIO pin 3", 0x01, 0x08),
4765
Jonathan Woithe92621f12006-02-28 11:47:47 +01004766 /* Switches to allow the digital IO pins to be enabled. The datasheet
4767 * is ambigious as to which NID is which; testing on laptops which
4768 * make this output available should provide clarification.
4769 */
4770 ALC_SPDIF_CTRL_SWITCH("SPDIF Playback Switch", 0x03, 0x01),
4771 ALC_SPDIF_CTRL_SWITCH("SPDIF Capture Switch", 0x06, 0x01),
4772
Jonathan Woithef8225f62008-01-08 12:16:54 +01004773 /* A switch allowing EAPD to be enabled. Some laptops seem to use
4774 * this output to turn on an external amplifier.
4775 */
4776 ALC_EAPD_CTRL_SWITCH("LINE-OUT EAPD Enable Switch", 0x0f, 0x02),
4777 ALC_EAPD_CTRL_SWITCH("HP-OUT EAPD Enable Switch", 0x10, 0x02),
4778
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01004779 { } /* end */
4780};
4781static struct hda_verb alc260_test_init_verbs[] = {
Jonathan Woithe5c8f8582006-02-28 11:43:27 +01004782 /* Enable all GPIOs as outputs with an initial value of 0 */
4783 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x0f},
4784 {0x01, AC_VERB_SET_GPIO_DATA, 0x00},
4785 {0x01, AC_VERB_SET_GPIO_MASK, 0x0f},
4786
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01004787 /* Enable retasking pins as output, initially without power amp */
4788 {0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
4789 {0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
4790 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
4791 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
4792 {0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
4793 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
4794
Jonathan Woithe92621f12006-02-28 11:47:47 +01004795 /* Disable digital (SPDIF) pins initially, but users can enable
4796 * them via a mixer switch. In the case of SPDIF-out, this initverb
4797 * payload also sets the generation to 0, output to be in "consumer"
4798 * PCM format, copyright asserted, no pre-emphasis and no validity
4799 * control.
4800 */
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01004801 {0x03, AC_VERB_SET_DIGI_CONVERT_1, 0},
4802 {0x06, AC_VERB_SET_DIGI_CONVERT_1, 0},
4803
Jonathan Woithef7ace402006-02-28 11:46:14 +01004804 /* Ensure mic1, mic2, line1 and line2 pin widgets take input from the
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01004805 * OUT1 sum bus when acting as an output.
4806 */
4807 {0x0b, AC_VERB_SET_CONNECT_SEL, 0},
4808 {0x0c, AC_VERB_SET_CONNECT_SEL, 0},
4809 {0x0d, AC_VERB_SET_CONNECT_SEL, 0},
4810 {0x0e, AC_VERB_SET_CONNECT_SEL, 0},
4811
4812 /* Start with output sum widgets muted and their output gains at min */
4813 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4814 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
4815 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
4816 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4817 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
4818 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
4819 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4820 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
4821 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
4822
Jonathan Woithecdcd9262006-02-28 11:36:42 +01004823 /* Unmute retasking pin widget output buffers since the default
4824 * state appears to be output. As the pin mode is changed by the
4825 * user the pin mode control will take care of enabling the pin's
4826 * input/output buffers as needed.
4827 */
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01004828 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
4829 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
4830 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
4831 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
4832 {0x13, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
4833 {0x12, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
4834 /* Also unmute the mono-out pin widget */
4835 {0x11, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
4836
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01004837 /* Mute capture amp left and right */
4838 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Jonathan Woithef7ace402006-02-28 11:46:14 +01004839 /* Set ADC connection select to match default mixer setting (mic1
4840 * pin)
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01004841 */
4842 {0x04, AC_VERB_SET_CONNECT_SEL, 0x00},
4843
4844 /* Do the same for the second ADC: mute capture input amp and
Jonathan Woithef7ace402006-02-28 11:46:14 +01004845 * set ADC connection to mic1 pin
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01004846 */
4847 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
4848 {0x05, AC_VERB_SET_CONNECT_SEL, 0x00},
4849
4850 /* Mute all inputs to mixer widget (even unconnected ones) */
4851 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)}, /* mic1 pin */
4852 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)}, /* mic2 pin */
4853 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)}, /* line1 pin */
4854 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)}, /* line2 pin */
4855 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)}, /* CD pin */
4856 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(5)}, /* Beep-gen pin */
4857 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(6)}, /* Line-out pin */
4858 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(7)}, /* HP-pin pin */
4859
4860 { }
4861};
4862#endif
4863
Takashi Iwai63300792008-01-24 15:31:36 +01004864#define alc260_pcm_analog_playback alc880_pcm_analog_alt_playback
4865#define alc260_pcm_analog_capture alc880_pcm_analog_capture
Linus Torvalds1da177e2005-04-16 15:20:36 -07004866
Takashi Iwaia3bcba32005-12-06 19:05:29 +01004867#define alc260_pcm_digital_playback alc880_pcm_digital_playback
4868#define alc260_pcm_digital_capture alc880_pcm_digital_capture
4869
Kailang Yangdf694da2005-12-05 19:42:22 +01004870/*
4871 * for BIOS auto-configuration
4872 */
4873
4874static int alc260_add_playback_controls(struct alc_spec *spec, hda_nid_t nid,
4875 const char *pfx)
4876{
4877 hda_nid_t nid_vol;
4878 unsigned long vol_val, sw_val;
4879 char name[32];
4880 int err;
4881
4882 if (nid >= 0x0f && nid < 0x11) {
4883 nid_vol = nid - 0x7;
4884 vol_val = HDA_COMPOSE_AMP_VAL(nid_vol, 3, 0, HDA_OUTPUT);
4885 sw_val = HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_OUTPUT);
4886 } else if (nid == 0x11) {
4887 nid_vol = nid - 0x7;
4888 vol_val = HDA_COMPOSE_AMP_VAL(nid_vol, 2, 0, HDA_OUTPUT);
4889 sw_val = HDA_COMPOSE_AMP_VAL(nid, 2, 0, HDA_OUTPUT);
4890 } else if (nid >= 0x12 && nid <= 0x15) {
4891 nid_vol = 0x08;
4892 vol_val = HDA_COMPOSE_AMP_VAL(nid_vol, 3, 0, HDA_OUTPUT);
4893 sw_val = HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_OUTPUT);
4894 } else
4895 return 0; /* N/A */
4896
4897 snprintf(name, sizeof(name), "%s Playback Volume", pfx);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004898 err = add_control(spec, ALC_CTL_WIDGET_VOL, name, vol_val);
4899 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +01004900 return err;
4901 snprintf(name, sizeof(name), "%s Playback Switch", pfx);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004902 err = add_control(spec, ALC_CTL_WIDGET_MUTE, name, sw_val);
4903 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +01004904 return err;
4905 return 1;
4906}
4907
4908/* add playback controls from the parsed DAC table */
4909static int alc260_auto_create_multi_out_ctls(struct alc_spec *spec,
4910 const struct auto_pin_cfg *cfg)
4911{
4912 hda_nid_t nid;
4913 int err;
4914
4915 spec->multiout.num_dacs = 1;
4916 spec->multiout.dac_nids = spec->private_dac_nids;
4917 spec->multiout.dac_nids[0] = 0x02;
4918
4919 nid = cfg->line_out_pins[0];
4920 if (nid) {
4921 err = alc260_add_playback_controls(spec, nid, "Front");
4922 if (err < 0)
4923 return err;
4924 }
4925
Takashi Iwai82bc9552006-03-21 11:24:42 +01004926 nid = cfg->speaker_pins[0];
Kailang Yangdf694da2005-12-05 19:42:22 +01004927 if (nid) {
4928 err = alc260_add_playback_controls(spec, nid, "Speaker");
4929 if (err < 0)
4930 return err;
4931 }
4932
Takashi Iwaieb06ed82006-09-20 17:10:27 +02004933 nid = cfg->hp_pins[0];
Kailang Yangdf694da2005-12-05 19:42:22 +01004934 if (nid) {
4935 err = alc260_add_playback_controls(spec, nid, "Headphone");
4936 if (err < 0)
4937 return err;
4938 }
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004939 return 0;
Kailang Yangdf694da2005-12-05 19:42:22 +01004940}
4941
4942/* create playback/capture controls for input pins */
4943static int alc260_auto_create_analog_input_ctls(struct alc_spec *spec,
4944 const struct auto_pin_cfg *cfg)
4945{
Kailang Yangdf694da2005-12-05 19:42:22 +01004946 struct hda_input_mux *imux = &spec->private_imux;
4947 int i, err, idx;
4948
4949 for (i = 0; i < AUTO_PIN_LAST; i++) {
4950 if (cfg->input_pins[i] >= 0x12) {
4951 idx = cfg->input_pins[i] - 0x12;
Takashi Iwai4a471b72005-12-07 13:56:29 +01004952 err = new_analog_input(spec, cfg->input_pins[i],
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004953 auto_pin_cfg_labels[i], idx,
4954 0x07);
Kailang Yangdf694da2005-12-05 19:42:22 +01004955 if (err < 0)
4956 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004957 imux->items[imux->num_items].label =
4958 auto_pin_cfg_labels[i];
Kailang Yangdf694da2005-12-05 19:42:22 +01004959 imux->items[imux->num_items].index = idx;
4960 imux->num_items++;
4961 }
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004962 if (cfg->input_pins[i] >= 0x0f && cfg->input_pins[i] <= 0x10){
Kailang Yangdf694da2005-12-05 19:42:22 +01004963 idx = cfg->input_pins[i] - 0x09;
Takashi Iwai4a471b72005-12-07 13:56:29 +01004964 err = new_analog_input(spec, cfg->input_pins[i],
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004965 auto_pin_cfg_labels[i], idx,
4966 0x07);
Kailang Yangdf694da2005-12-05 19:42:22 +01004967 if (err < 0)
4968 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004969 imux->items[imux->num_items].label =
4970 auto_pin_cfg_labels[i];
Kailang Yangdf694da2005-12-05 19:42:22 +01004971 imux->items[imux->num_items].index = idx;
4972 imux->num_items++;
4973 }
4974 }
4975 return 0;
4976}
4977
4978static void alc260_auto_set_output_and_unmute(struct hda_codec *codec,
4979 hda_nid_t nid, int pin_type,
4980 int sel_idx)
4981{
Takashi Iwaif6c7e542008-02-12 18:32:23 +01004982 alc_set_pin_output(codec, nid, pin_type);
Kailang Yangdf694da2005-12-05 19:42:22 +01004983 /* need the manual connection? */
4984 if (nid >= 0x12) {
4985 int idx = nid - 0x12;
4986 snd_hda_codec_write(codec, idx + 0x0b, 0,
4987 AC_VERB_SET_CONNECT_SEL, sel_idx);
Kailang Yangdf694da2005-12-05 19:42:22 +01004988 }
4989}
4990
4991static void alc260_auto_init_multi_out(struct hda_codec *codec)
4992{
4993 struct alc_spec *spec = codec->spec;
4994 hda_nid_t nid;
4995
Kailang Yangbc9f98a2007-04-12 13:06:07 +02004996 alc_subsystem_id(codec, 0x10, 0x15, 0x0f);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004997 nid = spec->autocfg.line_out_pins[0];
Takashi Iwaibaba8ee2007-04-23 17:17:48 +02004998 if (nid) {
4999 int pin_type = get_pin_type(spec->autocfg.line_out_type);
5000 alc260_auto_set_output_and_unmute(codec, nid, pin_type, 0);
5001 }
Kailang Yangdf694da2005-12-05 19:42:22 +01005002
Takashi Iwai82bc9552006-03-21 11:24:42 +01005003 nid = spec->autocfg.speaker_pins[0];
Kailang Yangdf694da2005-12-05 19:42:22 +01005004 if (nid)
5005 alc260_auto_set_output_and_unmute(codec, nid, PIN_OUT, 0);
5006
Takashi Iwaieb06ed82006-09-20 17:10:27 +02005007 nid = spec->autocfg.hp_pins[0];
Kailang Yangdf694da2005-12-05 19:42:22 +01005008 if (nid)
Takashi Iwaibaba8ee2007-04-23 17:17:48 +02005009 alc260_auto_set_output_and_unmute(codec, nid, PIN_HP, 0);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02005010}
Kailang Yangdf694da2005-12-05 19:42:22 +01005011
5012#define ALC260_PIN_CD_NID 0x16
5013static void alc260_auto_init_analog_input(struct hda_codec *codec)
5014{
5015 struct alc_spec *spec = codec->spec;
5016 int i;
5017
5018 for (i = 0; i < AUTO_PIN_LAST; i++) {
5019 hda_nid_t nid = spec->autocfg.input_pins[i];
5020 if (nid >= 0x12) {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02005021 snd_hda_codec_write(codec, nid, 0,
5022 AC_VERB_SET_PIN_WIDGET_CONTROL,
5023 i <= AUTO_PIN_FRONT_MIC ?
5024 PIN_VREF80 : PIN_IN);
Kailang Yangdf694da2005-12-05 19:42:22 +01005025 if (nid != ALC260_PIN_CD_NID)
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02005026 snd_hda_codec_write(codec, nid, 0,
5027 AC_VERB_SET_AMP_GAIN_MUTE,
Kailang Yangdf694da2005-12-05 19:42:22 +01005028 AMP_OUT_MUTE);
5029 }
5030 }
5031}
5032
5033/*
5034 * generic initialization of ADC, input mixers and output mixers
5035 */
5036static struct hda_verb alc260_volume_init_verbs[] = {
5037 /*
5038 * Unmute ADC0-1 and set the default input to mic-in
5039 */
5040 {0x04, AC_VERB_SET_CONNECT_SEL, 0x00},
5041 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5042 {0x05, AC_VERB_SET_CONNECT_SEL, 0x00},
5043 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5044
5045 /* Unmute input amps (CD, Line In, Mic 1 & Mic 2) of the analog-loopback
5046 * mixer widget
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02005047 * Note: PASD motherboards uses the Line In 2 as the input for
5048 * front panel mic (mic 2)
Kailang Yangdf694da2005-12-05 19:42:22 +01005049 */
5050 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +02005051 /* mute analog inputs */
5052 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5053 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5054 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
5055 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
5056 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Kailang Yangdf694da2005-12-05 19:42:22 +01005057
5058 /*
5059 * Set up output mixers (0x08 - 0x0a)
5060 */
5061 /* set vol=0 to output mixers */
5062 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5063 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5064 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5065 /* set up input amps for analog loopback */
5066 /* Amp Indices: DAC = 0, mixer = 1 */
5067 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5068 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
5069 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5070 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
5071 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5072 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
5073
5074 { }
5075};
5076
5077static int alc260_parse_auto_config(struct hda_codec *codec)
5078{
5079 struct alc_spec *spec = codec->spec;
5080 unsigned int wcap;
5081 int err;
5082 static hda_nid_t alc260_ignore[] = { 0x17, 0 };
5083
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02005084 err = snd_hda_parse_pin_def_config(codec, &spec->autocfg,
5085 alc260_ignore);
5086 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +01005087 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02005088 err = alc260_auto_create_multi_out_ctls(spec, &spec->autocfg);
5089 if (err < 0)
Takashi Iwai4a471b72005-12-07 13:56:29 +01005090 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02005091 if (!spec->kctl_alloc)
Kailang Yangdf694da2005-12-05 19:42:22 +01005092 return 0; /* can't find valid BIOS pin config */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02005093 err = alc260_auto_create_analog_input_ctls(spec, &spec->autocfg);
5094 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +01005095 return err;
5096
5097 spec->multiout.max_channels = 2;
5098
5099 if (spec->autocfg.dig_out_pin)
5100 spec->multiout.dig_out_nid = ALC260_DIGOUT_NID;
5101 if (spec->kctl_alloc)
5102 spec->mixers[spec->num_mixers++] = spec->kctl_alloc;
5103
5104 spec->init_verbs[spec->num_init_verbs++] = alc260_volume_init_verbs;
5105
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02005106 spec->num_mux_defs = 1;
Kailang Yangdf694da2005-12-05 19:42:22 +01005107 spec->input_mux = &spec->private_imux;
5108
5109 /* check whether NID 0x04 is valid */
Takashi Iwai4a471b72005-12-07 13:56:29 +01005110 wcap = get_wcaps(codec, 0x04);
Kailang Yangdf694da2005-12-05 19:42:22 +01005111 wcap = (wcap & AC_WCAP_TYPE) >> AC_WCAP_TYPE_SHIFT; /* get type */
Takashi Iwai67ebcb02008-02-19 15:03:57 +01005112 if (wcap != AC_WID_AUD_IN || spec->input_mux->num_items == 1) {
Kailang Yangdf694da2005-12-05 19:42:22 +01005113 spec->adc_nids = alc260_adc_nids_alt;
5114 spec->num_adc_nids = ARRAY_SIZE(alc260_adc_nids_alt);
5115 spec->mixers[spec->num_mixers] = alc260_capture_alt_mixer;
Kailang Yangdf694da2005-12-05 19:42:22 +01005116 } else {
5117 spec->adc_nids = alc260_adc_nids;
5118 spec->num_adc_nids = ARRAY_SIZE(alc260_adc_nids);
5119 spec->mixers[spec->num_mixers] = alc260_capture_mixer;
Kailang Yangdf694da2005-12-05 19:42:22 +01005120 }
Takashi Iwai4a471b72005-12-07 13:56:29 +01005121 spec->num_mixers++;
Kailang Yangdf694da2005-12-05 19:42:22 +01005122
5123 return 1;
5124}
5125
Takashi Iwaiae6b8132006-03-03 16:47:17 +01005126/* additional initialization for auto-configuration model */
5127static void alc260_auto_init(struct hda_codec *codec)
Kailang Yangdf694da2005-12-05 19:42:22 +01005128{
Takashi Iwaif6c7e542008-02-12 18:32:23 +01005129 struct alc_spec *spec = codec->spec;
Kailang Yangdf694da2005-12-05 19:42:22 +01005130 alc260_auto_init_multi_out(codec);
5131 alc260_auto_init_analog_input(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +01005132 if (spec->unsol_event)
5133 alc_sku_automute(codec);
Kailang Yangdf694da2005-12-05 19:42:22 +01005134}
5135
Takashi Iwaicb53c622007-08-10 17:21:45 +02005136#ifdef CONFIG_SND_HDA_POWER_SAVE
5137static struct hda_amp_list alc260_loopbacks[] = {
5138 { 0x07, HDA_INPUT, 0 },
5139 { 0x07, HDA_INPUT, 1 },
5140 { 0x07, HDA_INPUT, 2 },
5141 { 0x07, HDA_INPUT, 3 },
5142 { 0x07, HDA_INPUT, 4 },
5143 { } /* end */
5144};
5145#endif
5146
Kailang Yangdf694da2005-12-05 19:42:22 +01005147/*
5148 * ALC260 configurations
5149 */
Takashi Iwaif5fcc132006-11-24 17:07:44 +01005150static const char *alc260_models[ALC260_MODEL_LAST] = {
5151 [ALC260_BASIC] = "basic",
5152 [ALC260_HP] = "hp",
5153 [ALC260_HP_3013] = "hp-3013",
5154 [ALC260_FUJITSU_S702X] = "fujitsu",
5155 [ALC260_ACER] = "acer",
Kailang Yangbc9f98a2007-04-12 13:06:07 +02005156 [ALC260_WILL] = "will",
5157 [ALC260_REPLACER_672V] = "replacer",
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01005158#ifdef CONFIG_SND_DEBUG
Takashi Iwaif5fcc132006-11-24 17:07:44 +01005159 [ALC260_TEST] = "test",
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01005160#endif
Takashi Iwaif5fcc132006-11-24 17:07:44 +01005161 [ALC260_AUTO] = "auto",
5162};
5163
5164static struct snd_pci_quirk alc260_cfg_tbl[] = {
Jonathan Woithebd869482006-11-28 11:35:52 +01005165 SND_PCI_QUIRK(0x1025, 0x007b, "Acer C20x", ALC260_ACER),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01005166 SND_PCI_QUIRK(0x1025, 0x008f, "Acer", ALC260_ACER),
Takashi Iwai9720b712007-03-13 21:46:23 +01005167 SND_PCI_QUIRK(0x103c, 0x2808, "HP d5700", ALC260_HP_3013),
Takashi Iwaia8a5d062007-03-15 15:10:28 +01005168 SND_PCI_QUIRK(0x103c, 0x280a, "HP d5750", ALC260_HP_3013),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01005169 SND_PCI_QUIRK(0x103c, 0x3010, "HP", ALC260_HP_3013),
5170 SND_PCI_QUIRK(0x103c, 0x3011, "HP", ALC260_HP),
5171 SND_PCI_QUIRK(0x103c, 0x3012, "HP", ALC260_HP_3013),
5172 SND_PCI_QUIRK(0x103c, 0x3013, "HP", ALC260_HP_3013),
5173 SND_PCI_QUIRK(0x103c, 0x3014, "HP", ALC260_HP),
5174 SND_PCI_QUIRK(0x103c, 0x3015, "HP", ALC260_HP),
5175 SND_PCI_QUIRK(0x103c, 0x3016, "HP", ALC260_HP),
5176 SND_PCI_QUIRK(0x104d, 0x81bb, "Sony VAIO", ALC260_BASIC),
5177 SND_PCI_QUIRK(0x104d, 0x81cc, "Sony VAIO", ALC260_BASIC),
5178 SND_PCI_QUIRK(0x104d, 0x81cd, "Sony VAIO", ALC260_BASIC),
5179 SND_PCI_QUIRK(0x10cf, 0x1326, "Fujitsu S702X", ALC260_FUJITSU_S702X),
5180 SND_PCI_QUIRK(0x152d, 0x0729, "CTL U553W", ALC260_BASIC),
Kailang Yangbc9f98a2007-04-12 13:06:07 +02005181 SND_PCI_QUIRK(0x161f, 0x2057, "Replacer 672V", ALC260_REPLACER_672V),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01005182 SND_PCI_QUIRK(0x1631, 0xc017, "PB V7900", ALC260_WILL),
Takashi Iwai16ded522005-06-10 19:58:24 +02005183 {}
5184};
5185
Kailang Yangdf694da2005-12-05 19:42:22 +01005186static struct alc_config_preset alc260_presets[] = {
5187 [ALC260_BASIC] = {
5188 .mixers = { alc260_base_output_mixer,
5189 alc260_input_mixer,
5190 alc260_pc_beep_mixer,
5191 alc260_capture_mixer },
5192 .init_verbs = { alc260_init_verbs },
5193 .num_dacs = ARRAY_SIZE(alc260_dac_nids),
5194 .dac_nids = alc260_dac_nids,
5195 .num_adc_nids = ARRAY_SIZE(alc260_adc_nids),
5196 .adc_nids = alc260_adc_nids,
5197 .num_channel_mode = ARRAY_SIZE(alc260_modes),
5198 .channel_mode = alc260_modes,
5199 .input_mux = &alc260_capture_source,
5200 },
5201 [ALC260_HP] = {
Takashi Iwaibec15c32008-01-28 18:16:30 +01005202 .mixers = { alc260_hp_output_mixer,
Kailang Yangdf694da2005-12-05 19:42:22 +01005203 alc260_input_mixer,
5204 alc260_capture_alt_mixer },
Takashi Iwaibec15c32008-01-28 18:16:30 +01005205 .init_verbs = { alc260_init_verbs,
5206 alc260_hp_unsol_verbs },
Kailang Yangdf694da2005-12-05 19:42:22 +01005207 .num_dacs = ARRAY_SIZE(alc260_dac_nids),
5208 .dac_nids = alc260_dac_nids,
5209 .num_adc_nids = ARRAY_SIZE(alc260_hp_adc_nids),
5210 .adc_nids = alc260_hp_adc_nids,
5211 .num_channel_mode = ARRAY_SIZE(alc260_modes),
5212 .channel_mode = alc260_modes,
5213 .input_mux = &alc260_capture_source,
Takashi Iwaibec15c32008-01-28 18:16:30 +01005214 .unsol_event = alc260_hp_unsol_event,
5215 .init_hook = alc260_hp_automute,
Kailang Yangdf694da2005-12-05 19:42:22 +01005216 },
5217 [ALC260_HP_3013] = {
5218 .mixers = { alc260_hp_3013_mixer,
5219 alc260_input_mixer,
5220 alc260_capture_alt_mixer },
Takashi Iwaibec15c32008-01-28 18:16:30 +01005221 .init_verbs = { alc260_hp_3013_init_verbs,
5222 alc260_hp_3013_unsol_verbs },
Kailang Yangdf694da2005-12-05 19:42:22 +01005223 .num_dacs = ARRAY_SIZE(alc260_dac_nids),
5224 .dac_nids = alc260_dac_nids,
5225 .num_adc_nids = ARRAY_SIZE(alc260_hp_adc_nids),
5226 .adc_nids = alc260_hp_adc_nids,
5227 .num_channel_mode = ARRAY_SIZE(alc260_modes),
5228 .channel_mode = alc260_modes,
5229 .input_mux = &alc260_capture_source,
Takashi Iwaibec15c32008-01-28 18:16:30 +01005230 .unsol_event = alc260_hp_3013_unsol_event,
5231 .init_hook = alc260_hp_3013_automute,
Kailang Yangdf694da2005-12-05 19:42:22 +01005232 },
5233 [ALC260_FUJITSU_S702X] = {
5234 .mixers = { alc260_fujitsu_mixer,
5235 alc260_capture_mixer },
5236 .init_verbs = { alc260_fujitsu_init_verbs },
5237 .num_dacs = ARRAY_SIZE(alc260_dac_nids),
5238 .dac_nids = alc260_dac_nids,
Jonathan Woithed57fdac2006-02-28 11:38:35 +01005239 .num_adc_nids = ARRAY_SIZE(alc260_dual_adc_nids),
5240 .adc_nids = alc260_dual_adc_nids,
Kailang Yangdf694da2005-12-05 19:42:22 +01005241 .num_channel_mode = ARRAY_SIZE(alc260_modes),
5242 .channel_mode = alc260_modes,
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02005243 .num_mux_defs = ARRAY_SIZE(alc260_fujitsu_capture_sources),
5244 .input_mux = alc260_fujitsu_capture_sources,
Kailang Yangdf694da2005-12-05 19:42:22 +01005245 },
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01005246 [ALC260_ACER] = {
5247 .mixers = { alc260_acer_mixer,
5248 alc260_capture_mixer },
5249 .init_verbs = { alc260_acer_init_verbs },
5250 .num_dacs = ARRAY_SIZE(alc260_dac_nids),
5251 .dac_nids = alc260_dac_nids,
5252 .num_adc_nids = ARRAY_SIZE(alc260_dual_adc_nids),
5253 .adc_nids = alc260_dual_adc_nids,
5254 .num_channel_mode = ARRAY_SIZE(alc260_modes),
5255 .channel_mode = alc260_modes,
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02005256 .num_mux_defs = ARRAY_SIZE(alc260_acer_capture_sources),
5257 .input_mux = alc260_acer_capture_sources,
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01005258 },
Kailang Yangbc9f98a2007-04-12 13:06:07 +02005259 [ALC260_WILL] = {
5260 .mixers = { alc260_will_mixer,
5261 alc260_capture_mixer },
5262 .init_verbs = { alc260_init_verbs, alc260_will_verbs },
5263 .num_dacs = ARRAY_SIZE(alc260_dac_nids),
5264 .dac_nids = alc260_dac_nids,
5265 .num_adc_nids = ARRAY_SIZE(alc260_adc_nids),
5266 .adc_nids = alc260_adc_nids,
5267 .dig_out_nid = ALC260_DIGOUT_NID,
5268 .num_channel_mode = ARRAY_SIZE(alc260_modes),
5269 .channel_mode = alc260_modes,
5270 .input_mux = &alc260_capture_source,
5271 },
5272 [ALC260_REPLACER_672V] = {
5273 .mixers = { alc260_replacer_672v_mixer,
5274 alc260_capture_mixer },
5275 .init_verbs = { alc260_init_verbs, alc260_replacer_672v_verbs },
5276 .num_dacs = ARRAY_SIZE(alc260_dac_nids),
5277 .dac_nids = alc260_dac_nids,
5278 .num_adc_nids = ARRAY_SIZE(alc260_adc_nids),
5279 .adc_nids = alc260_adc_nids,
5280 .dig_out_nid = ALC260_DIGOUT_NID,
5281 .num_channel_mode = ARRAY_SIZE(alc260_modes),
5282 .channel_mode = alc260_modes,
5283 .input_mux = &alc260_capture_source,
5284 .unsol_event = alc260_replacer_672v_unsol_event,
5285 .init_hook = alc260_replacer_672v_automute,
5286 },
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01005287#ifdef CONFIG_SND_DEBUG
5288 [ALC260_TEST] = {
5289 .mixers = { alc260_test_mixer,
5290 alc260_capture_mixer },
5291 .init_verbs = { alc260_test_init_verbs },
5292 .num_dacs = ARRAY_SIZE(alc260_test_dac_nids),
5293 .dac_nids = alc260_test_dac_nids,
5294 .num_adc_nids = ARRAY_SIZE(alc260_test_adc_nids),
5295 .adc_nids = alc260_test_adc_nids,
5296 .num_channel_mode = ARRAY_SIZE(alc260_modes),
5297 .channel_mode = alc260_modes,
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02005298 .num_mux_defs = ARRAY_SIZE(alc260_test_capture_sources),
5299 .input_mux = alc260_test_capture_sources,
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01005300 },
5301#endif
Kailang Yangdf694da2005-12-05 19:42:22 +01005302};
5303
Linus Torvalds1da177e2005-04-16 15:20:36 -07005304static int patch_alc260(struct hda_codec *codec)
5305{
5306 struct alc_spec *spec;
Kailang Yangdf694da2005-12-05 19:42:22 +01005307 int err, board_config;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005308
Takashi Iwaie560d8d2005-09-09 14:21:46 +02005309 spec = kzalloc(sizeof(*spec), GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005310 if (spec == NULL)
5311 return -ENOMEM;
5312
5313 codec->spec = spec;
5314
Takashi Iwaif5fcc132006-11-24 17:07:44 +01005315 board_config = snd_hda_check_board_config(codec, ALC260_MODEL_LAST,
5316 alc260_models,
5317 alc260_cfg_tbl);
5318 if (board_config < 0) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +02005319 snd_printd(KERN_INFO "hda_codec: Unknown model for ALC260, "
5320 "trying auto-probe from BIOS...\n");
Kailang Yangdf694da2005-12-05 19:42:22 +01005321 board_config = ALC260_AUTO;
Takashi Iwai16ded522005-06-10 19:58:24 +02005322 }
5323
Kailang Yangdf694da2005-12-05 19:42:22 +01005324 if (board_config == ALC260_AUTO) {
5325 /* automatic parse from the BIOS config */
5326 err = alc260_parse_auto_config(codec);
5327 if (err < 0) {
5328 alc_free(codec);
5329 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02005330 } else if (!err) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +02005331 printk(KERN_INFO
5332 "hda_codec: Cannot set up configuration "
5333 "from BIOS. Using base mode...\n");
Kailang Yangdf694da2005-12-05 19:42:22 +01005334 board_config = ALC260_BASIC;
5335 }
Takashi Iwai16ded522005-06-10 19:58:24 +02005336 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07005337
Kailang Yangdf694da2005-12-05 19:42:22 +01005338 if (board_config != ALC260_AUTO)
5339 setup_preset(spec, &alc260_presets[board_config]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005340
5341 spec->stream_name_analog = "ALC260 Analog";
5342 spec->stream_analog_playback = &alc260_pcm_analog_playback;
5343 spec->stream_analog_capture = &alc260_pcm_analog_capture;
5344
Takashi Iwaia3bcba32005-12-06 19:05:29 +01005345 spec->stream_name_digital = "ALC260 Digital";
5346 spec->stream_digital_playback = &alc260_pcm_digital_playback;
5347 spec->stream_digital_capture = &alc260_pcm_digital_capture;
5348
Takashi Iwai2134ea42008-01-10 16:53:55 +01005349 spec->vmaster_nid = 0x08;
5350
Linus Torvalds1da177e2005-04-16 15:20:36 -07005351 codec->patch_ops = alc_patch_ops;
Kailang Yangdf694da2005-12-05 19:42:22 +01005352 if (board_config == ALC260_AUTO)
Takashi Iwaiae6b8132006-03-03 16:47:17 +01005353 spec->init_hook = alc260_auto_init;
Takashi Iwaicb53c622007-08-10 17:21:45 +02005354#ifdef CONFIG_SND_HDA_POWER_SAVE
5355 if (!spec->loopback.amplist)
5356 spec->loopback.amplist = alc260_loopbacks;
5357#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07005358
5359 return 0;
5360}
5361
Takashi Iwaie9edcee2005-06-13 14:16:38 +02005362
Linus Torvalds1da177e2005-04-16 15:20:36 -07005363/*
5364 * ALC882 support
5365 *
5366 * ALC882 is almost identical with ALC880 but has cleaner and more flexible
5367 * configuration. Each pin widget can choose any input DACs and a mixer.
5368 * Each ADC is connected from a mixer of all inputs. This makes possible
5369 * 6-channel independent captures.
5370 *
5371 * In addition, an independent DAC for the multi-playback (not used in this
5372 * driver yet).
5373 */
Kailang Yangdf694da2005-12-05 19:42:22 +01005374#define ALC882_DIGOUT_NID 0x06
5375#define ALC882_DIGIN_NID 0x0a
Linus Torvalds1da177e2005-04-16 15:20:36 -07005376
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +01005377static struct hda_channel_mode alc882_ch_modes[1] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07005378 { 8, NULL }
5379};
5380
5381static hda_nid_t alc882_dac_nids[4] = {
5382 /* front, rear, clfe, rear_surr */
5383 0x02, 0x03, 0x04, 0x05
5384};
5385
Kailang Yangdf694da2005-12-05 19:42:22 +01005386/* identical with ALC880 */
5387#define alc882_adc_nids alc880_adc_nids
5388#define alc882_adc_nids_alt alc880_adc_nids_alt
Linus Torvalds1da177e2005-04-16 15:20:36 -07005389
Takashi Iwaie1406342008-02-11 18:32:32 +01005390static hda_nid_t alc882_capsrc_nids[3] = { 0x24, 0x23, 0x22 };
5391static hda_nid_t alc882_capsrc_nids_alt[2] = { 0x23, 0x22 };
5392
Linus Torvalds1da177e2005-04-16 15:20:36 -07005393/* input MUX */
5394/* FIXME: should be a matrix-type input source selection */
5395
5396static struct hda_input_mux alc882_capture_source = {
5397 .num_items = 4,
5398 .items = {
5399 { "Mic", 0x0 },
5400 { "Front Mic", 0x1 },
5401 { "Line", 0x2 },
5402 { "CD", 0x4 },
5403 },
5404};
Linus Torvalds1da177e2005-04-16 15:20:36 -07005405#define alc882_mux_enum_info alc_mux_enum_info
5406#define alc882_mux_enum_get alc_mux_enum_get
5407
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02005408static int alc882_mux_enum_put(struct snd_kcontrol *kcontrol,
5409 struct snd_ctl_elem_value *ucontrol)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005410{
5411 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
5412 struct alc_spec *spec = codec->spec;
5413 const struct hda_input_mux *imux = spec->input_mux;
5414 unsigned int adc_idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
Takashi Iwai88c71a92008-02-14 17:27:17 +01005415 hda_nid_t nid = spec->capsrc_nids ?
5416 spec->capsrc_nids[adc_idx] : spec->adc_nids[adc_idx];
Linus Torvalds1da177e2005-04-16 15:20:36 -07005417 unsigned int *cur_val = &spec->cur_mux[adc_idx];
5418 unsigned int i, idx;
5419
5420 idx = ucontrol->value.enumerated.item[0];
5421 if (idx >= imux->num_items)
5422 idx = imux->num_items - 1;
Takashi Iwai82beb8f2007-08-10 17:09:26 +02005423 if (*cur_val == idx)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005424 return 0;
5425 for (i = 0; i < imux->num_items; i++) {
Takashi Iwai47fd8302007-08-10 17:11:07 +02005426 unsigned int v = (i == idx) ? 0 : HDA_AMP_MUTE;
5427 snd_hda_codec_amp_stereo(codec, nid, HDA_INPUT,
Takashi Iwai82beb8f2007-08-10 17:09:26 +02005428 imux->items[i].index,
Takashi Iwai47fd8302007-08-10 17:11:07 +02005429 HDA_AMP_MUTE, v);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005430 }
5431 *cur_val = idx;
5432 return 1;
5433}
5434
Kailang Yangdf694da2005-12-05 19:42:22 +01005435/*
Kailang Yang272a5272007-05-14 11:00:38 +02005436 * 2ch mode
5437 */
5438static struct hda_verb alc882_3ST_ch2_init[] = {
5439 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
5440 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
5441 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
5442 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
5443 { } /* end */
5444};
5445
5446/*
5447 * 6ch mode
5448 */
5449static struct hda_verb alc882_3ST_ch6_init[] = {
5450 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
5451 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
5452 { 0x18, AC_VERB_SET_CONNECT_SEL, 0x02 },
5453 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
5454 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
5455 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x01 },
5456 { } /* end */
5457};
5458
5459static struct hda_channel_mode alc882_3ST_6ch_modes[2] = {
5460 { 2, alc882_3ST_ch2_init },
5461 { 6, alc882_3ST_ch6_init },
5462};
5463
5464/*
Kailang Yangdf694da2005-12-05 19:42:22 +01005465 * 6ch mode
5466 */
5467static struct hda_verb alc882_sixstack_ch6_init[] = {
5468 { 0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
5469 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
5470 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
5471 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
5472 { } /* end */
5473};
5474
5475/*
5476 * 8ch mode
5477 */
5478static struct hda_verb alc882_sixstack_ch8_init[] = {
5479 { 0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
5480 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
5481 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
5482 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
5483 { } /* end */
5484};
5485
5486static struct hda_channel_mode alc882_sixstack_modes[2] = {
5487 { 6, alc882_sixstack_ch6_init },
5488 { 8, alc882_sixstack_ch8_init },
5489};
5490
Takashi Iwai87350ad2007-08-16 18:19:38 +02005491/*
5492 * macbook pro ALC885 can switch LineIn to LineOut without loosing Mic
5493 */
5494
5495/*
5496 * 2ch mode
5497 */
5498static struct hda_verb alc885_mbp_ch2_init[] = {
5499 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
5500 { 0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5501 { 0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5502 { } /* end */
5503};
5504
5505/*
5506 * 6ch mode
5507 */
5508static struct hda_verb alc885_mbp_ch6_init[] = {
5509 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
5510 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5511 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x01 },
5512 { 0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5513 { 0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
5514 { } /* end */
5515};
5516
5517static struct hda_channel_mode alc885_mbp_6ch_modes[2] = {
5518 { 2, alc885_mbp_ch2_init },
5519 { 6, alc885_mbp_ch6_init },
5520};
5521
5522
Linus Torvalds1da177e2005-04-16 15:20:36 -07005523/* Pin assignment: Front=0x14, Rear=0x15, CLFE=0x16, Side=0x17
5524 * Mic=0x18, Front Mic=0x19, Line-In=0x1a, HP=0x1b
5525 */
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01005526static struct snd_kcontrol_new alc882_base_mixer[] = {
Takashi Iwai05acb862005-06-10 19:50:25 +02005527 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01005528 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Takashi Iwai05acb862005-06-10 19:50:25 +02005529 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01005530 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
Takashi Iwai05acb862005-06-10 19:50:25 +02005531 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
5532 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01005533 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
5534 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
Takashi Iwai05acb862005-06-10 19:50:25 +02005535 HDA_CODEC_VOLUME("Side Playback Volume", 0x0f, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01005536 HDA_BIND_MUTE("Side Playback Switch", 0x0f, 2, HDA_INPUT),
Linus Torvalds1da177e2005-04-16 15:20:36 -07005537 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
5538 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
5539 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
5540 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
5541 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
5542 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01005543 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Linus Torvalds1da177e2005-04-16 15:20:36 -07005544 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
5545 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01005546 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
Linus Torvalds1da177e2005-04-16 15:20:36 -07005547 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
5548 HDA_CODEC_VOLUME("PC Speaker Playback Volume", 0x0b, 0x05, HDA_INPUT),
5549 HDA_CODEC_MUTE("PC Speaker Playback Switch", 0x0b, 0x05, HDA_INPUT),
Linus Torvalds1da177e2005-04-16 15:20:36 -07005550 { } /* end */
5551};
5552
Takashi Iwai87350ad2007-08-16 18:19:38 +02005553static struct snd_kcontrol_new alc885_mbp3_mixer[] = {
Takashi Iwai2134ea42008-01-10 16:53:55 +01005554 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x00, HDA_OUTPUT),
5555 HDA_BIND_MUTE ("Front Playback Switch", 0x0c, 0x02, HDA_INPUT),
5556 HDA_CODEC_MUTE ("Speaker Playback Switch", 0x14, 0x00, HDA_OUTPUT),
5557 HDA_CODEC_VOLUME("Line-Out Playback Volume", 0x0d, 0x00, HDA_OUTPUT),
5558 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
5559 HDA_CODEC_MUTE ("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
Takashi Iwai87350ad2007-08-16 18:19:38 +02005560 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x00, HDA_INPUT),
5561 HDA_CODEC_MUTE ("Mic Playback Switch", 0x0b, 0x00, HDA_INPUT),
Takashi Iwai2134ea42008-01-10 16:53:55 +01005562 HDA_CODEC_VOLUME("Line Boost", 0x1a, 0x00, HDA_INPUT),
Takashi Iwai87350ad2007-08-16 18:19:38 +02005563 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0x00, HDA_INPUT),
5564 { } /* end */
5565};
Kailang Yangbdd148a2007-05-08 15:19:08 +02005566static struct snd_kcontrol_new alc882_w2jc_mixer[] = {
5567 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
5568 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
5569 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
5570 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
5571 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
5572 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
5573 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
5574 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
5575 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
5576 HDA_CODEC_VOLUME("PC Speaker Playback Volume", 0x0b, 0x05, HDA_INPUT),
5577 HDA_CODEC_MUTE("PC Speaker Playback Switch", 0x0b, 0x05, HDA_INPUT),
5578 { } /* end */
5579};
5580
Kailang Yang272a5272007-05-14 11:00:38 +02005581static struct snd_kcontrol_new alc882_targa_mixer[] = {
5582 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
5583 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
5584 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
5585 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
5586 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
5587 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
5588 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
5589 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
5590 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Takashi Iwai96fe7cc2007-09-07 10:57:44 +02005591 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Kailang Yang272a5272007-05-14 11:00:38 +02005592 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
5593 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Takashi Iwai96fe7cc2007-09-07 10:57:44 +02005594 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
Kailang Yang272a5272007-05-14 11:00:38 +02005595 { } /* end */
5596};
5597
5598/* Pin assignment: Front=0x14, HP = 0x15, Front = 0x16, ???
5599 * Front Mic=0x18, Line In = 0x1a, Line In = 0x1b, CD = 0x1c
5600 */
5601static struct snd_kcontrol_new alc882_asus_a7j_mixer[] = {
5602 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
5603 HDA_CODEC_MUTE("Front Playback Switch", 0x14, 0x0, HDA_OUTPUT),
5604 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
5605 HDA_CODEC_MUTE("Mobile Front Playback Switch", 0x16, 0x0, HDA_OUTPUT),
5606 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
5607 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
5608 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
5609 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
5610 HDA_CODEC_VOLUME("Mobile Line Playback Volume", 0x0b, 0x03, HDA_INPUT),
5611 HDA_CODEC_MUTE("Mobile Line Playback Switch", 0x0b, 0x03, HDA_INPUT),
5612 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
5613 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Takashi Iwai96fe7cc2007-09-07 10:57:44 +02005614 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Kailang Yang272a5272007-05-14 11:00:38 +02005615 { } /* end */
5616};
5617
Takashi Iwai914759b2007-09-06 14:52:04 +02005618static struct snd_kcontrol_new alc882_asus_a7m_mixer[] = {
5619 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
5620 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
5621 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
5622 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
5623 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
5624 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
5625 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
5626 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
5627 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
5628 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
5629 HDA_CODEC_VOLUME("PC Speaker Playback Volume", 0x0b, 0x05, HDA_INPUT),
5630 HDA_CODEC_MUTE("PC Speaker Playback Switch", 0x0b, 0x05, HDA_INPUT),
5631 { } /* end */
5632};
5633
Kailang Yangdf694da2005-12-05 19:42:22 +01005634static struct snd_kcontrol_new alc882_chmode_mixer[] = {
5635 {
5636 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
5637 .name = "Channel Mode",
5638 .info = alc_ch_mode_info,
5639 .get = alc_ch_mode_get,
5640 .put = alc_ch_mode_put,
5641 },
5642 { } /* end */
5643};
5644
Linus Torvalds1da177e2005-04-16 15:20:36 -07005645static struct hda_verb alc882_init_verbs[] = {
5646 /* Front mixer: unmute input/output amp left and right (volume = 0) */
Takashi Iwai05acb862005-06-10 19:50:25 +02005647 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5648 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5649 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005650 /* Rear mixer */
Takashi Iwai05acb862005-06-10 19:50:25 +02005651 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5652 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5653 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005654 /* CLFE mixer */
Takashi Iwai05acb862005-06-10 19:50:25 +02005655 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5656 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5657 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005658 /* Side mixer */
Takashi Iwai05acb862005-06-10 19:50:25 +02005659 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5660 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5661 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005662
Takashi Iwaie9edcee2005-06-13 14:16:38 +02005663 /* Front Pin: output 0 (0x0c) */
Takashi Iwai05acb862005-06-10 19:50:25 +02005664 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwai05acb862005-06-10 19:50:25 +02005665 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005666 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02005667 /* Rear Pin: output 1 (0x0d) */
Takashi Iwai05acb862005-06-10 19:50:25 +02005668 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwai05acb862005-06-10 19:50:25 +02005669 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005670 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02005671 /* CLFE Pin: output 2 (0x0e) */
Takashi Iwai05acb862005-06-10 19:50:25 +02005672 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwai05acb862005-06-10 19:50:25 +02005673 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005674 {0x16, AC_VERB_SET_CONNECT_SEL, 0x02},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02005675 /* Side Pin: output 3 (0x0f) */
Takashi Iwai05acb862005-06-10 19:50:25 +02005676 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwai05acb862005-06-10 19:50:25 +02005677 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005678 {0x17, AC_VERB_SET_CONNECT_SEL, 0x03},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02005679 /* Mic (rear) pin: input vref at 80% */
Takashi Iwai16ded522005-06-10 19:58:24 +02005680 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02005681 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
5682 /* Front Mic pin: input vref at 80% */
Takashi Iwai16ded522005-06-10 19:58:24 +02005683 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02005684 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
5685 /* Line In pin: input */
Takashi Iwai05acb862005-06-10 19:50:25 +02005686 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02005687 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
5688 /* Line-2 In: Headphone output (output 0 - 0x0c) */
5689 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
5690 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5691 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005692 /* CD pin widget for input */
Takashi Iwai05acb862005-06-10 19:50:25 +02005693 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005694
5695 /* FIXME: use matrix-type input source selection */
5696 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
5697 /* Input mixer1: unmute Mic, F-Mic, Line, CD inputs */
Takashi Iwai05acb862005-06-10 19:50:25 +02005698 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5699 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
5700 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
5701 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005702 /* Input mixer2 */
Takashi Iwai05acb862005-06-10 19:50:25 +02005703 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5704 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
5705 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
5706 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005707 /* Input mixer3 */
Takashi Iwai05acb862005-06-10 19:50:25 +02005708 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5709 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
5710 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
5711 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
5712 /* ADC1: mute amp left and right */
5713 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Takashi Iwai71fe7b82005-05-25 18:11:40 +02005714 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
Takashi Iwai05acb862005-06-10 19:50:25 +02005715 /* ADC2: mute amp left and right */
5716 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Takashi Iwai71fe7b82005-05-25 18:11:40 +02005717 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
Takashi Iwai05acb862005-06-10 19:50:25 +02005718 /* ADC3: mute amp left and right */
5719 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Takashi Iwai71fe7b82005-05-25 18:11:40 +02005720 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005721
5722 { }
5723};
5724
Takashi Iwai4b146cb2006-07-28 14:42:36 +02005725static struct hda_verb alc882_eapd_verbs[] = {
5726 /* change to EAPD mode */
5727 {0x20, AC_VERB_SET_COEF_INDEX, 0x07},
Andrew L. Neporadab373bde2006-11-07 11:37:08 +01005728 {0x20, AC_VERB_SET_PROC_COEF, 0x3060},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02005729 { }
Takashi Iwai4b146cb2006-07-28 14:42:36 +02005730};
5731
Tobin Davis9102cd12006-12-15 10:02:12 +01005732/* Mac Pro test */
5733static struct snd_kcontrol_new alc882_macpro_mixer[] = {
5734 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
5735 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
5736 HDA_CODEC_MUTE("Headphone Playback Switch", 0x18, 0x0, HDA_OUTPUT),
5737 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x01, HDA_INPUT),
5738 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x01, HDA_INPUT),
5739 HDA_CODEC_VOLUME("PC Speaker Playback Volume", 0x0b, 0x02, HDA_INPUT),
5740 HDA_CODEC_MUTE("PC Speaker Playback Switch", 0x0b, 0x02, HDA_INPUT),
5741 { } /* end */
5742};
5743
5744static struct hda_verb alc882_macpro_init_verbs[] = {
5745 /* Front mixer: unmute input/output amp left and right (volume = 0) */
5746 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5747 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5748 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5749 /* Front Pin: output 0 (0x0c) */
5750 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5751 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5752 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
5753 /* Front Mic pin: input vref at 80% */
5754 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
5755 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
5756 /* Speaker: output */
5757 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5758 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5759 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x04},
5760 /* Headphone output (output 0 - 0x0c) */
5761 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
5762 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5763 {0x18, AC_VERB_SET_CONNECT_SEL, 0x00},
5764
5765 /* FIXME: use matrix-type input source selection */
5766 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
5767 /* Input mixer1: unmute Mic, F-Mic, Line, CD inputs */
5768 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5769 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
5770 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
5771 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
5772 /* Input mixer2 */
5773 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5774 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
5775 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
5776 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
5777 /* Input mixer3 */
5778 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5779 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
5780 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
5781 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
5782 /* ADC1: mute amp left and right */
5783 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5784 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
5785 /* ADC2: mute amp left and right */
5786 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5787 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
5788 /* ADC3: mute amp left and right */
5789 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5790 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
5791
5792 { }
5793};
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02005794
Takashi Iwai87350ad2007-08-16 18:19:38 +02005795/* Macbook Pro rev3 */
5796static struct hda_verb alc885_mbp3_init_verbs[] = {
5797 /* Front mixer: unmute input/output amp left and right (volume = 0) */
5798 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5799 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5800 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5801 /* Rear mixer */
5802 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5803 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5804 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5805 /* Front Pin: output 0 (0x0c) */
5806 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5807 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5808 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
5809 /* HP Pin: output 0 (0x0d) */
5810 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc4},
5811 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
5812 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
5813 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
5814 /* Mic (rear) pin: input vref at 80% */
5815 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
5816 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
5817 /* Front Mic pin: input vref at 80% */
5818 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
5819 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
5820 /* Line In pin: use output 1 when in LineOut mode */
5821 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
5822 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
5823 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x01},
5824
5825 /* FIXME: use matrix-type input source selection */
5826 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
5827 /* Input mixer1: unmute Mic, F-Mic, Line, CD inputs */
5828 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5829 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
5830 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
5831 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
5832 /* Input mixer2 */
5833 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5834 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
5835 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
5836 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
5837 /* Input mixer3 */
5838 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5839 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
5840 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
5841 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
5842 /* ADC1: mute amp left and right */
5843 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5844 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
5845 /* ADC2: mute amp left and right */
5846 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5847 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
5848 /* ADC3: mute amp left and right */
5849 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5850 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
5851
5852 { }
5853};
5854
Nicola Fagnanic54728d2007-07-19 23:28:52 +02005855/* iMac 24 mixer. */
5856static struct snd_kcontrol_new alc885_imac24_mixer[] = {
5857 HDA_CODEC_VOLUME("Master Playback Volume", 0x0c, 0x00, HDA_OUTPUT),
5858 HDA_CODEC_MUTE("Master Playback Switch", 0x0c, 0x00, HDA_INPUT),
5859 { } /* end */
5860};
5861
5862/* iMac 24 init verbs. */
5863static struct hda_verb alc885_imac24_init_verbs[] = {
5864 /* Internal speakers: output 0 (0x0c) */
5865 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5866 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5867 {0x18, AC_VERB_SET_CONNECT_SEL, 0x00},
5868 /* Internal speakers: output 0 (0x0c) */
5869 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5870 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5871 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x00},
5872 /* Headphone: output 0 (0x0c) */
5873 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
5874 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5875 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
5876 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
5877 /* Front Mic: input vref at 80% */
5878 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
5879 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
5880 { }
5881};
5882
5883/* Toggle speaker-output according to the hp-jack state */
5884static void alc885_imac24_automute(struct hda_codec *codec)
5885{
5886 unsigned int present;
5887
5888 present = snd_hda_codec_read(codec, 0x14, 0,
5889 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +02005890 snd_hda_codec_amp_stereo(codec, 0x18, HDA_OUTPUT, 0,
5891 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
5892 snd_hda_codec_amp_stereo(codec, 0x1a, HDA_OUTPUT, 0,
5893 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
Nicola Fagnanic54728d2007-07-19 23:28:52 +02005894}
5895
5896/* Processes unsolicited events. */
5897static void alc885_imac24_unsol_event(struct hda_codec *codec,
5898 unsigned int res)
5899{
5900 /* Headphone insertion or removal. */
5901 if ((res >> 26) == ALC880_HP_EVENT)
5902 alc885_imac24_automute(codec);
5903}
5904
Takashi Iwai87350ad2007-08-16 18:19:38 +02005905static void alc885_mbp3_automute(struct hda_codec *codec)
5906{
5907 unsigned int present;
5908
5909 present = snd_hda_codec_read(codec, 0x15, 0,
5910 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
5911 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
5912 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
5913 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
5914 HDA_AMP_MUTE, present ? 0 : HDA_AMP_MUTE);
5915
5916}
5917static void alc885_mbp3_unsol_event(struct hda_codec *codec,
5918 unsigned int res)
5919{
5920 /* Headphone insertion or removal. */
5921 if ((res >> 26) == ALC880_HP_EVENT)
5922 alc885_mbp3_automute(codec);
5923}
5924
5925
Kailang Yang272a5272007-05-14 11:00:38 +02005926static struct hda_verb alc882_targa_verbs[] = {
5927 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5928 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
5929
5930 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
5931 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5932
5933 {0x18, AC_VERB_SET_CONNECT_SEL, 0x02}, /* mic/clfe */
5934 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x01}, /* line/surround */
5935 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
5936
5937 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
5938 {0x01, AC_VERB_SET_GPIO_MASK, 0x03},
5939 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x03},
5940 {0x01, AC_VERB_SET_GPIO_DATA, 0x03},
5941 { } /* end */
5942};
5943
5944/* toggle speaker-output according to the hp-jack state */
5945static void alc882_targa_automute(struct hda_codec *codec)
5946{
5947 unsigned int present;
5948
5949 present = snd_hda_codec_read(codec, 0x14, 0,
5950 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +02005951 snd_hda_codec_amp_stereo(codec, 0x1b, HDA_OUTPUT, 0,
5952 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
Takashi Iwai82beb8f2007-08-10 17:09:26 +02005953 snd_hda_codec_write_cache(codec, 1, 0, AC_VERB_SET_GPIO_DATA,
5954 present ? 1 : 3);
Kailang Yang272a5272007-05-14 11:00:38 +02005955}
5956
5957static void alc882_targa_unsol_event(struct hda_codec *codec, unsigned int res)
5958{
5959 /* Looks like the unsol event is incompatible with the standard
5960 * definition. 4bit tag is placed at 26 bit!
5961 */
5962 if (((res >> 26) == ALC880_HP_EVENT)) {
5963 alc882_targa_automute(codec);
5964 }
5965}
5966
5967static struct hda_verb alc882_asus_a7j_verbs[] = {
5968 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5969 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
5970
5971 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
5972 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5973 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5974
5975 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00}, /* Front */
5976 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
5977 {0x16, AC_VERB_SET_CONNECT_SEL, 0x00}, /* Front */
5978
5979 {0x18, AC_VERB_SET_CONNECT_SEL, 0x02}, /* mic/clfe */
5980 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x01}, /* line/surround */
5981 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
5982 { } /* end */
5983};
5984
Takashi Iwai914759b2007-09-06 14:52:04 +02005985static struct hda_verb alc882_asus_a7m_verbs[] = {
5986 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5987 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
5988
5989 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
5990 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5991 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5992
5993 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00}, /* Front */
5994 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
5995 {0x16, AC_VERB_SET_CONNECT_SEL, 0x00}, /* Front */
5996
5997 {0x18, AC_VERB_SET_CONNECT_SEL, 0x02}, /* mic/clfe */
5998 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x01}, /* line/surround */
5999 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
6000 { } /* end */
6001};
6002
Tobin Davis9102cd12006-12-15 10:02:12 +01006003static void alc882_gpio_mute(struct hda_codec *codec, int pin, int muted)
6004{
6005 unsigned int gpiostate, gpiomask, gpiodir;
6006
6007 gpiostate = snd_hda_codec_read(codec, codec->afg, 0,
6008 AC_VERB_GET_GPIO_DATA, 0);
6009
6010 if (!muted)
6011 gpiostate |= (1 << pin);
6012 else
6013 gpiostate &= ~(1 << pin);
6014
6015 gpiomask = snd_hda_codec_read(codec, codec->afg, 0,
6016 AC_VERB_GET_GPIO_MASK, 0);
6017 gpiomask |= (1 << pin);
6018
6019 gpiodir = snd_hda_codec_read(codec, codec->afg, 0,
6020 AC_VERB_GET_GPIO_DIRECTION, 0);
6021 gpiodir |= (1 << pin);
6022
6023
6024 snd_hda_codec_write(codec, codec->afg, 0,
6025 AC_VERB_SET_GPIO_MASK, gpiomask);
6026 snd_hda_codec_write(codec, codec->afg, 0,
6027 AC_VERB_SET_GPIO_DIRECTION, gpiodir);
6028
6029 msleep(1);
6030
6031 snd_hda_codec_write(codec, codec->afg, 0,
6032 AC_VERB_SET_GPIO_DATA, gpiostate);
6033}
6034
Takashi Iwai7debbe52007-08-16 15:01:03 +02006035/* set up GPIO at initialization */
6036static void alc885_macpro_init_hook(struct hda_codec *codec)
6037{
6038 alc882_gpio_mute(codec, 0, 0);
6039 alc882_gpio_mute(codec, 1, 0);
6040}
6041
6042/* set up GPIO and update auto-muting at initialization */
6043static void alc885_imac24_init_hook(struct hda_codec *codec)
6044{
6045 alc885_macpro_init_hook(codec);
6046 alc885_imac24_automute(codec);
6047}
6048
Kailang Yangdf694da2005-12-05 19:42:22 +01006049/*
6050 * generic initialization of ADC, input mixers and output mixers
6051 */
6052static struct hda_verb alc882_auto_init_verbs[] = {
6053 /*
6054 * Unmute ADC0-2 and set the default input to mic-in
6055 */
6056 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
6057 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
6058 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
6059 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
6060 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
6061 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
6062
Takashi Iwaicb53c622007-08-10 17:21:45 +02006063 /* Mute input amps (CD, Line In, Mic 1 & Mic 2) of the analog-loopback
Kailang Yangdf694da2005-12-05 19:42:22 +01006064 * mixer widget
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006065 * Note: PASD motherboards uses the Line In 2 as the input for
6066 * front panel mic (mic 2)
Kailang Yangdf694da2005-12-05 19:42:22 +01006067 */
6068 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +02006069 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
6070 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
6071 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
6072 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
6073 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Kailang Yangdf694da2005-12-05 19:42:22 +01006074
6075 /*
6076 * Set up output mixers (0x0c - 0x0f)
6077 */
6078 /* set vol=0 to output mixers */
6079 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
6080 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
6081 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
6082 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
6083 /* set up input amps for analog loopback */
6084 /* Amp Indices: DAC = 0, mixer = 1 */
6085 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
6086 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
6087 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
6088 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
6089 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
6090 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
6091 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
6092 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
6093 {0x26, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
6094 {0x26, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
6095
6096 /* FIXME: use matrix-type input source selection */
6097 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
6098 /* Input mixer1: unmute Mic, F-Mic, Line, CD inputs */
6099 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
6100 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
6101 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
6102 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
6103 /* Input mixer2 */
6104 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
6105 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
6106 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
6107 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
6108 /* Input mixer3 */
6109 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
6110 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
6111 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
6112 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
6113
6114 { }
6115};
6116
6117/* capture mixer elements */
6118static struct snd_kcontrol_new alc882_capture_alt_mixer[] = {
6119 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
6120 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
6121 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x09, 0x0, HDA_INPUT),
6122 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x09, 0x0, HDA_INPUT),
6123 {
6124 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
6125 /* The multiple "Capture Source" controls confuse alsamixer
6126 * So call somewhat different..
Kailang Yangdf694da2005-12-05 19:42:22 +01006127 */
6128 /* .name = "Capture Source", */
6129 .name = "Input Source",
6130 .count = 2,
6131 .info = alc882_mux_enum_info,
6132 .get = alc882_mux_enum_get,
6133 .put = alc882_mux_enum_put,
6134 },
6135 { } /* end */
6136};
6137
6138static struct snd_kcontrol_new alc882_capture_mixer[] = {
6139 HDA_CODEC_VOLUME("Capture Volume", 0x07, 0x0, HDA_INPUT),
6140 HDA_CODEC_MUTE("Capture Switch", 0x07, 0x0, HDA_INPUT),
6141 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x08, 0x0, HDA_INPUT),
6142 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x08, 0x0, HDA_INPUT),
6143 HDA_CODEC_VOLUME_IDX("Capture Volume", 2, 0x09, 0x0, HDA_INPUT),
6144 HDA_CODEC_MUTE_IDX("Capture Switch", 2, 0x09, 0x0, HDA_INPUT),
6145 {
6146 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
6147 /* The multiple "Capture Source" controls confuse alsamixer
6148 * So call somewhat different..
Kailang Yangdf694da2005-12-05 19:42:22 +01006149 */
6150 /* .name = "Capture Source", */
6151 .name = "Input Source",
6152 .count = 3,
6153 .info = alc882_mux_enum_info,
6154 .get = alc882_mux_enum_get,
6155 .put = alc882_mux_enum_put,
6156 },
6157 { } /* end */
6158};
6159
Takashi Iwaicb53c622007-08-10 17:21:45 +02006160#ifdef CONFIG_SND_HDA_POWER_SAVE
6161#define alc882_loopbacks alc880_loopbacks
6162#endif
6163
Kailang Yangdf694da2005-12-05 19:42:22 +01006164/* pcm configuration: identiacal with ALC880 */
6165#define alc882_pcm_analog_playback alc880_pcm_analog_playback
6166#define alc882_pcm_analog_capture alc880_pcm_analog_capture
6167#define alc882_pcm_digital_playback alc880_pcm_digital_playback
6168#define alc882_pcm_digital_capture alc880_pcm_digital_capture
6169
6170/*
6171 * configuration and preset
6172 */
Takashi Iwaif5fcc132006-11-24 17:07:44 +01006173static const char *alc882_models[ALC882_MODEL_LAST] = {
6174 [ALC882_3ST_DIG] = "3stack-dig",
6175 [ALC882_6ST_DIG] = "6stack-dig",
6176 [ALC882_ARIMA] = "arima",
Kailang Yangbdd148a2007-05-08 15:19:08 +02006177 [ALC882_W2JC] = "w2jc",
Takashi Iwai0438a002007-09-06 14:54:11 +02006178 [ALC882_TARGA] = "targa",
6179 [ALC882_ASUS_A7J] = "asus-a7j",
6180 [ALC882_ASUS_A7M] = "asus-a7m",
Tobin Davis9102cd12006-12-15 10:02:12 +01006181 [ALC885_MACPRO] = "macpro",
Takashi Iwai87350ad2007-08-16 18:19:38 +02006182 [ALC885_MBP3] = "mbp3",
Nicola Fagnanic54728d2007-07-19 23:28:52 +02006183 [ALC885_IMAC24] = "imac24",
Takashi Iwaif5fcc132006-11-24 17:07:44 +01006184 [ALC882_AUTO] = "auto",
6185};
6186
6187static struct snd_pci_quirk alc882_cfg_tbl[] = {
6188 SND_PCI_QUIRK(0x1019, 0x6668, "ECS", ALC882_6ST_DIG),
Kailang Yang272a5272007-05-14 11:00:38 +02006189 SND_PCI_QUIRK(0x1043, 0x060d, "Asus A7J", ALC882_ASUS_A7J),
Kailang Yangac8842a2007-09-20 12:51:39 +02006190 SND_PCI_QUIRK(0x1043, 0x1243, "Asus A7J", ALC882_ASUS_A7J),
Takashi Iwai914759b2007-09-06 14:52:04 +02006191 SND_PCI_QUIRK(0x1043, 0x13c2, "Asus A7M", ALC882_ASUS_A7M),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01006192 SND_PCI_QUIRK(0x1043, 0x1971, "Asus W2JC", ALC882_W2JC),
Claudio Matsuokac5d9f1c2007-07-19 23:18:32 +02006193 SND_PCI_QUIRK(0x1043, 0x817f, "Asus P5LD2", ALC882_6ST_DIG),
Tobin Davis7b9470d2006-12-28 13:56:48 +01006194 SND_PCI_QUIRK(0x1043, 0x81d8, "Asus P5WD", ALC882_6ST_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01006195 SND_PCI_QUIRK(0x105b, 0x6668, "Foxconn", ALC882_6ST_DIG),
Travis Place3e0e4692008-06-20 16:51:45 +02006196 SND_PCI_QUIRK(0x106b, 0x00a0, "Apple iMac 24''", ALC885_IMAC24),
Jiang zhe44447042008-01-28 12:28:24 +01006197 SND_PCI_QUIRK(0x1458, 0xa002, "Gigabyte P35 DS3R", ALC882_6ST_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01006198 SND_PCI_QUIRK(0x1462, 0x28fb, "Targa T8", ALC882_TARGA), /* MSI-1049 T8 */
6199 SND_PCI_QUIRK(0x1462, 0x6668, "MSI", ALC882_6ST_DIG),
6200 SND_PCI_QUIRK(0x161f, 0x2054, "Arima W820", ALC882_ARIMA),
Kailang Yangdf694da2005-12-05 19:42:22 +01006201 {}
6202};
6203
6204static struct alc_config_preset alc882_presets[] = {
6205 [ALC882_3ST_DIG] = {
6206 .mixers = { alc882_base_mixer },
6207 .init_verbs = { alc882_init_verbs },
6208 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
6209 .dac_nids = alc882_dac_nids,
6210 .dig_out_nid = ALC882_DIGOUT_NID,
Kailang Yangdf694da2005-12-05 19:42:22 +01006211 .dig_in_nid = ALC882_DIGIN_NID,
6212 .num_channel_mode = ARRAY_SIZE(alc882_ch_modes),
6213 .channel_mode = alc882_ch_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02006214 .need_dac_fix = 1,
Kailang Yangdf694da2005-12-05 19:42:22 +01006215 .input_mux = &alc882_capture_source,
6216 },
6217 [ALC882_6ST_DIG] = {
6218 .mixers = { alc882_base_mixer, alc882_chmode_mixer },
6219 .init_verbs = { alc882_init_verbs },
6220 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
6221 .dac_nids = alc882_dac_nids,
6222 .dig_out_nid = ALC882_DIGOUT_NID,
Kailang Yangdf694da2005-12-05 19:42:22 +01006223 .dig_in_nid = ALC882_DIGIN_NID,
6224 .num_channel_mode = ARRAY_SIZE(alc882_sixstack_modes),
6225 .channel_mode = alc882_sixstack_modes,
6226 .input_mux = &alc882_capture_source,
6227 },
Takashi Iwai4b146cb2006-07-28 14:42:36 +02006228 [ALC882_ARIMA] = {
6229 .mixers = { alc882_base_mixer, alc882_chmode_mixer },
6230 .init_verbs = { alc882_init_verbs, alc882_eapd_verbs },
6231 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
6232 .dac_nids = alc882_dac_nids,
6233 .num_channel_mode = ARRAY_SIZE(alc882_sixstack_modes),
6234 .channel_mode = alc882_sixstack_modes,
6235 .input_mux = &alc882_capture_source,
6236 },
Kailang Yangbdd148a2007-05-08 15:19:08 +02006237 [ALC882_W2JC] = {
6238 .mixers = { alc882_w2jc_mixer, alc882_chmode_mixer },
6239 .init_verbs = { alc882_init_verbs, alc882_eapd_verbs,
6240 alc880_gpio1_init_verbs },
6241 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
6242 .dac_nids = alc882_dac_nids,
6243 .num_channel_mode = ARRAY_SIZE(alc880_threestack_modes),
6244 .channel_mode = alc880_threestack_modes,
6245 .need_dac_fix = 1,
6246 .input_mux = &alc882_capture_source,
6247 .dig_out_nid = ALC882_DIGOUT_NID,
6248 },
Takashi Iwai87350ad2007-08-16 18:19:38 +02006249 [ALC885_MBP3] = {
6250 .mixers = { alc885_mbp3_mixer, alc882_chmode_mixer },
6251 .init_verbs = { alc885_mbp3_init_verbs,
6252 alc880_gpio1_init_verbs },
6253 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
6254 .dac_nids = alc882_dac_nids,
6255 .channel_mode = alc885_mbp_6ch_modes,
6256 .num_channel_mode = ARRAY_SIZE(alc885_mbp_6ch_modes),
6257 .input_mux = &alc882_capture_source,
6258 .dig_out_nid = ALC882_DIGOUT_NID,
6259 .dig_in_nid = ALC882_DIGIN_NID,
6260 .unsol_event = alc885_mbp3_unsol_event,
6261 .init_hook = alc885_mbp3_automute,
6262 },
Tobin Davis9102cd12006-12-15 10:02:12 +01006263 [ALC885_MACPRO] = {
6264 .mixers = { alc882_macpro_mixer },
6265 .init_verbs = { alc882_macpro_init_verbs },
6266 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
6267 .dac_nids = alc882_dac_nids,
6268 .dig_out_nid = ALC882_DIGOUT_NID,
6269 .dig_in_nid = ALC882_DIGIN_NID,
6270 .num_channel_mode = ARRAY_SIZE(alc882_ch_modes),
6271 .channel_mode = alc882_ch_modes,
6272 .input_mux = &alc882_capture_source,
Takashi Iwai7debbe52007-08-16 15:01:03 +02006273 .init_hook = alc885_macpro_init_hook,
Tobin Davis9102cd12006-12-15 10:02:12 +01006274 },
Nicola Fagnanic54728d2007-07-19 23:28:52 +02006275 [ALC885_IMAC24] = {
6276 .mixers = { alc885_imac24_mixer },
6277 .init_verbs = { alc885_imac24_init_verbs },
6278 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
6279 .dac_nids = alc882_dac_nids,
6280 .dig_out_nid = ALC882_DIGOUT_NID,
6281 .dig_in_nid = ALC882_DIGIN_NID,
6282 .num_channel_mode = ARRAY_SIZE(alc882_ch_modes),
6283 .channel_mode = alc882_ch_modes,
6284 .input_mux = &alc882_capture_source,
6285 .unsol_event = alc885_imac24_unsol_event,
Takashi Iwai7debbe52007-08-16 15:01:03 +02006286 .init_hook = alc885_imac24_init_hook,
Nicola Fagnanic54728d2007-07-19 23:28:52 +02006287 },
Kailang Yang272a5272007-05-14 11:00:38 +02006288 [ALC882_TARGA] = {
6289 .mixers = { alc882_targa_mixer, alc882_chmode_mixer,
6290 alc882_capture_mixer },
6291 .init_verbs = { alc882_init_verbs, alc882_targa_verbs},
6292 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
6293 .dac_nids = alc882_dac_nids,
6294 .dig_out_nid = ALC882_DIGOUT_NID,
6295 .num_adc_nids = ARRAY_SIZE(alc882_adc_nids),
6296 .adc_nids = alc882_adc_nids,
Takashi Iwaie1406342008-02-11 18:32:32 +01006297 .capsrc_nids = alc882_capsrc_nids,
Kailang Yang272a5272007-05-14 11:00:38 +02006298 .num_channel_mode = ARRAY_SIZE(alc882_3ST_6ch_modes),
6299 .channel_mode = alc882_3ST_6ch_modes,
6300 .need_dac_fix = 1,
6301 .input_mux = &alc882_capture_source,
6302 .unsol_event = alc882_targa_unsol_event,
6303 .init_hook = alc882_targa_automute,
6304 },
6305 [ALC882_ASUS_A7J] = {
6306 .mixers = { alc882_asus_a7j_mixer, alc882_chmode_mixer,
6307 alc882_capture_mixer },
6308 .init_verbs = { alc882_init_verbs, alc882_asus_a7j_verbs},
6309 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
6310 .dac_nids = alc882_dac_nids,
6311 .dig_out_nid = ALC882_DIGOUT_NID,
6312 .num_adc_nids = ARRAY_SIZE(alc882_adc_nids),
6313 .adc_nids = alc882_adc_nids,
Takashi Iwaie1406342008-02-11 18:32:32 +01006314 .capsrc_nids = alc882_capsrc_nids,
Kailang Yang272a5272007-05-14 11:00:38 +02006315 .num_channel_mode = ARRAY_SIZE(alc882_3ST_6ch_modes),
6316 .channel_mode = alc882_3ST_6ch_modes,
6317 .need_dac_fix = 1,
6318 .input_mux = &alc882_capture_source,
6319 },
Takashi Iwai914759b2007-09-06 14:52:04 +02006320 [ALC882_ASUS_A7M] = {
6321 .mixers = { alc882_asus_a7m_mixer, alc882_chmode_mixer },
6322 .init_verbs = { alc882_init_verbs, alc882_eapd_verbs,
6323 alc880_gpio1_init_verbs,
6324 alc882_asus_a7m_verbs },
6325 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
6326 .dac_nids = alc882_dac_nids,
6327 .dig_out_nid = ALC882_DIGOUT_NID,
6328 .num_channel_mode = ARRAY_SIZE(alc880_threestack_modes),
6329 .channel_mode = alc880_threestack_modes,
6330 .need_dac_fix = 1,
6331 .input_mux = &alc882_capture_source,
6332 },
Kailang Yangdf694da2005-12-05 19:42:22 +01006333};
6334
6335
6336/*
Takashi Iwaif95474e2007-07-10 00:47:43 +02006337 * Pin config fixes
6338 */
6339enum {
6340 PINFIX_ABIT_AW9D_MAX
6341};
6342
6343static struct alc_pincfg alc882_abit_aw9d_pinfix[] = {
6344 { 0x15, 0x01080104 }, /* side */
6345 { 0x16, 0x01011012 }, /* rear */
6346 { 0x17, 0x01016011 }, /* clfe */
6347 { }
6348};
6349
6350static const struct alc_pincfg *alc882_pin_fixes[] = {
6351 [PINFIX_ABIT_AW9D_MAX] = alc882_abit_aw9d_pinfix,
6352};
6353
6354static struct snd_pci_quirk alc882_pinfix_tbl[] = {
6355 SND_PCI_QUIRK(0x147b, 0x107a, "Abit AW9D-MAX", PINFIX_ABIT_AW9D_MAX),
6356 {}
6357};
6358
6359/*
Kailang Yangdf694da2005-12-05 19:42:22 +01006360 * BIOS auto configuration
6361 */
6362static void alc882_auto_set_output_and_unmute(struct hda_codec *codec,
6363 hda_nid_t nid, int pin_type,
6364 int dac_idx)
6365{
6366 /* set as output */
6367 struct alc_spec *spec = codec->spec;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006368 int idx;
6369
Takashi Iwaif6c7e542008-02-12 18:32:23 +01006370 alc_set_pin_output(codec, nid, pin_type);
Kailang Yangdf694da2005-12-05 19:42:22 +01006371 if (spec->multiout.dac_nids[dac_idx] == 0x25)
6372 idx = 4;
6373 else
6374 idx = spec->multiout.dac_nids[dac_idx] - 2;
Kailang Yangdf694da2005-12-05 19:42:22 +01006375 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_CONNECT_SEL, idx);
6376
6377}
6378
6379static void alc882_auto_init_multi_out(struct hda_codec *codec)
6380{
6381 struct alc_spec *spec = codec->spec;
6382 int i;
6383
Kailang Yangbc9f98a2007-04-12 13:06:07 +02006384 alc_subsystem_id(codec, 0x15, 0x1b, 0x14);
Kailang Yangdf694da2005-12-05 19:42:22 +01006385 for (i = 0; i <= HDA_SIDE; i++) {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006386 hda_nid_t nid = spec->autocfg.line_out_pins[i];
Takashi Iwaibaba8ee2007-04-23 17:17:48 +02006387 int pin_type = get_pin_type(spec->autocfg.line_out_type);
Kailang Yangdf694da2005-12-05 19:42:22 +01006388 if (nid)
Takashi Iwaibaba8ee2007-04-23 17:17:48 +02006389 alc882_auto_set_output_and_unmute(codec, nid, pin_type,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006390 i);
Kailang Yangdf694da2005-12-05 19:42:22 +01006391 }
6392}
6393
6394static void alc882_auto_init_hp_out(struct hda_codec *codec)
6395{
6396 struct alc_spec *spec = codec->spec;
6397 hda_nid_t pin;
6398
Takashi Iwaieb06ed82006-09-20 17:10:27 +02006399 pin = spec->autocfg.hp_pins[0];
Kailang Yangdf694da2005-12-05 19:42:22 +01006400 if (pin) /* connect to front */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006401 /* use dac 0 */
6402 alc882_auto_set_output_and_unmute(codec, pin, PIN_HP, 0);
Takashi Iwaif6c7e542008-02-12 18:32:23 +01006403 pin = spec->autocfg.speaker_pins[0];
6404 if (pin)
6405 alc882_auto_set_output_and_unmute(codec, pin, PIN_OUT, 0);
Kailang Yangdf694da2005-12-05 19:42:22 +01006406}
6407
6408#define alc882_is_input_pin(nid) alc880_is_input_pin(nid)
6409#define ALC882_PIN_CD_NID ALC880_PIN_CD_NID
6410
6411static void alc882_auto_init_analog_input(struct hda_codec *codec)
6412{
6413 struct alc_spec *spec = codec->spec;
6414 int i;
6415
6416 for (i = 0; i < AUTO_PIN_LAST; i++) {
6417 hda_nid_t nid = spec->autocfg.input_pins[i];
Takashi Iwai7194cae2008-03-06 16:58:17 +01006418 unsigned int vref;
6419 if (!nid)
6420 continue;
6421 vref = PIN_IN;
6422 if (1 /*i <= AUTO_PIN_FRONT_MIC*/) {
Kailang Yang531240f2008-05-27 12:10:25 +02006423 unsigned int pincap;
6424 pincap = snd_hda_param_read(codec, nid, AC_PAR_PIN_CAP);
6425 if ((pincap >> AC_PINCAP_VREF_SHIFT) &
Takashi Iwai7194cae2008-03-06 16:58:17 +01006426 AC_PINCAP_VREF_80)
6427 vref = PIN_VREF80;
Kailang Yangdf694da2005-12-05 19:42:22 +01006428 }
Takashi Iwai7194cae2008-03-06 16:58:17 +01006429 snd_hda_codec_write(codec, nid, 0,
6430 AC_VERB_SET_PIN_WIDGET_CONTROL, vref);
6431 if (get_wcaps(codec, nid) & AC_WCAP_OUT_AMP)
6432 snd_hda_codec_write(codec, nid, 0,
6433 AC_VERB_SET_AMP_GAIN_MUTE,
6434 AMP_OUT_MUTE);
Kailang Yangdf694da2005-12-05 19:42:22 +01006435 }
6436}
6437
Takashi Iwai776e1842007-08-29 15:07:11 +02006438/* add mic boosts if needed */
6439static int alc_auto_add_mic_boost(struct hda_codec *codec)
6440{
6441 struct alc_spec *spec = codec->spec;
6442 int err;
6443 hda_nid_t nid;
6444
6445 nid = spec->autocfg.input_pins[AUTO_PIN_MIC];
Takashi Iwaice22e032008-01-11 17:38:35 +01006446 if (nid && (get_wcaps(codec, nid) & AC_WCAP_IN_AMP)) {
Takashi Iwai776e1842007-08-29 15:07:11 +02006447 err = add_control(spec, ALC_CTL_WIDGET_VOL,
6448 "Mic Boost",
6449 HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_INPUT));
6450 if (err < 0)
6451 return err;
6452 }
6453 nid = spec->autocfg.input_pins[AUTO_PIN_FRONT_MIC];
Takashi Iwaice22e032008-01-11 17:38:35 +01006454 if (nid && (get_wcaps(codec, nid) & AC_WCAP_IN_AMP)) {
Takashi Iwai776e1842007-08-29 15:07:11 +02006455 err = add_control(spec, ALC_CTL_WIDGET_VOL,
6456 "Front Mic Boost",
6457 HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_INPUT));
6458 if (err < 0)
6459 return err;
6460 }
6461 return 0;
6462}
6463
Kailang Yangdf694da2005-12-05 19:42:22 +01006464/* almost identical with ALC880 parser... */
6465static int alc882_parse_auto_config(struct hda_codec *codec)
6466{
6467 struct alc_spec *spec = codec->spec;
6468 int err = alc880_parse_auto_config(codec);
6469
6470 if (err < 0)
6471 return err;
Takashi Iwai776e1842007-08-29 15:07:11 +02006472 else if (!err)
6473 return 0; /* no config found */
6474
6475 err = alc_auto_add_mic_boost(codec);
6476 if (err < 0)
6477 return err;
6478
6479 /* hack - override the init verbs */
6480 spec->init_verbs[0] = alc882_auto_init_verbs;
6481
6482 return 1; /* config found */
Kailang Yangdf694da2005-12-05 19:42:22 +01006483}
6484
Takashi Iwaiae6b8132006-03-03 16:47:17 +01006485/* additional initialization for auto-configuration model */
6486static void alc882_auto_init(struct hda_codec *codec)
Kailang Yangdf694da2005-12-05 19:42:22 +01006487{
Takashi Iwaif6c7e542008-02-12 18:32:23 +01006488 struct alc_spec *spec = codec->spec;
Kailang Yangdf694da2005-12-05 19:42:22 +01006489 alc882_auto_init_multi_out(codec);
6490 alc882_auto_init_hp_out(codec);
6491 alc882_auto_init_analog_input(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +01006492 if (spec->unsol_event)
6493 alc_sku_automute(codec);
Kailang Yangdf694da2005-12-05 19:42:22 +01006494}
6495
Takashi Iwai7943a8a2008-04-16 17:29:09 +02006496static int patch_alc883(struct hda_codec *codec); /* called in patch_alc882() */
6497
Linus Torvalds1da177e2005-04-16 15:20:36 -07006498static int patch_alc882(struct hda_codec *codec)
6499{
6500 struct alc_spec *spec;
Kailang Yangdf694da2005-12-05 19:42:22 +01006501 int err, board_config;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006502
Takashi Iwaie560d8d2005-09-09 14:21:46 +02006503 spec = kzalloc(sizeof(*spec), GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006504 if (spec == NULL)
6505 return -ENOMEM;
6506
Linus Torvalds1da177e2005-04-16 15:20:36 -07006507 codec->spec = spec;
6508
Takashi Iwaif5fcc132006-11-24 17:07:44 +01006509 board_config = snd_hda_check_board_config(codec, ALC882_MODEL_LAST,
6510 alc882_models,
6511 alc882_cfg_tbl);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006512
Kailang Yangdf694da2005-12-05 19:42:22 +01006513 if (board_config < 0 || board_config >= ALC882_MODEL_LAST) {
Tobin Davis081d17c2007-02-15 17:46:18 +01006514 /* Pick up systems that don't supply PCI SSID */
6515 switch (codec->subsystem_id) {
6516 case 0x106b0c00: /* Mac Pro */
6517 board_config = ALC885_MACPRO;
6518 break;
Nicola Fagnanic54728d2007-07-19 23:28:52 +02006519 case 0x106b1000: /* iMac 24 */
6520 board_config = ALC885_IMAC24;
6521 break;
Takashi Iwaic7e07572008-06-26 14:42:51 +02006522 case 0x106b00a1: /* Macbook (might be wrong - PCI SSID?) */
Takashi Iwai87350ad2007-08-16 18:19:38 +02006523 case 0x106b2c00: /* Macbook Pro rev3 */
Takashi Iwaic7e07572008-06-26 14:42:51 +02006524 case 0x106b3600: /* Macbook 3.1 */
Takashi Iwai87350ad2007-08-16 18:19:38 +02006525 board_config = ALC885_MBP3;
6526 break;
Tobin Davis081d17c2007-02-15 17:46:18 +01006527 default:
Takashi Iwai7943a8a2008-04-16 17:29:09 +02006528 /* ALC889A is handled better as ALC888-compatible */
6529 if (codec->revision_id == 0x100103) {
6530 alc_free(codec);
6531 return patch_alc883(codec);
6532 }
Tobin Davis081d17c2007-02-15 17:46:18 +01006533 printk(KERN_INFO "hda_codec: Unknown model for ALC882, "
6534 "trying auto-probe from BIOS...\n");
6535 board_config = ALC882_AUTO;
6536 }
Kailang Yangdf694da2005-12-05 19:42:22 +01006537 }
Takashi Iwaie9edcee2005-06-13 14:16:38 +02006538
Takashi Iwaif95474e2007-07-10 00:47:43 +02006539 alc_fix_pincfg(codec, alc882_pinfix_tbl, alc882_pin_fixes);
6540
Kailang Yangdf694da2005-12-05 19:42:22 +01006541 if (board_config == ALC882_AUTO) {
6542 /* automatic parse from the BIOS config */
6543 err = alc882_parse_auto_config(codec);
6544 if (err < 0) {
6545 alc_free(codec);
6546 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006547 } else if (!err) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006548 printk(KERN_INFO
6549 "hda_codec: Cannot set up configuration "
6550 "from BIOS. Using base mode...\n");
Kailang Yangdf694da2005-12-05 19:42:22 +01006551 board_config = ALC882_3ST_DIG;
6552 }
6553 }
6554
6555 if (board_config != ALC882_AUTO)
6556 setup_preset(spec, &alc882_presets[board_config]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006557
Kailang Yang2f893282008-05-27 12:14:47 +02006558 if (codec->vendor_id == 0x10ec0885) {
6559 spec->stream_name_analog = "ALC885 Analog";
6560 spec->stream_name_digital = "ALC885 Digital";
6561 } else {
6562 spec->stream_name_analog = "ALC882 Analog";
6563 spec->stream_name_digital = "ALC882 Digital";
6564 }
6565
Kailang Yangdf694da2005-12-05 19:42:22 +01006566 spec->stream_analog_playback = &alc882_pcm_analog_playback;
6567 spec->stream_analog_capture = &alc882_pcm_analog_capture;
Takashi Iwai63300792008-01-24 15:31:36 +01006568 /* FIXME: setup DAC5 */
6569 /*spec->stream_analog_alt_playback = &alc880_pcm_analog_alt_playback;*/
6570 spec->stream_analog_alt_capture = &alc880_pcm_analog_alt_capture;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006571
Kailang Yangdf694da2005-12-05 19:42:22 +01006572 spec->stream_digital_playback = &alc882_pcm_digital_playback;
6573 spec->stream_digital_capture = &alc882_pcm_digital_capture;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006574
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006575 if (!spec->adc_nids && spec->input_mux) {
Kailang Yangdf694da2005-12-05 19:42:22 +01006576 /* check whether NID 0x07 is valid */
Takashi Iwai4a471b72005-12-07 13:56:29 +01006577 unsigned int wcap = get_wcaps(codec, 0x07);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006578 /* get type */
6579 wcap = (wcap & AC_WCAP_TYPE) >> AC_WCAP_TYPE_SHIFT;
Kailang Yangdf694da2005-12-05 19:42:22 +01006580 if (wcap != AC_WID_AUD_IN) {
6581 spec->adc_nids = alc882_adc_nids_alt;
6582 spec->num_adc_nids = ARRAY_SIZE(alc882_adc_nids_alt);
Takashi Iwaie1406342008-02-11 18:32:32 +01006583 spec->capsrc_nids = alc882_capsrc_nids_alt;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006584 spec->mixers[spec->num_mixers] =
6585 alc882_capture_alt_mixer;
Kailang Yangdf694da2005-12-05 19:42:22 +01006586 spec->num_mixers++;
6587 } else {
6588 spec->adc_nids = alc882_adc_nids;
6589 spec->num_adc_nids = ARRAY_SIZE(alc882_adc_nids);
Takashi Iwaie1406342008-02-11 18:32:32 +01006590 spec->capsrc_nids = alc882_capsrc_nids;
Kailang Yangdf694da2005-12-05 19:42:22 +01006591 spec->mixers[spec->num_mixers] = alc882_capture_mixer;
6592 spec->num_mixers++;
6593 }
6594 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07006595
Takashi Iwai2134ea42008-01-10 16:53:55 +01006596 spec->vmaster_nid = 0x0c;
6597
Linus Torvalds1da177e2005-04-16 15:20:36 -07006598 codec->patch_ops = alc_patch_ops;
Kailang Yangdf694da2005-12-05 19:42:22 +01006599 if (board_config == ALC882_AUTO)
Takashi Iwaiae6b8132006-03-03 16:47:17 +01006600 spec->init_hook = alc882_auto_init;
Takashi Iwaicb53c622007-08-10 17:21:45 +02006601#ifdef CONFIG_SND_HDA_POWER_SAVE
6602 if (!spec->loopback.amplist)
6603 spec->loopback.amplist = alc882_loopbacks;
6604#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07006605
6606 return 0;
6607}
6608
6609/*
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006610 * ALC883 support
6611 *
6612 * ALC883 is almost identical with ALC880 but has cleaner and more flexible
6613 * configuration. Each pin widget can choose any input DACs and a mixer.
6614 * Each ADC is connected from a mixer of all inputs. This makes possible
6615 * 6-channel independent captures.
6616 *
6617 * In addition, an independent DAC for the multi-playback (not used in this
6618 * driver yet).
6619 */
6620#define ALC883_DIGOUT_NID 0x06
6621#define ALC883_DIGIN_NID 0x0a
6622
6623static hda_nid_t alc883_dac_nids[4] = {
6624 /* front, rear, clfe, rear_surr */
Herton Ronaldo Krzesinskif32a19e2008-03-18 09:27:08 +01006625 0x02, 0x03, 0x04, 0x05
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006626};
6627
6628static hda_nid_t alc883_adc_nids[2] = {
6629 /* ADC1-2 */
6630 0x08, 0x09,
6631};
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006632
Takashi Iwaie1406342008-02-11 18:32:32 +01006633static hda_nid_t alc883_capsrc_nids[2] = { 0x23, 0x22 };
6634
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006635/* input MUX */
6636/* FIXME: should be a matrix-type input source selection */
6637
6638static struct hda_input_mux alc883_capture_source = {
6639 .num_items = 4,
6640 .items = {
6641 { "Mic", 0x0 },
6642 { "Front Mic", 0x1 },
6643 { "Line", 0x2 },
6644 { "CD", 0x4 },
6645 },
6646};
Kailang Yangbc9f98a2007-04-12 13:06:07 +02006647
Jiang zhe17bba1b2008-06-04 12:11:07 +02006648static struct hda_input_mux alc883_3stack_6ch_intel = {
6649 .num_items = 4,
6650 .items = {
6651 { "Mic", 0x1 },
6652 { "Front Mic", 0x0 },
6653 { "Line", 0x2 },
6654 { "CD", 0x4 },
6655 },
6656};
6657
Kailang Yangbc9f98a2007-04-12 13:06:07 +02006658static struct hda_input_mux alc883_lenovo_101e_capture_source = {
6659 .num_items = 2,
6660 .items = {
6661 { "Mic", 0x1 },
6662 { "Line", 0x2 },
6663 },
6664};
6665
Kailang Yang272a5272007-05-14 11:00:38 +02006666static struct hda_input_mux alc883_lenovo_nb0763_capture_source = {
6667 .num_items = 4,
6668 .items = {
6669 { "Mic", 0x0 },
6670 { "iMic", 0x1 },
6671 { "Line", 0x2 },
6672 { "CD", 0x4 },
6673 },
6674};
6675
Jiang zhefb97dc62008-03-06 11:07:11 +01006676static struct hda_input_mux alc883_fujitsu_pi2515_capture_source = {
6677 .num_items = 2,
6678 .items = {
6679 { "Mic", 0x0 },
6680 { "Int Mic", 0x1 },
6681 },
6682};
6683
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006684#define alc883_mux_enum_info alc_mux_enum_info
6685#define alc883_mux_enum_get alc_mux_enum_get
Takashi Iwaie1406342008-02-11 18:32:32 +01006686/* ALC883 has the ALC882-type input selection */
6687#define alc883_mux_enum_put alc882_mux_enum_put
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006688
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006689/*
6690 * 2ch mode
6691 */
6692static struct hda_channel_mode alc883_3ST_2ch_modes[1] = {
6693 { 2, NULL }
6694};
6695
6696/*
6697 * 2ch mode
6698 */
6699static struct hda_verb alc883_3ST_ch2_init[] = {
6700 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
6701 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6702 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
6703 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6704 { } /* end */
6705};
6706
6707/*
Tobin Davisb2011312007-09-17 12:45:11 +02006708 * 4ch mode
6709 */
6710static struct hda_verb alc883_3ST_ch4_init[] = {
6711 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
6712 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6713 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6714 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6715 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x01 },
6716 { } /* end */
6717};
6718
6719/*
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006720 * 6ch mode
6721 */
6722static struct hda_verb alc883_3ST_ch6_init[] = {
6723 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6724 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6725 { 0x18, AC_VERB_SET_CONNECT_SEL, 0x02 },
6726 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6727 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6728 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x01 },
6729 { } /* end */
6730};
6731
Tobin Davisb2011312007-09-17 12:45:11 +02006732static struct hda_channel_mode alc883_3ST_6ch_modes[3] = {
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006733 { 2, alc883_3ST_ch2_init },
Tobin Davisb2011312007-09-17 12:45:11 +02006734 { 4, alc883_3ST_ch4_init },
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006735 { 6, alc883_3ST_ch6_init },
6736};
6737
6738/*
Jiang zhe17bba1b2008-06-04 12:11:07 +02006739 * 2ch mode
6740 */
6741static struct hda_verb alc883_3ST_ch2_intel_init[] = {
6742 { 0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
6743 { 0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6744 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
6745 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6746 { } /* end */
6747};
6748
6749/*
6750 * 4ch mode
6751 */
6752static struct hda_verb alc883_3ST_ch4_intel_init[] = {
6753 { 0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
6754 { 0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6755 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6756 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6757 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x01 },
6758 { } /* end */
6759};
6760
6761/*
6762 * 6ch mode
6763 */
6764static struct hda_verb alc883_3ST_ch6_intel_init[] = {
6765 { 0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6766 { 0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6767 { 0x19, AC_VERB_SET_CONNECT_SEL, 0x02 },
6768 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6769 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6770 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x01 },
6771 { } /* end */
6772};
6773
6774static struct hda_channel_mode alc883_3ST_6ch_intel_modes[3] = {
6775 { 2, alc883_3ST_ch2_intel_init },
6776 { 4, alc883_3ST_ch4_intel_init },
6777 { 6, alc883_3ST_ch6_intel_init },
6778};
6779
6780/*
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006781 * 6ch mode
6782 */
6783static struct hda_verb alc883_sixstack_ch6_init[] = {
6784 { 0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
6785 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6786 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6787 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6788 { } /* end */
6789};
6790
6791/*
6792 * 8ch mode
6793 */
6794static struct hda_verb alc883_sixstack_ch8_init[] = {
6795 { 0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6796 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6797 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6798 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6799 { } /* end */
6800};
6801
6802static struct hda_channel_mode alc883_sixstack_modes[2] = {
6803 { 6, alc883_sixstack_ch6_init },
6804 { 8, alc883_sixstack_ch8_init },
6805};
6806
Andrew L. Neporadab373bde2006-11-07 11:37:08 +01006807static struct hda_verb alc883_medion_eapd_verbs[] = {
6808 /* eanable EAPD on medion laptop */
6809 {0x20, AC_VERB_SET_COEF_INDEX, 0x07},
6810 {0x20, AC_VERB_SET_PROC_COEF, 0x3070},
6811 { }
6812};
6813
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006814/* Pin assignment: Front=0x14, Rear=0x15, CLFE=0x16, Side=0x17
6815 * Mic=0x18, Front Mic=0x19, Line-In=0x1a, HP=0x1b
6816 */
6817
6818static struct snd_kcontrol_new alc883_base_mixer[] = {
6819 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
6820 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
6821 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
6822 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
6823 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
6824 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
6825 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
6826 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
6827 HDA_CODEC_VOLUME("Side Playback Volume", 0x0f, 0x0, HDA_OUTPUT),
6828 HDA_BIND_MUTE("Side Playback Switch", 0x0f, 2, HDA_INPUT),
6829 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
6830 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
6831 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
6832 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
6833 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
6834 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01006835 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006836 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
6837 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01006838 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006839 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
6840 HDA_CODEC_VOLUME("PC Speaker Playback Volume", 0x0b, 0x05, HDA_INPUT),
6841 HDA_CODEC_MUTE("PC Speaker Playback Switch", 0x0b, 0x05, HDA_INPUT),
6842 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
6843 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
6844 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x09, 0x0, HDA_INPUT),
6845 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x09, 0x0, HDA_INPUT),
6846 {
6847 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
6848 /* .name = "Capture Source", */
6849 .name = "Input Source",
6850 .count = 2,
6851 .info = alc883_mux_enum_info,
6852 .get = alc883_mux_enum_get,
6853 .put = alc883_mux_enum_put,
6854 },
6855 { } /* end */
6856};
6857
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +01006858static struct snd_kcontrol_new alc883_mitac_mixer[] = {
6859 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
6860 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
6861 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
6862 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
6863 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
6864 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
6865 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
6866 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
6867 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
6868 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
6869 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
6870 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
6871 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
6872 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
6873 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
6874 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x09, 0x0, HDA_INPUT),
6875 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x09, 0x0, HDA_INPUT),
6876 {
6877 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
6878 /* .name = "Capture Source", */
6879 .name = "Input Source",
6880 .count = 2,
6881 .info = alc883_mux_enum_info,
6882 .get = alc883_mux_enum_get,
6883 .put = alc883_mux_enum_put,
6884 },
6885 { } /* end */
6886};
6887
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01006888static struct snd_kcontrol_new alc883_clevo_m720_mixer[] = {
Jiang zhe368c7a92008-03-04 11:20:33 +01006889 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
6890 HDA_BIND_MUTE("Headphone Playback Switch", 0x0c, 2, HDA_INPUT),
6891 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
6892 HDA_BIND_MUTE("Speaker Playback Switch", 0x0d, 2, HDA_INPUT),
6893 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
6894 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
6895 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
6896 HDA_CODEC_VOLUME("Int Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
6897 HDA_CODEC_VOLUME("Int Mic Boost", 0x19, 0, HDA_INPUT),
6898 HDA_CODEC_MUTE("Int Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
6899 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
6900 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
6901 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x09, 0x0, HDA_INPUT),
6902 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x09, 0x0, HDA_INPUT),
6903 {
6904 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
6905 /* .name = "Capture Source", */
6906 .name = "Input Source",
6907 .count = 2,
6908 .info = alc883_mux_enum_info,
6909 .get = alc883_mux_enum_get,
6910 .put = alc883_mux_enum_put,
6911 },
6912 { } /* end */
6913};
6914
Jiang zhefb97dc62008-03-06 11:07:11 +01006915static struct snd_kcontrol_new alc883_2ch_fujitsu_pi2515_mixer[] = {
6916 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
6917 HDA_BIND_MUTE("Headphone Playback Switch", 0x0c, 2, HDA_INPUT),
6918 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
6919 HDA_BIND_MUTE("Speaker Playback Switch", 0x0d, 2, HDA_INPUT),
6920 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
6921 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
6922 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
6923 HDA_CODEC_VOLUME("Int Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
6924 HDA_CODEC_VOLUME("Int Mic Boost", 0x19, 0, HDA_INPUT),
6925 HDA_CODEC_MUTE("Int Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
6926 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
6927 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
6928 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x09, 0x0, HDA_INPUT),
6929 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x09, 0x0, HDA_INPUT),
6930 {
6931 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
6932 /* .name = "Capture Source", */
6933 .name = "Input Source",
6934 .count = 2,
6935 .info = alc883_mux_enum_info,
6936 .get = alc883_mux_enum_get,
6937 .put = alc883_mux_enum_put,
6938 },
6939 { } /* end */
6940};
6941
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006942static struct snd_kcontrol_new alc883_3ST_2ch_mixer[] = {
6943 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
6944 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
6945 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
6946 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
6947 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
6948 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
6949 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
6950 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01006951 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006952 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
6953 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01006954 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006955 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
6956 HDA_CODEC_VOLUME("PC Speaker Playback Volume", 0x0b, 0x05, HDA_INPUT),
6957 HDA_CODEC_MUTE("PC Speaker Playback Switch", 0x0b, 0x05, HDA_INPUT),
6958 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
6959 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
6960 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x09, 0x0, HDA_INPUT),
6961 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x09, 0x0, HDA_INPUT),
6962 {
6963 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
6964 /* .name = "Capture Source", */
6965 .name = "Input Source",
6966 .count = 2,
6967 .info = alc883_mux_enum_info,
6968 .get = alc883_mux_enum_get,
6969 .put = alc883_mux_enum_put,
6970 },
6971 { } /* end */
6972};
6973
6974static struct snd_kcontrol_new alc883_3ST_6ch_mixer[] = {
6975 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
6976 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
6977 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
6978 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
6979 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
6980 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
6981 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
6982 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
6983 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
6984 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
6985 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
6986 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
6987 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
6988 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01006989 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006990 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
6991 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01006992 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006993 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
6994 HDA_CODEC_VOLUME("PC Speaker Playback Volume", 0x0b, 0x05, HDA_INPUT),
6995 HDA_CODEC_MUTE("PC Speaker Playback Switch", 0x0b, 0x05, HDA_INPUT),
6996 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
6997 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
6998 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x09, 0x0, HDA_INPUT),
6999 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x09, 0x0, HDA_INPUT),
7000 {
7001 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
7002 /* .name = "Capture Source", */
7003 .name = "Input Source",
7004 .count = 2,
7005 .info = alc883_mux_enum_info,
7006 .get = alc883_mux_enum_get,
7007 .put = alc883_mux_enum_put,
7008 },
7009 { } /* end */
7010};
7011
Jiang zhe17bba1b2008-06-04 12:11:07 +02007012static struct snd_kcontrol_new alc883_3ST_6ch_intel_mixer[] = {
7013 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7014 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
7015 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
7016 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
7017 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0,
7018 HDA_OUTPUT),
7019 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
7020 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
7021 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
7022 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
7023 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
7024 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
7025 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
7026 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
7027 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
7028 HDA_CODEC_VOLUME("Mic Boost", 0x19, 0, HDA_INPUT),
7029 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
7030 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
7031 HDA_CODEC_VOLUME("Front Mic Boost", 0x18, 0, HDA_INPUT),
7032 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
7033 HDA_CODEC_VOLUME("PC Speaker Playback Volume", 0x0b, 0x05, HDA_INPUT),
7034 HDA_CODEC_MUTE("PC Speaker Playback Switch", 0x0b, 0x05, HDA_INPUT),
7035 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
7036 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
7037 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x09, 0x0, HDA_INPUT),
7038 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x09, 0x0, HDA_INPUT),
7039 {
7040 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
7041 /* .name = "Capture Source", */
7042 .name = "Input Source",
7043 .count = 2,
7044 .info = alc883_mux_enum_info,
7045 .get = alc883_mux_enum_get,
7046 .put = alc883_mux_enum_put,
7047 },
7048 { } /* end */
7049};
7050
Takashi Iwaid1d985f2006-11-23 19:27:12 +01007051static struct snd_kcontrol_new alc883_fivestack_mixer[] = {
Tobin Davisc07584c2006-10-13 12:32:16 +02007052 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwai0701e062008-06-27 16:30:57 +02007053 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Tobin Davisc07584c2006-10-13 12:32:16 +02007054 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
Takashi Iwai0701e062008-06-27 16:30:57 +02007055 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
Tobin Davisc07584c2006-10-13 12:32:16 +02007056 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
7057 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
Takashi Iwai0701e062008-06-27 16:30:57 +02007058 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
7059 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
Tobin Davisc07584c2006-10-13 12:32:16 +02007060 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
7061 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
7062 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
7063 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
7064 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
7065 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01007066 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Tobin Davisc07584c2006-10-13 12:32:16 +02007067 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
7068 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01007069 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
Tobin Davisc07584c2006-10-13 12:32:16 +02007070 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
7071 HDA_CODEC_VOLUME("PC Speaker Playback Volume", 0x0b, 0x05, HDA_INPUT),
7072 HDA_CODEC_MUTE("PC Speaker Playback Switch", 0x0b, 0x05, HDA_INPUT),
7073 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
7074 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
7075
7076 {
7077 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
7078 /* .name = "Capture Source", */
7079 .name = "Input Source",
7080 .count = 1,
7081 .info = alc883_mux_enum_info,
7082 .get = alc883_mux_enum_get,
7083 .put = alc883_mux_enum_put,
7084 },
7085 { } /* end */
7086};
7087
Kailang Yangccc656c2006-10-17 12:32:26 +02007088static struct snd_kcontrol_new alc883_tagra_mixer[] = {
7089 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7090 HDA_CODEC_MUTE("Headphone Playback Switch", 0x14, 0x0, HDA_OUTPUT),
7091 HDA_CODEC_MUTE("Front Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
7092 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
7093 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
7094 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
7095 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
7096 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
7097 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
7098 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
7099 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
7100 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
7101 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
7102 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01007103 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Kailang Yangccc656c2006-10-17 12:32:26 +02007104 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
7105 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
7106 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
7107 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x09, 0x0, HDA_INPUT),
7108 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x09, 0x0, HDA_INPUT),
7109 {
7110 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
7111 /* .name = "Capture Source", */
7112 .name = "Input Source",
7113 .count = 2,
7114 .info = alc883_mux_enum_info,
7115 .get = alc883_mux_enum_get,
7116 .put = alc883_mux_enum_put,
7117 },
7118 { } /* end */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02007119};
Kailang Yangccc656c2006-10-17 12:32:26 +02007120
7121static struct snd_kcontrol_new alc883_tagra_2ch_mixer[] = {
7122 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7123 HDA_CODEC_MUTE("Headphone Playback Switch", 0x14, 0x0, HDA_OUTPUT),
7124 HDA_CODEC_MUTE("Front Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
7125 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
7126 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
7127 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01007128 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Kailang Yangccc656c2006-10-17 12:32:26 +02007129 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Jiang zhe4383fae2008-04-14 12:58:57 +02007130 HDA_CODEC_VOLUME("Int Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
7131 HDA_CODEC_VOLUME("Int Mic Boost", 0x19, 0, HDA_INPUT),
7132 HDA_CODEC_MUTE("Int Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Kailang Yangccc656c2006-10-17 12:32:26 +02007133 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
7134 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
7135 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x09, 0x0, HDA_INPUT),
7136 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x09, 0x0, HDA_INPUT),
7137 {
7138 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
7139 /* .name = "Capture Source", */
7140 .name = "Input Source",
7141 .count = 2,
7142 .info = alc883_mux_enum_info,
7143 .get = alc883_mux_enum_get,
7144 .put = alc883_mux_enum_put,
7145 },
7146 { } /* end */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02007147};
Kailang Yangccc656c2006-10-17 12:32:26 +02007148
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007149static struct snd_kcontrol_new alc883_lenovo_101e_2ch_mixer[] = {
7150 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7151 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Takashi Iwai86cd9292008-01-28 18:09:56 +01007152 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
7153 HDA_BIND_MUTE("Speaker Playback Switch", 0x0d, 2, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007154 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
7155 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
7156 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
7157 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
7158 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
7159 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
7160 {
7161 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
7162 /* .name = "Capture Source", */
7163 .name = "Input Source",
7164 .count = 1,
7165 .info = alc883_mux_enum_info,
7166 .get = alc883_mux_enum_get,
7167 .put = alc883_mux_enum_put,
7168 },
7169 { } /* end */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02007170};
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007171
Kailang Yang272a5272007-05-14 11:00:38 +02007172static struct snd_kcontrol_new alc883_lenovo_nb0763_mixer[] = {
7173 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7174 HDA_BIND_MUTE("Speaker Playback Switch", 0x0c, 2, HDA_INPUT),
7175 HDA_CODEC_MUTE("Headphone Playback Switch", 0x14, 0x0, HDA_OUTPUT),
7176 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
7177 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
7178 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
7179 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
7180 HDA_CODEC_VOLUME("iMic Playback Volume", 0x0b, 0x1, HDA_INPUT),
7181 HDA_CODEC_MUTE("iMic Playback Switch", 0x0b, 0x1, HDA_INPUT),
7182 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
7183 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
7184 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x09, 0x0, HDA_INPUT),
7185 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x09, 0x0, HDA_INPUT),
7186 {
7187 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
7188 /* .name = "Capture Source", */
7189 .name = "Input Source",
7190 .count = 2,
7191 .info = alc883_mux_enum_info,
7192 .get = alc883_mux_enum_get,
7193 .put = alc883_mux_enum_put,
7194 },
7195 { } /* end */
7196};
7197
7198static struct snd_kcontrol_new alc883_medion_md2_mixer[] = {
7199 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7200 HDA_CODEC_MUTE("Headphone Playback Switch", 0x14, 0x0, HDA_OUTPUT),
7201 HDA_CODEC_MUTE("Front Playback Switch", 0x15, 0x0, HDA_OUTPUT),
7202 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
7203 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
7204 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
7205 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
7206 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
7207 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
7208 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
7209 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
7210 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x09, 0x0, HDA_INPUT),
7211 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x09, 0x0, HDA_INPUT),
7212 {
7213 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
7214 /* .name = "Capture Source", */
7215 .name = "Input Source",
7216 .count = 2,
7217 .info = alc883_mux_enum_info,
7218 .get = alc883_mux_enum_get,
7219 .put = alc883_mux_enum_put,
7220 },
7221 { } /* end */
7222};
7223
Tobin Davis2880a862007-08-07 11:50:26 +02007224static struct snd_kcontrol_new alc883_acer_aspire_mixer[] = {
Kailang Yangd1a991a2007-08-15 16:21:59 +02007225 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7226 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Tobin Davis2880a862007-08-07 11:50:26 +02007227 HDA_CODEC_MUTE("Headphone Playback Switch", 0x14, 0x0, HDA_OUTPUT),
Tobin Davis2880a862007-08-07 11:50:26 +02007228 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
7229 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
Kailang Yangd1a991a2007-08-15 16:21:59 +02007230 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
7231 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
7232 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Tobin Davis2880a862007-08-07 11:50:26 +02007233 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
7234 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
7235 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x09, 0x0, HDA_INPUT),
7236 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x09, 0x0, HDA_INPUT),
7237 {
7238 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
7239 /* .name = "Capture Source", */
7240 .name = "Input Source",
7241 .count = 2,
7242 .info = alc883_mux_enum_info,
7243 .get = alc883_mux_enum_get,
7244 .put = alc883_mux_enum_put,
7245 },
7246 { } /* end */
Kailang Yangd1a991a2007-08-15 16:21:59 +02007247};
Tobin Davis2880a862007-08-07 11:50:26 +02007248
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007249static struct snd_kcontrol_new alc883_chmode_mixer[] = {
7250 {
7251 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
7252 .name = "Channel Mode",
7253 .info = alc_ch_mode_info,
7254 .get = alc_ch_mode_get,
7255 .put = alc_ch_mode_put,
7256 },
7257 { } /* end */
7258};
7259
7260static struct hda_verb alc883_init_verbs[] = {
7261 /* ADC1: mute amp left and right */
7262 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7263 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
7264 /* ADC2: mute amp left and right */
7265 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7266 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
7267 /* Front mixer: unmute input/output amp left and right (volume = 0) */
7268 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7269 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7270 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
7271 /* Rear mixer */
7272 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7273 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7274 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
7275 /* CLFE mixer */
7276 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7277 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7278 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
7279 /* Side mixer */
7280 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7281 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7282 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
7283
Takashi Iwaicb53c622007-08-10 17:21:45 +02007284 /* mute analog input loopbacks */
7285 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7286 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
7287 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
7288 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
7289 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007290
7291 /* Front Pin: output 0 (0x0c) */
7292 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7293 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7294 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
7295 /* Rear Pin: output 1 (0x0d) */
7296 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7297 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7298 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01},
7299 /* CLFE Pin: output 2 (0x0e) */
7300 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7301 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7302 {0x16, AC_VERB_SET_CONNECT_SEL, 0x02},
7303 /* Side Pin: output 3 (0x0f) */
7304 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7305 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7306 {0x17, AC_VERB_SET_CONNECT_SEL, 0x03},
7307 /* Mic (rear) pin: input vref at 80% */
7308 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
7309 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
7310 /* Front Mic pin: input vref at 80% */
7311 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
7312 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
7313 /* Line In pin: input */
7314 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
7315 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
7316 /* Line-2 In: Headphone output (output 0 - 0x0c) */
7317 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
7318 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7319 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
7320 /* CD pin widget for input */
7321 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
7322
7323 /* FIXME: use matrix-type input source selection */
7324 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
7325 /* Input mixer2 */
7326 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7327 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7328 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
7329 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(4)},
7330 /* Input mixer3 */
7331 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7332 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7333 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
7334 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(4)},
7335 { }
7336};
7337
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +01007338/* toggle speaker-output according to the hp-jack state */
7339static void alc883_mitac_hp_automute(struct hda_codec *codec)
7340{
7341 unsigned int present;
7342
7343 present = snd_hda_codec_read(codec, 0x15, 0,
7344 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
7345 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
7346 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
7347 snd_hda_codec_amp_stereo(codec, 0x17, HDA_OUTPUT, 0,
7348 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
7349}
7350
7351/* auto-toggle front mic */
7352/*
7353static void alc883_mitac_mic_automute(struct hda_codec *codec)
7354{
7355 unsigned int present;
7356 unsigned char bits;
7357
7358 present = snd_hda_codec_read(codec, 0x18, 0,
7359 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
7360 bits = present ? HDA_AMP_MUTE : 0;
7361 snd_hda_codec_amp_stereo(codec, 0x0b, HDA_INPUT, 1, HDA_AMP_MUTE, bits);
7362}
7363*/
7364
7365static void alc883_mitac_automute(struct hda_codec *codec)
7366{
7367 alc883_mitac_hp_automute(codec);
7368 /* alc883_mitac_mic_automute(codec); */
7369}
7370
7371static void alc883_mitac_unsol_event(struct hda_codec *codec,
7372 unsigned int res)
7373{
7374 switch (res >> 26) {
7375 case ALC880_HP_EVENT:
7376 alc883_mitac_hp_automute(codec);
7377 break;
7378 case ALC880_MIC_EVENT:
7379 /* alc883_mitac_mic_automute(codec); */
7380 break;
7381 }
7382}
7383
7384static struct hda_verb alc883_mitac_verbs[] = {
7385 /* HP */
7386 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
7387 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
7388 /* Subwoofer */
7389 {0x17, AC_VERB_SET_CONNECT_SEL, 0x02},
7390 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7391
7392 /* enable unsolicited event */
7393 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
7394 /* {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_MIC_EVENT | AC_USRSP_EN}, */
7395
7396 { } /* end */
7397};
7398
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01007399static struct hda_verb alc883_clevo_m720_verbs[] = {
Jiang zhe368c7a92008-03-04 11:20:33 +01007400 /* HP */
7401 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
7402 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
7403 /* Int speaker */
7404 {0x14, AC_VERB_SET_CONNECT_SEL, 0x01},
7405 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7406
7407 /* enable unsolicited event */
7408 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01007409 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_MIC_EVENT | AC_USRSP_EN},
Jiang zhe368c7a92008-03-04 11:20:33 +01007410
7411 { } /* end */
7412};
7413
Jiang zhefb97dc62008-03-06 11:07:11 +01007414static struct hda_verb alc883_2ch_fujitsu_pi2515_verbs[] = {
7415 /* HP */
7416 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
7417 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
7418 /* Subwoofer */
7419 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01},
7420 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7421
7422 /* enable unsolicited event */
7423 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
7424
7425 { } /* end */
7426};
7427
Kailang Yangccc656c2006-10-17 12:32:26 +02007428static struct hda_verb alc883_tagra_verbs[] = {
7429 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7430 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7431
7432 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
7433 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7434
7435 {0x18, AC_VERB_SET_CONNECT_SEL, 0x02}, /* mic/clfe */
7436 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x01}, /* line/surround */
7437 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
7438
7439 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02007440 {0x01, AC_VERB_SET_GPIO_MASK, 0x03},
7441 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x03},
7442 {0x01, AC_VERB_SET_GPIO_DATA, 0x03},
Kailang Yangccc656c2006-10-17 12:32:26 +02007443
7444 { } /* end */
7445};
7446
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007447static struct hda_verb alc883_lenovo_101e_verbs[] = {
7448 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
7449 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_FRONT_EVENT|AC_USRSP_EN},
7450 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT|AC_USRSP_EN},
7451 { } /* end */
7452};
7453
Kailang Yang272a5272007-05-14 11:00:38 +02007454static struct hda_verb alc883_lenovo_nb0763_verbs[] = {
7455 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
7456 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
7457 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
7458 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
7459 { } /* end */
7460};
7461
7462static struct hda_verb alc888_lenovo_ms7195_verbs[] = {
7463 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7464 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7465 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
7466 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_FRONT_EVENT | AC_USRSP_EN},
7467 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
7468 { } /* end */
7469};
7470
Kailang Yang189609a2007-08-20 11:31:23 +02007471static struct hda_verb alc883_haier_w66_verbs[] = {
7472 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7473 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7474
7475 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7476
7477 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
7478 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
7479 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
7480 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
7481 { } /* end */
7482};
7483
Claudio Matsuoka4723c022007-07-13 14:36:19 +02007484static struct hda_verb alc888_3st_hp_verbs[] = {
Claudio Matsuoka8341de62007-07-06 12:10:45 +02007485 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00}, /* Front: output 0 (0x0c) */
Herton Ronaldo Krzesinskif32a19e2008-03-18 09:27:08 +01007486 {0x16, AC_VERB_SET_CONNECT_SEL, 0x01}, /* Rear : output 1 (0x0d) */
7487 {0x18, AC_VERB_SET_CONNECT_SEL, 0x02}, /* CLFE : output 2 (0x0e) */
Claudio Matsuoka8341de62007-07-06 12:10:45 +02007488 { }
7489};
7490
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01007491static struct hda_verb alc888_6st_dell_verbs[] = {
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01007492 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
7493 { }
7494};
7495
Claudio Matsuoka4723c022007-07-13 14:36:19 +02007496static struct hda_verb alc888_3st_hp_2ch_init[] = {
Claudio Matsuoka8341de62007-07-06 12:10:45 +02007497 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
7498 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
7499 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
7500 { 0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
7501 { }
7502};
7503
Claudio Matsuoka4723c022007-07-13 14:36:19 +02007504static struct hda_verb alc888_3st_hp_6ch_init[] = {
Claudio Matsuoka8341de62007-07-06 12:10:45 +02007505 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
7506 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
7507 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
7508 { 0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
7509 { }
7510};
7511
Claudio Matsuoka4723c022007-07-13 14:36:19 +02007512static struct hda_channel_mode alc888_3st_hp_modes[2] = {
7513 { 2, alc888_3st_hp_2ch_init },
7514 { 6, alc888_3st_hp_6ch_init },
Claudio Matsuoka8341de62007-07-06 12:10:45 +02007515};
7516
Kailang Yang272a5272007-05-14 11:00:38 +02007517/* toggle front-jack and RCA according to the hp-jack state */
7518static void alc888_lenovo_ms7195_front_automute(struct hda_codec *codec)
7519{
7520 unsigned int present;
7521
7522 present = snd_hda_codec_read(codec, 0x1b, 0,
7523 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +02007524 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
7525 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
7526 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
7527 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
Kailang Yang272a5272007-05-14 11:00:38 +02007528}
7529
7530/* toggle RCA according to the front-jack state */
7531static void alc888_lenovo_ms7195_rca_automute(struct hda_codec *codec)
7532{
7533 unsigned int present;
7534
7535 present = snd_hda_codec_read(codec, 0x14, 0,
7536 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +02007537 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
7538 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
Kailang Yang272a5272007-05-14 11:00:38 +02007539}
Takashi Iwai47fd8302007-08-10 17:11:07 +02007540
Kailang Yang272a5272007-05-14 11:00:38 +02007541static void alc883_lenovo_ms7195_unsol_event(struct hda_codec *codec,
7542 unsigned int res)
7543{
7544 if ((res >> 26) == ALC880_HP_EVENT)
7545 alc888_lenovo_ms7195_front_automute(codec);
7546 if ((res >> 26) == ALC880_FRONT_EVENT)
7547 alc888_lenovo_ms7195_rca_automute(codec);
7548}
7549
7550static struct hda_verb alc883_medion_md2_verbs[] = {
7551 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7552 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7553
7554 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
7555
7556 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
7557 { } /* end */
7558};
7559
7560/* toggle speaker-output according to the hp-jack state */
7561static void alc883_medion_md2_automute(struct hda_codec *codec)
7562{
7563 unsigned int present;
7564
7565 present = snd_hda_codec_read(codec, 0x14, 0,
7566 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +02007567 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
7568 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
Kailang Yang272a5272007-05-14 11:00:38 +02007569}
7570
7571static void alc883_medion_md2_unsol_event(struct hda_codec *codec,
7572 unsigned int res)
7573{
7574 if ((res >> 26) == ALC880_HP_EVENT)
7575 alc883_medion_md2_automute(codec);
7576}
7577
Kailang Yangccc656c2006-10-17 12:32:26 +02007578/* toggle speaker-output according to the hp-jack state */
7579static void alc883_tagra_automute(struct hda_codec *codec)
7580{
7581 unsigned int present;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02007582 unsigned char bits;
Kailang Yangccc656c2006-10-17 12:32:26 +02007583
7584 present = snd_hda_codec_read(codec, 0x14, 0,
7585 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +02007586 bits = present ? HDA_AMP_MUTE : 0;
7587 snd_hda_codec_amp_stereo(codec, 0x1b, HDA_OUTPUT, 0,
7588 HDA_AMP_MUTE, bits);
Takashi Iwai82beb8f2007-08-10 17:09:26 +02007589 snd_hda_codec_write_cache(codec, 1, 0, AC_VERB_SET_GPIO_DATA,
7590 present ? 1 : 3);
Kailang Yangccc656c2006-10-17 12:32:26 +02007591}
7592
7593static void alc883_tagra_unsol_event(struct hda_codec *codec, unsigned int res)
7594{
7595 if ((res >> 26) == ALC880_HP_EVENT)
7596 alc883_tagra_automute(codec);
7597}
7598
Jiang zhe368c7a92008-03-04 11:20:33 +01007599/* toggle speaker-output according to the hp-jack state */
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01007600static void alc883_clevo_m720_hp_automute(struct hda_codec *codec)
Jiang zhe368c7a92008-03-04 11:20:33 +01007601{
7602 unsigned int present;
7603 unsigned char bits;
7604
7605 present = snd_hda_codec_read(codec, 0x15, 0, AC_VERB_GET_PIN_SENSE, 0)
7606 & AC_PINSENSE_PRESENCE;
7607 bits = present ? HDA_AMP_MUTE : 0;
7608 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
7609 HDA_AMP_MUTE, bits);
7610}
7611
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01007612static void alc883_clevo_m720_mic_automute(struct hda_codec *codec)
7613{
7614 unsigned int present;
7615
7616 present = snd_hda_codec_read(codec, 0x18, 0,
7617 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
7618 snd_hda_codec_amp_stereo(codec, 0x0b, HDA_INPUT, 1,
7619 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
7620}
7621
7622static void alc883_clevo_m720_automute(struct hda_codec *codec)
7623{
7624 alc883_clevo_m720_hp_automute(codec);
7625 alc883_clevo_m720_mic_automute(codec);
7626}
7627
7628static void alc883_clevo_m720_unsol_event(struct hda_codec *codec,
Jiang zhe368c7a92008-03-04 11:20:33 +01007629 unsigned int res)
7630{
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01007631 switch (res >> 26) {
7632 case ALC880_HP_EVENT:
7633 alc883_clevo_m720_hp_automute(codec);
7634 break;
7635 case ALC880_MIC_EVENT:
7636 alc883_clevo_m720_mic_automute(codec);
7637 break;
7638 }
Jiang zhe368c7a92008-03-04 11:20:33 +01007639}
7640
Jiang zhefb97dc62008-03-06 11:07:11 +01007641/* toggle speaker-output according to the hp-jack state */
7642static void alc883_2ch_fujitsu_pi2515_automute(struct hda_codec *codec)
7643{
7644 unsigned int present;
7645 unsigned char bits;
7646
7647 present = snd_hda_codec_read(codec, 0x14, 0, AC_VERB_GET_PIN_SENSE, 0)
7648 & AC_PINSENSE_PRESENCE;
7649 bits = present ? HDA_AMP_MUTE : 0;
7650 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
7651 HDA_AMP_MUTE, bits);
7652}
7653
7654static void alc883_2ch_fujitsu_pi2515_unsol_event(struct hda_codec *codec,
7655 unsigned int res)
7656{
7657 if ((res >> 26) == ALC880_HP_EVENT)
7658 alc883_2ch_fujitsu_pi2515_automute(codec);
7659}
7660
Kailang Yang189609a2007-08-20 11:31:23 +02007661static void alc883_haier_w66_automute(struct hda_codec *codec)
7662{
7663 unsigned int present;
7664 unsigned char bits;
7665
7666 present = snd_hda_codec_read(codec, 0x1b, 0,
7667 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
7668 bits = present ? 0x80 : 0;
7669 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
7670 0x80, bits);
7671}
7672
7673static void alc883_haier_w66_unsol_event(struct hda_codec *codec,
7674 unsigned int res)
7675{
7676 if ((res >> 26) == ALC880_HP_EVENT)
7677 alc883_haier_w66_automute(codec);
7678}
7679
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007680static void alc883_lenovo_101e_ispeaker_automute(struct hda_codec *codec)
7681{
7682 unsigned int present;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02007683 unsigned char bits;
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007684
7685 present = snd_hda_codec_read(codec, 0x14, 0,
7686 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +02007687 bits = present ? HDA_AMP_MUTE : 0;
7688 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
7689 HDA_AMP_MUTE, bits);
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007690}
7691
7692static void alc883_lenovo_101e_all_automute(struct hda_codec *codec)
7693{
7694 unsigned int present;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02007695 unsigned char bits;
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007696
7697 present = snd_hda_codec_read(codec, 0x1b, 0,
7698 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +02007699 bits = present ? HDA_AMP_MUTE : 0;
7700 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
7701 HDA_AMP_MUTE, bits);
7702 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
7703 HDA_AMP_MUTE, bits);
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007704}
7705
7706static void alc883_lenovo_101e_unsol_event(struct hda_codec *codec,
7707 unsigned int res)
7708{
7709 if ((res >> 26) == ALC880_HP_EVENT)
7710 alc883_lenovo_101e_all_automute(codec);
7711 if ((res >> 26) == ALC880_FRONT_EVENT)
7712 alc883_lenovo_101e_ispeaker_automute(codec);
7713}
7714
Takashi Iwai676a9b52007-08-16 15:23:35 +02007715/* toggle speaker-output according to the hp-jack state */
7716static void alc883_acer_aspire_automute(struct hda_codec *codec)
7717{
7718 unsigned int present;
7719
7720 present = snd_hda_codec_read(codec, 0x14, 0,
7721 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
7722 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
7723 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
7724 snd_hda_codec_amp_stereo(codec, 0x16, HDA_OUTPUT, 0,
7725 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
7726}
7727
7728static void alc883_acer_aspire_unsol_event(struct hda_codec *codec,
7729 unsigned int res)
7730{
7731 if ((res >> 26) == ALC880_HP_EVENT)
7732 alc883_acer_aspire_automute(codec);
7733}
7734
Kailang Yangd1a991a2007-08-15 16:21:59 +02007735static struct hda_verb alc883_acer_eapd_verbs[] = {
7736 /* HP Pin: output 0 (0x0c) */
7737 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
7738 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7739 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
7740 /* Front Pin: output 0 (0x0c) */
Takashi Iwai676a9b52007-08-16 15:23:35 +02007741 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7742 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
Kailang Yangd1a991a2007-08-15 16:21:59 +02007743 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Kailang Yangd1a991a2007-08-15 16:21:59 +02007744 {0x16, AC_VERB_SET_CONNECT_SEL, 0x00},
7745 /* eanable EAPD on medion laptop */
7746 {0x20, AC_VERB_SET_COEF_INDEX, 0x07},
7747 {0x20, AC_VERB_SET_PROC_COEF, 0x3050},
Takashi Iwai676a9b52007-08-16 15:23:35 +02007748 /* enable unsolicited event */
7749 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
Kailang Yangd1a991a2007-08-15 16:21:59 +02007750 { }
7751};
7752
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01007753static void alc888_6st_dell_front_automute(struct hda_codec *codec)
7754{
7755 unsigned int present;
7756
7757 present = snd_hda_codec_read(codec, 0x1b, 0,
7758 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
7759 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
7760 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
7761 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
7762 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
7763 snd_hda_codec_amp_stereo(codec, 0x16, HDA_OUTPUT, 0,
7764 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
7765 snd_hda_codec_amp_stereo(codec, 0x17, HDA_OUTPUT, 0,
7766 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
7767}
7768
7769static void alc888_6st_dell_unsol_event(struct hda_codec *codec,
7770 unsigned int res)
7771{
7772 switch (res >> 26) {
7773 case ALC880_HP_EVENT:
7774 printk("hp_event\n");
7775 alc888_6st_dell_front_automute(codec);
7776 break;
7777 }
7778}
7779
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007780/*
7781 * generic initialization of ADC, input mixers and output mixers
7782 */
7783static struct hda_verb alc883_auto_init_verbs[] = {
7784 /*
7785 * Unmute ADC0-2 and set the default input to mic-in
7786 */
7787 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
7788 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7789 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
7790 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7791
Takashi Iwaicb53c622007-08-10 17:21:45 +02007792 /* Mute input amps (CD, Line In, Mic 1 & Mic 2) of the analog-loopback
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007793 * mixer widget
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02007794 * Note: PASD motherboards uses the Line In 2 as the input for
7795 * front panel mic (mic 2)
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007796 */
7797 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +02007798 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7799 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
7800 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
7801 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
7802 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007803
7804 /*
7805 * Set up output mixers (0x0c - 0x0f)
7806 */
7807 /* set vol=0 to output mixers */
7808 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7809 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7810 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7811 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7812 /* set up input amps for analog loopback */
7813 /* Amp Indices: DAC = 0, mixer = 1 */
7814 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7815 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7816 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7817 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7818 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7819 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7820 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7821 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7822 {0x26, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7823 {0x26, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7824
7825 /* FIXME: use matrix-type input source selection */
7826 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
7827 /* Input mixer1 */
7828 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7829 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7830 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02007831 /* {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(3)}, */
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007832 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(4)},
7833 /* Input mixer2 */
7834 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7835 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7836 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02007837 /* {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(3)}, */
Andy Shevchenkoe3cde642007-12-03 16:50:58 +01007838 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(4)},
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007839
7840 { }
7841};
7842
7843/* capture mixer elements */
7844static struct snd_kcontrol_new alc883_capture_mixer[] = {
7845 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
7846 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
7847 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x09, 0x0, HDA_INPUT),
7848 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x09, 0x0, HDA_INPUT),
7849 {
7850 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
7851 /* The multiple "Capture Source" controls confuse alsamixer
7852 * So call somewhat different..
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007853 */
7854 /* .name = "Capture Source", */
7855 .name = "Input Source",
7856 .count = 2,
7857 .info = alc882_mux_enum_info,
7858 .get = alc882_mux_enum_get,
7859 .put = alc882_mux_enum_put,
7860 },
7861 { } /* end */
7862};
7863
Takashi Iwaicb53c622007-08-10 17:21:45 +02007864#ifdef CONFIG_SND_HDA_POWER_SAVE
7865#define alc883_loopbacks alc880_loopbacks
7866#endif
7867
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007868/* pcm configuration: identiacal with ALC880 */
7869#define alc883_pcm_analog_playback alc880_pcm_analog_playback
7870#define alc883_pcm_analog_capture alc880_pcm_analog_capture
Takashi Iwai63300792008-01-24 15:31:36 +01007871#define alc883_pcm_analog_alt_capture alc880_pcm_analog_alt_capture
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007872#define alc883_pcm_digital_playback alc880_pcm_digital_playback
7873#define alc883_pcm_digital_capture alc880_pcm_digital_capture
7874
7875/*
7876 * configuration and preset
7877 */
Takashi Iwaif5fcc132006-11-24 17:07:44 +01007878static const char *alc883_models[ALC883_MODEL_LAST] = {
7879 [ALC883_3ST_2ch_DIG] = "3stack-dig",
7880 [ALC883_3ST_6ch_DIG] = "3stack-6ch-dig",
7881 [ALC883_3ST_6ch] = "3stack-6ch",
7882 [ALC883_6ST_DIG] = "6stack-dig",
7883 [ALC883_TARGA_DIG] = "targa-dig",
7884 [ALC883_TARGA_2ch_DIG] = "targa-2ch-dig",
Takashi Iwaif5fcc132006-11-24 17:07:44 +01007885 [ALC883_ACER] = "acer",
Tobin Davis2880a862007-08-07 11:50:26 +02007886 [ALC883_ACER_ASPIRE] = "acer-aspire",
Takashi Iwaif5fcc132006-11-24 17:07:44 +01007887 [ALC883_MEDION] = "medion",
Kailang Yang272a5272007-05-14 11:00:38 +02007888 [ALC883_MEDION_MD2] = "medion-md2",
Takashi Iwaif5fcc132006-11-24 17:07:44 +01007889 [ALC883_LAPTOP_EAPD] = "laptop-eapd",
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007890 [ALC883_LENOVO_101E_2ch] = "lenovo-101e",
Kailang Yang272a5272007-05-14 11:00:38 +02007891 [ALC883_LENOVO_NB0763] = "lenovo-nb0763",
7892 [ALC888_LENOVO_MS7195_DIG] = "lenovo-ms7195-dig",
Kailang Yang189609a2007-08-20 11:31:23 +02007893 [ALC883_HAIER_W66] = "haier-w66",
Claudio Matsuoka4723c022007-07-13 14:36:19 +02007894 [ALC888_3ST_HP] = "3stack-hp",
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01007895 [ALC888_6ST_DELL] = "6stack-dell",
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +01007896 [ALC883_MITAC] = "mitac",
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01007897 [ALC883_CLEVO_M720] = "clevo-m720",
Jiang zhefb97dc62008-03-06 11:07:11 +01007898 [ALC883_FUJITSU_PI2515] = "fujitsu-pi2515",
Jiang zhe17bba1b2008-06-04 12:11:07 +02007899 [ALC883_3ST_6ch_INTEL] = "3stack-6ch-intel",
Takashi Iwaif5fcc132006-11-24 17:07:44 +01007900 [ALC883_AUTO] = "auto",
7901};
7902
7903static struct snd_pci_quirk alc883_cfg_tbl[] = {
7904 SND_PCI_QUIRK(0x1019, 0x6668, "ECS", ALC883_3ST_6ch_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01007905 SND_PCI_QUIRK(0x1025, 0x006c, "Acer Aspire 9810", ALC883_ACER_ASPIRE),
7906 SND_PCI_QUIRK(0x1025, 0x0110, "Acer Aspire", ALC883_ACER_ASPIRE),
7907 SND_PCI_QUIRK(0x1025, 0x0112, "Acer Aspire 9303", ALC883_ACER_ASPIRE),
7908 SND_PCI_QUIRK(0x1025, 0, "Acer laptop", ALC883_ACER), /* default Acer */
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01007909 SND_PCI_QUIRK(0x1028, 0x020d, "Dell Inspiron 530", ALC888_6ST_DELL),
Tobin Davisfebe3372007-06-12 11:27:46 +02007910 SND_PCI_QUIRK(0x103c, 0x2a3d, "HP Pavillion", ALC883_6ST_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01007911 SND_PCI_QUIRK(0x103c, 0x2a4f, "HP Samba", ALC888_3ST_HP),
7912 SND_PCI_QUIRK(0x103c, 0x2a60, "HP Lucknow", ALC888_3ST_HP),
Herton Ronaldo Krzesinski5d85f8d2008-03-20 12:13:46 +01007913 SND_PCI_QUIRK(0x103c, 0x2a61, "HP Nettle", ALC883_6ST_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01007914 SND_PCI_QUIRK(0x1043, 0x8249, "Asus M2A-VM HDMI", ALC883_3ST_6ch_DIG),
Travis Place97ec7102008-05-23 18:31:46 +02007915 SND_PCI_QUIRK(0x105b, 0x0ce8, "Foxconn P35AX-S", ALC883_6ST_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01007916 SND_PCI_QUIRK(0x105b, 0x6668, "Foxconn", ALC883_6ST_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01007917 SND_PCI_QUIRK(0x1071, 0x8253, "Mitac 8252d", ALC883_MITAC),
7918 SND_PCI_QUIRK(0x1071, 0x8258, "Evesham Voyaeger", ALC883_LAPTOP_EAPD),
7919 SND_PCI_QUIRK(0x108e, 0x534d, NULL, ALC883_3ST_6ch),
Tobin Davis57b14f22007-04-18 23:05:36 +02007920 SND_PCI_QUIRK(0x1458, 0xa002, "MSI", ALC883_6ST_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01007921 SND_PCI_QUIRK(0x1462, 0x0349, "MSI", ALC883_TARGA_2ch_DIG),
7922 SND_PCI_QUIRK(0x1462, 0x040d, "MSI", ALC883_TARGA_2ch_DIG),
7923 SND_PCI_QUIRK(0x1462, 0x0579, "MSI", ALC883_TARGA_2ch_DIG),
Jiang zhe4383fae2008-04-14 12:58:57 +02007924 SND_PCI_QUIRK(0x1462, 0x2fb3, "MSI", ALC883_TARGA_2ch_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01007925 SND_PCI_QUIRK(0x1462, 0x3729, "MSI S420", ALC883_TARGA_DIG),
Pascal Terjan82808232008-03-04 11:33:28 +01007926 SND_PCI_QUIRK(0x1462, 0x3783, "NEC S970", ALC883_TARGA_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01007927 SND_PCI_QUIRK(0x1462, 0x3b7f, "MSI", ALC883_TARGA_2ch_DIG),
7928 SND_PCI_QUIRK(0x1462, 0x3ef9, "MSI", ALC883_TARGA_DIG),
7929 SND_PCI_QUIRK(0x1462, 0x3fc1, "MSI", ALC883_TARGA_DIG),
7930 SND_PCI_QUIRK(0x1462, 0x3fc3, "MSI", ALC883_TARGA_DIG),
7931 SND_PCI_QUIRK(0x1462, 0x3fcc, "MSI", ALC883_TARGA_DIG),
7932 SND_PCI_QUIRK(0x1462, 0x3fdf, "MSI", ALC883_TARGA_DIG),
7933 SND_PCI_QUIRK(0x1462, 0x4314, "MSI", ALC883_TARGA_DIG),
7934 SND_PCI_QUIRK(0x1462, 0x4319, "MSI", ALC883_TARGA_DIG),
7935 SND_PCI_QUIRK(0x1462, 0x4324, "MSI", ALC883_TARGA_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01007936 SND_PCI_QUIRK(0x1462, 0x6668, "MSI", ALC883_6ST_DIG),
7937 SND_PCI_QUIRK(0x1462, 0x7187, "MSI", ALC883_6ST_DIG),
Tobin Davis799f88a2007-05-29 14:29:08 +02007938 SND_PCI_QUIRK(0x1462, 0x7250, "MSI", ALC883_6ST_DIG),
Herton Ronaldo Krzesinski86d34b72008-03-18 09:27:59 +01007939 SND_PCI_QUIRK(0x1462, 0x7267, "MSI", ALC883_3ST_6ch_DIG),
Leonard Norrgard2a296cb2007-01-08 11:28:22 +01007940 SND_PCI_QUIRK(0x1462, 0x7280, "MSI", ALC883_6ST_DIG),
Tobin Davis2dcd5222007-07-10 17:04:57 +02007941 SND_PCI_QUIRK(0x1462, 0x7327, "MSI", ALC883_6ST_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01007942 SND_PCI_QUIRK(0x1462, 0xa422, "MSI", ALC883_TARGA_2ch_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01007943 SND_PCI_QUIRK(0x147b, 0x1083, "Abit IP35-PRO", ALC883_6ST_DIG),
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01007944 SND_PCI_QUIRK(0x1558, 0x0721, "Clevo laptop M720R", ALC883_CLEVO_M720),
7945 SND_PCI_QUIRK(0x1558, 0x0722, "Clevo laptop M720SR", ALC883_CLEVO_M720),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01007946 SND_PCI_QUIRK(0x1558, 0, "Clevo laptop", ALC883_LAPTOP_EAPD),
Daniel T Chene60623b2007-05-29 03:41:55 -04007947 SND_PCI_QUIRK(0x15d9, 0x8780, "Supermicro PDSBA", ALC883_3ST_6ch),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01007948 SND_PCI_QUIRK(0x161f, 0x2054, "Medion laptop", ALC883_MEDION),
Jiang zhefb97dc62008-03-06 11:07:11 +01007949 SND_PCI_QUIRK(0x1734, 0x1108, "Fujitsu AMILO Pi2515", ALC883_FUJITSU_PI2515),
Kailang Yang272a5272007-05-14 11:00:38 +02007950 SND_PCI_QUIRK(0x17aa, 0x101e, "Lenovo 101e", ALC883_LENOVO_101E_2ch),
Kailang Yang272a5272007-05-14 11:00:38 +02007951 SND_PCI_QUIRK(0x17aa, 0x2085, "Lenovo NB0763", ALC883_LENOVO_NB0763),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01007952 SND_PCI_QUIRK(0x17aa, 0x3bfc, "Lenovo NB0763", ALC883_LENOVO_NB0763),
7953 SND_PCI_QUIRK(0x17aa, 0x3bfd, "Lenovo NB0763", ALC883_LENOVO_NB0763),
Kailang Yang272a5272007-05-14 11:00:38 +02007954 SND_PCI_QUIRK(0x17c0, 0x4071, "MEDION MD2", ALC883_MEDION_MD2),
Andrew Paprocki0b167bf2008-02-03 10:15:44 +01007955 SND_PCI_QUIRK(0x17f2, 0x5000, "Albatron KI690-AM2", ALC883_6ST_DIG),
Kailang Yang189609a2007-08-20 11:31:23 +02007956 SND_PCI_QUIRK(0x1991, 0x5625, "Haier W66", ALC883_HAIER_W66),
Jiang zhe17bba1b2008-06-04 12:11:07 +02007957 SND_PCI_QUIRK(0x8086, 0x0001, "DG33BUC", ALC883_3ST_6ch_INTEL),
7958 SND_PCI_QUIRK(0x8086, 0x0002, "DG33FBC", ALC883_3ST_6ch_INTEL),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01007959 SND_PCI_QUIRK(0x8086, 0xd601, "D102GGC", ALC883_3ST_6ch),
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007960 {}
7961};
7962
7963static struct alc_config_preset alc883_presets[] = {
7964 [ALC883_3ST_2ch_DIG] = {
7965 .mixers = { alc883_3ST_2ch_mixer },
7966 .init_verbs = { alc883_init_verbs },
7967 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
7968 .dac_nids = alc883_dac_nids,
7969 .dig_out_nid = ALC883_DIGOUT_NID,
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007970 .dig_in_nid = ALC883_DIGIN_NID,
7971 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
7972 .channel_mode = alc883_3ST_2ch_modes,
7973 .input_mux = &alc883_capture_source,
7974 },
7975 [ALC883_3ST_6ch_DIG] = {
7976 .mixers = { alc883_3ST_6ch_mixer, alc883_chmode_mixer },
7977 .init_verbs = { alc883_init_verbs },
7978 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
7979 .dac_nids = alc883_dac_nids,
7980 .dig_out_nid = ALC883_DIGOUT_NID,
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007981 .dig_in_nid = ALC883_DIGIN_NID,
7982 .num_channel_mode = ARRAY_SIZE(alc883_3ST_6ch_modes),
7983 .channel_mode = alc883_3ST_6ch_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02007984 .need_dac_fix = 1,
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007985 .input_mux = &alc883_capture_source,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02007986 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007987 [ALC883_3ST_6ch] = {
7988 .mixers = { alc883_3ST_6ch_mixer, alc883_chmode_mixer },
7989 .init_verbs = { alc883_init_verbs },
7990 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
7991 .dac_nids = alc883_dac_nids,
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007992 .num_channel_mode = ARRAY_SIZE(alc883_3ST_6ch_modes),
7993 .channel_mode = alc883_3ST_6ch_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02007994 .need_dac_fix = 1,
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007995 .input_mux = &alc883_capture_source,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02007996 },
Jiang zhe17bba1b2008-06-04 12:11:07 +02007997 [ALC883_3ST_6ch_INTEL] = {
7998 .mixers = { alc883_3ST_6ch_intel_mixer, alc883_chmode_mixer },
7999 .init_verbs = { alc883_init_verbs },
8000 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
8001 .dac_nids = alc883_dac_nids,
8002 .dig_out_nid = ALC883_DIGOUT_NID,
8003 .dig_in_nid = ALC883_DIGIN_NID,
8004 .num_channel_mode = ARRAY_SIZE(alc883_3ST_6ch_intel_modes),
8005 .channel_mode = alc883_3ST_6ch_intel_modes,
8006 .need_dac_fix = 1,
8007 .input_mux = &alc883_3stack_6ch_intel,
8008 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008009 [ALC883_6ST_DIG] = {
8010 .mixers = { alc883_base_mixer, alc883_chmode_mixer },
8011 .init_verbs = { alc883_init_verbs },
8012 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
8013 .dac_nids = alc883_dac_nids,
8014 .dig_out_nid = ALC883_DIGOUT_NID,
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008015 .dig_in_nid = ALC883_DIGIN_NID,
8016 .num_channel_mode = ARRAY_SIZE(alc883_sixstack_modes),
8017 .channel_mode = alc883_sixstack_modes,
8018 .input_mux = &alc883_capture_source,
8019 },
Kailang Yangccc656c2006-10-17 12:32:26 +02008020 [ALC883_TARGA_DIG] = {
8021 .mixers = { alc883_tagra_mixer, alc883_chmode_mixer },
8022 .init_verbs = { alc883_init_verbs, alc883_tagra_verbs},
8023 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
8024 .dac_nids = alc883_dac_nids,
8025 .dig_out_nid = ALC883_DIGOUT_NID,
Kailang Yangccc656c2006-10-17 12:32:26 +02008026 .num_channel_mode = ARRAY_SIZE(alc883_3ST_6ch_modes),
8027 .channel_mode = alc883_3ST_6ch_modes,
8028 .need_dac_fix = 1,
8029 .input_mux = &alc883_capture_source,
8030 .unsol_event = alc883_tagra_unsol_event,
8031 .init_hook = alc883_tagra_automute,
8032 },
8033 [ALC883_TARGA_2ch_DIG] = {
8034 .mixers = { alc883_tagra_2ch_mixer},
8035 .init_verbs = { alc883_init_verbs, alc883_tagra_verbs},
8036 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
8037 .dac_nids = alc883_dac_nids,
8038 .dig_out_nid = ALC883_DIGOUT_NID,
Kailang Yangccc656c2006-10-17 12:32:26 +02008039 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
8040 .channel_mode = alc883_3ST_2ch_modes,
8041 .input_mux = &alc883_capture_source,
8042 .unsol_event = alc883_tagra_unsol_event,
8043 .init_hook = alc883_tagra_automute,
8044 },
Vladimir Avdoninbab282b2006-08-22 13:31:58 +02008045 [ALC883_ACER] = {
Takashi Iwai676a9b52007-08-16 15:23:35 +02008046 .mixers = { alc883_base_mixer },
Vladimir Avdoninbab282b2006-08-22 13:31:58 +02008047 /* On TravelMate laptops, GPIO 0 enables the internal speaker
8048 * and the headphone jack. Turn this on and rely on the
8049 * standard mute methods whenever the user wants to turn
8050 * these outputs off.
8051 */
8052 .init_verbs = { alc883_init_verbs, alc880_gpio1_init_verbs },
8053 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
8054 .dac_nids = alc883_dac_nids,
Vladimir Avdoninbab282b2006-08-22 13:31:58 +02008055 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
8056 .channel_mode = alc883_3ST_2ch_modes,
8057 .input_mux = &alc883_capture_source,
8058 },
Tobin Davis2880a862007-08-07 11:50:26 +02008059 [ALC883_ACER_ASPIRE] = {
Takashi Iwai676a9b52007-08-16 15:23:35 +02008060 .mixers = { alc883_acer_aspire_mixer },
Kailang Yangd1a991a2007-08-15 16:21:59 +02008061 .init_verbs = { alc883_init_verbs, alc883_acer_eapd_verbs },
Tobin Davis2880a862007-08-07 11:50:26 +02008062 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
8063 .dac_nids = alc883_dac_nids,
8064 .dig_out_nid = ALC883_DIGOUT_NID,
Tobin Davis2880a862007-08-07 11:50:26 +02008065 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
8066 .channel_mode = alc883_3ST_2ch_modes,
8067 .input_mux = &alc883_capture_source,
Takashi Iwai676a9b52007-08-16 15:23:35 +02008068 .unsol_event = alc883_acer_aspire_unsol_event,
8069 .init_hook = alc883_acer_aspire_automute,
Kailang Yangd1a991a2007-08-15 16:21:59 +02008070 },
Tobin Davisc07584c2006-10-13 12:32:16 +02008071 [ALC883_MEDION] = {
8072 .mixers = { alc883_fivestack_mixer,
8073 alc883_chmode_mixer },
8074 .init_verbs = { alc883_init_verbs,
Andrew L. Neporadab373bde2006-11-07 11:37:08 +01008075 alc883_medion_eapd_verbs },
Tobin Davisc07584c2006-10-13 12:32:16 +02008076 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
8077 .dac_nids = alc883_dac_nids,
Tobin Davisc07584c2006-10-13 12:32:16 +02008078 .num_channel_mode = ARRAY_SIZE(alc883_sixstack_modes),
8079 .channel_mode = alc883_sixstack_modes,
8080 .input_mux = &alc883_capture_source,
Andrew L. Neporadab373bde2006-11-07 11:37:08 +01008081 },
Kailang Yang272a5272007-05-14 11:00:38 +02008082 [ALC883_MEDION_MD2] = {
8083 .mixers = { alc883_medion_md2_mixer},
8084 .init_verbs = { alc883_init_verbs, alc883_medion_md2_verbs},
8085 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
8086 .dac_nids = alc883_dac_nids,
8087 .dig_out_nid = ALC883_DIGOUT_NID,
Kailang Yang272a5272007-05-14 11:00:38 +02008088 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
8089 .channel_mode = alc883_3ST_2ch_modes,
8090 .input_mux = &alc883_capture_source,
8091 .unsol_event = alc883_medion_md2_unsol_event,
8092 .init_hook = alc883_medion_md2_automute,
8093 },
Andrew L. Neporadab373bde2006-11-07 11:37:08 +01008094 [ALC883_LAPTOP_EAPD] = {
Takashi Iwai676a9b52007-08-16 15:23:35 +02008095 .mixers = { alc883_base_mixer },
Andrew L. Neporadab373bde2006-11-07 11:37:08 +01008096 .init_verbs = { alc883_init_verbs, alc882_eapd_verbs },
8097 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
8098 .dac_nids = alc883_dac_nids,
Andrew L. Neporadab373bde2006-11-07 11:37:08 +01008099 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
8100 .channel_mode = alc883_3ST_2ch_modes,
8101 .input_mux = &alc883_capture_source,
8102 },
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01008103 [ALC883_CLEVO_M720] = {
8104 .mixers = { alc883_clevo_m720_mixer },
8105 .init_verbs = { alc883_init_verbs, alc883_clevo_m720_verbs },
Jiang zhe368c7a92008-03-04 11:20:33 +01008106 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
8107 .dac_nids = alc883_dac_nids,
8108 .dig_out_nid = ALC883_DIGOUT_NID,
8109 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
8110 .channel_mode = alc883_3ST_2ch_modes,
8111 .input_mux = &alc883_capture_source,
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01008112 .unsol_event = alc883_clevo_m720_unsol_event,
8113 .init_hook = alc883_clevo_m720_automute,
Jiang zhe368c7a92008-03-04 11:20:33 +01008114 },
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008115 [ALC883_LENOVO_101E_2ch] = {
8116 .mixers = { alc883_lenovo_101e_2ch_mixer},
8117 .init_verbs = { alc883_init_verbs, alc883_lenovo_101e_verbs},
8118 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
8119 .dac_nids = alc883_dac_nids,
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008120 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
8121 .channel_mode = alc883_3ST_2ch_modes,
8122 .input_mux = &alc883_lenovo_101e_capture_source,
8123 .unsol_event = alc883_lenovo_101e_unsol_event,
8124 .init_hook = alc883_lenovo_101e_all_automute,
8125 },
Kailang Yang272a5272007-05-14 11:00:38 +02008126 [ALC883_LENOVO_NB0763] = {
8127 .mixers = { alc883_lenovo_nb0763_mixer },
8128 .init_verbs = { alc883_init_verbs, alc883_lenovo_nb0763_verbs},
8129 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
8130 .dac_nids = alc883_dac_nids,
Kailang Yang272a5272007-05-14 11:00:38 +02008131 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
8132 .channel_mode = alc883_3ST_2ch_modes,
8133 .need_dac_fix = 1,
8134 .input_mux = &alc883_lenovo_nb0763_capture_source,
8135 .unsol_event = alc883_medion_md2_unsol_event,
8136 .init_hook = alc883_medion_md2_automute,
8137 },
8138 [ALC888_LENOVO_MS7195_DIG] = {
8139 .mixers = { alc883_3ST_6ch_mixer, alc883_chmode_mixer },
8140 .init_verbs = { alc883_init_verbs, alc888_lenovo_ms7195_verbs},
8141 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
8142 .dac_nids = alc883_dac_nids,
8143 .dig_out_nid = ALC883_DIGOUT_NID,
Kailang Yang272a5272007-05-14 11:00:38 +02008144 .num_channel_mode = ARRAY_SIZE(alc883_3ST_6ch_modes),
8145 .channel_mode = alc883_3ST_6ch_modes,
8146 .need_dac_fix = 1,
8147 .input_mux = &alc883_capture_source,
8148 .unsol_event = alc883_lenovo_ms7195_unsol_event,
8149 .init_hook = alc888_lenovo_ms7195_front_automute,
Kailang Yang189609a2007-08-20 11:31:23 +02008150 },
8151 [ALC883_HAIER_W66] = {
8152 .mixers = { alc883_tagra_2ch_mixer},
8153 .init_verbs = { alc883_init_verbs, alc883_haier_w66_verbs},
8154 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
8155 .dac_nids = alc883_dac_nids,
8156 .dig_out_nid = ALC883_DIGOUT_NID,
Kailang Yang189609a2007-08-20 11:31:23 +02008157 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
8158 .channel_mode = alc883_3ST_2ch_modes,
8159 .input_mux = &alc883_capture_source,
8160 .unsol_event = alc883_haier_w66_unsol_event,
8161 .init_hook = alc883_haier_w66_automute,
Herton Ronaldo Krzesinskieea64192008-03-20 12:14:59 +01008162 },
Claudio Matsuoka4723c022007-07-13 14:36:19 +02008163 [ALC888_3ST_HP] = {
Herton Ronaldo Krzesinskieea64192008-03-20 12:14:59 +01008164 .mixers = { alc883_3ST_6ch_mixer, alc883_chmode_mixer },
Claudio Matsuoka4723c022007-07-13 14:36:19 +02008165 .init_verbs = { alc883_init_verbs, alc888_3st_hp_verbs },
Claudio Matsuoka8341de62007-07-06 12:10:45 +02008166 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
8167 .dac_nids = alc883_dac_nids,
Claudio Matsuoka4723c022007-07-13 14:36:19 +02008168 .num_channel_mode = ARRAY_SIZE(alc888_3st_hp_modes),
8169 .channel_mode = alc888_3st_hp_modes,
Claudio Matsuoka8341de62007-07-06 12:10:45 +02008170 .need_dac_fix = 1,
8171 .input_mux = &alc883_capture_source,
8172 },
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01008173 [ALC888_6ST_DELL] = {
Herton Ronaldo Krzesinskif24dbdc2008-03-20 12:14:28 +01008174 .mixers = { alc883_base_mixer, alc883_chmode_mixer },
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01008175 .init_verbs = { alc883_init_verbs, alc888_6st_dell_verbs },
8176 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
8177 .dac_nids = alc883_dac_nids,
8178 .dig_out_nid = ALC883_DIGOUT_NID,
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01008179 .dig_in_nid = ALC883_DIGIN_NID,
8180 .num_channel_mode = ARRAY_SIZE(alc883_sixstack_modes),
8181 .channel_mode = alc883_sixstack_modes,
8182 .input_mux = &alc883_capture_source,
8183 .unsol_event = alc888_6st_dell_unsol_event,
8184 .init_hook = alc888_6st_dell_front_automute,
8185 },
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +01008186 [ALC883_MITAC] = {
8187 .mixers = { alc883_mitac_mixer },
8188 .init_verbs = { alc883_init_verbs, alc883_mitac_verbs },
8189 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
8190 .dac_nids = alc883_dac_nids,
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +01008191 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
8192 .channel_mode = alc883_3ST_2ch_modes,
8193 .input_mux = &alc883_capture_source,
8194 .unsol_event = alc883_mitac_unsol_event,
8195 .init_hook = alc883_mitac_automute,
8196 },
Jiang zhefb97dc62008-03-06 11:07:11 +01008197 [ALC883_FUJITSU_PI2515] = {
8198 .mixers = { alc883_2ch_fujitsu_pi2515_mixer },
8199 .init_verbs = { alc883_init_verbs,
8200 alc883_2ch_fujitsu_pi2515_verbs},
8201 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
8202 .dac_nids = alc883_dac_nids,
8203 .dig_out_nid = ALC883_DIGOUT_NID,
8204 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
8205 .channel_mode = alc883_3ST_2ch_modes,
8206 .input_mux = &alc883_fujitsu_pi2515_capture_source,
8207 .unsol_event = alc883_2ch_fujitsu_pi2515_unsol_event,
8208 .init_hook = alc883_2ch_fujitsu_pi2515_automute,
8209 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008210};
8211
8212
8213/*
8214 * BIOS auto configuration
8215 */
8216static void alc883_auto_set_output_and_unmute(struct hda_codec *codec,
8217 hda_nid_t nid, int pin_type,
8218 int dac_idx)
8219{
8220 /* set as output */
8221 struct alc_spec *spec = codec->spec;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02008222 int idx;
8223
Takashi Iwaif6c7e542008-02-12 18:32:23 +01008224 alc_set_pin_output(codec, nid, pin_type);
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008225 if (spec->multiout.dac_nids[dac_idx] == 0x25)
8226 idx = 4;
8227 else
8228 idx = spec->multiout.dac_nids[dac_idx] - 2;
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008229 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_CONNECT_SEL, idx);
8230
8231}
8232
8233static void alc883_auto_init_multi_out(struct hda_codec *codec)
8234{
8235 struct alc_spec *spec = codec->spec;
8236 int i;
8237
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008238 alc_subsystem_id(codec, 0x15, 0x1b, 0x14);
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008239 for (i = 0; i <= HDA_SIDE; i++) {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02008240 hda_nid_t nid = spec->autocfg.line_out_pins[i];
Takashi Iwaibaba8ee2007-04-23 17:17:48 +02008241 int pin_type = get_pin_type(spec->autocfg.line_out_type);
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008242 if (nid)
Takashi Iwaibaba8ee2007-04-23 17:17:48 +02008243 alc883_auto_set_output_and_unmute(codec, nid, pin_type,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02008244 i);
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008245 }
8246}
8247
8248static void alc883_auto_init_hp_out(struct hda_codec *codec)
8249{
8250 struct alc_spec *spec = codec->spec;
8251 hda_nid_t pin;
8252
Takashi Iwaieb06ed82006-09-20 17:10:27 +02008253 pin = spec->autocfg.hp_pins[0];
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008254 if (pin) /* connect to front */
8255 /* use dac 0 */
8256 alc883_auto_set_output_and_unmute(codec, pin, PIN_HP, 0);
Takashi Iwaif6c7e542008-02-12 18:32:23 +01008257 pin = spec->autocfg.speaker_pins[0];
8258 if (pin)
8259 alc883_auto_set_output_and_unmute(codec, pin, PIN_OUT, 0);
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008260}
8261
8262#define alc883_is_input_pin(nid) alc880_is_input_pin(nid)
8263#define ALC883_PIN_CD_NID ALC880_PIN_CD_NID
8264
8265static void alc883_auto_init_analog_input(struct hda_codec *codec)
8266{
8267 struct alc_spec *spec = codec->spec;
8268 int i;
8269
8270 for (i = 0; i < AUTO_PIN_LAST; i++) {
8271 hda_nid_t nid = spec->autocfg.input_pins[i];
8272 if (alc883_is_input_pin(nid)) {
8273 snd_hda_codec_write(codec, nid, 0,
8274 AC_VERB_SET_PIN_WIDGET_CONTROL,
8275 (i <= AUTO_PIN_FRONT_MIC ?
8276 PIN_VREF80 : PIN_IN));
8277 if (nid != ALC883_PIN_CD_NID)
8278 snd_hda_codec_write(codec, nid, 0,
8279 AC_VERB_SET_AMP_GAIN_MUTE,
8280 AMP_OUT_MUTE);
8281 }
8282 }
8283}
8284
8285/* almost identical with ALC880 parser... */
8286static int alc883_parse_auto_config(struct hda_codec *codec)
8287{
8288 struct alc_spec *spec = codec->spec;
8289 int err = alc880_parse_auto_config(codec);
8290
8291 if (err < 0)
8292 return err;
Takashi Iwai776e1842007-08-29 15:07:11 +02008293 else if (!err)
8294 return 0; /* no config found */
8295
8296 err = alc_auto_add_mic_boost(codec);
8297 if (err < 0)
8298 return err;
8299
8300 /* hack - override the init verbs */
8301 spec->init_verbs[0] = alc883_auto_init_verbs;
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008302 spec->mixers[spec->num_mixers] = alc883_capture_mixer;
8303 spec->num_mixers++;
Takashi Iwai776e1842007-08-29 15:07:11 +02008304
8305 return 1; /* config found */
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008306}
8307
8308/* additional initialization for auto-configuration model */
8309static void alc883_auto_init(struct hda_codec *codec)
8310{
Takashi Iwaif6c7e542008-02-12 18:32:23 +01008311 struct alc_spec *spec = codec->spec;
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008312 alc883_auto_init_multi_out(codec);
8313 alc883_auto_init_hp_out(codec);
8314 alc883_auto_init_analog_input(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +01008315 if (spec->unsol_event)
8316 alc_sku_automute(codec);
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008317}
8318
8319static int patch_alc883(struct hda_codec *codec)
8320{
8321 struct alc_spec *spec;
8322 int err, board_config;
8323
8324 spec = kzalloc(sizeof(*spec), GFP_KERNEL);
8325 if (spec == NULL)
8326 return -ENOMEM;
8327
8328 codec->spec = spec;
8329
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +02008330 alc_fix_pll_init(codec, 0x20, 0x0a, 10);
8331
Takashi Iwaif5fcc132006-11-24 17:07:44 +01008332 board_config = snd_hda_check_board_config(codec, ALC883_MODEL_LAST,
8333 alc883_models,
8334 alc883_cfg_tbl);
8335 if (board_config < 0) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008336 printk(KERN_INFO "hda_codec: Unknown model for ALC883, "
8337 "trying auto-probe from BIOS...\n");
8338 board_config = ALC883_AUTO;
8339 }
8340
8341 if (board_config == ALC883_AUTO) {
8342 /* automatic parse from the BIOS config */
8343 err = alc883_parse_auto_config(codec);
8344 if (err < 0) {
8345 alc_free(codec);
8346 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02008347 } else if (!err) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008348 printk(KERN_INFO
8349 "hda_codec: Cannot set up configuration "
8350 "from BIOS. Using base mode...\n");
8351 board_config = ALC883_3ST_2ch_DIG;
8352 }
8353 }
8354
8355 if (board_config != ALC883_AUTO)
8356 setup_preset(spec, &alc883_presets[board_config]);
8357
Kailang Yang2f893282008-05-27 12:14:47 +02008358 switch (codec->vendor_id) {
8359 case 0x10ec0888:
8360 spec->stream_name_analog = "ALC888 Analog";
8361 spec->stream_name_digital = "ALC888 Digital";
8362 break;
8363 case 0x10ec0889:
8364 spec->stream_name_analog = "ALC889 Analog";
8365 spec->stream_name_digital = "ALC889 Digital";
8366 break;
8367 default:
8368 spec->stream_name_analog = "ALC883 Analog";
8369 spec->stream_name_digital = "ALC883 Digital";
8370 break;
8371 }
8372
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008373 spec->stream_analog_playback = &alc883_pcm_analog_playback;
8374 spec->stream_analog_capture = &alc883_pcm_analog_capture;
Takashi Iwai63300792008-01-24 15:31:36 +01008375 spec->stream_analog_alt_capture = &alc883_pcm_analog_alt_capture;
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008376
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008377 spec->stream_digital_playback = &alc883_pcm_digital_playback;
8378 spec->stream_digital_capture = &alc883_pcm_digital_capture;
8379
Takashi Iwaie1406342008-02-11 18:32:32 +01008380 spec->num_adc_nids = ARRAY_SIZE(alc883_adc_nids);
8381 spec->adc_nids = alc883_adc_nids;
8382 spec->capsrc_nids = alc883_capsrc_nids;
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008383
Takashi Iwai2134ea42008-01-10 16:53:55 +01008384 spec->vmaster_nid = 0x0c;
8385
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008386 codec->patch_ops = alc_patch_ops;
8387 if (board_config == ALC883_AUTO)
8388 spec->init_hook = alc883_auto_init;
Kailang Yangf9423e72008-05-27 12:32:25 +02008389 else if (codec->vendor_id == 0x10ec0888)
8390 spec->init_hook = alc888_coef_init;
8391
Takashi Iwaicb53c622007-08-10 17:21:45 +02008392#ifdef CONFIG_SND_HDA_POWER_SAVE
8393 if (!spec->loopback.amplist)
8394 spec->loopback.amplist = alc883_loopbacks;
8395#endif
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008396
8397 return 0;
8398}
8399
8400/*
Kailang Yangdf694da2005-12-05 19:42:22 +01008401 * ALC262 support
8402 */
8403
8404#define ALC262_DIGOUT_NID ALC880_DIGOUT_NID
8405#define ALC262_DIGIN_NID ALC880_DIGIN_NID
8406
8407#define alc262_dac_nids alc260_dac_nids
8408#define alc262_adc_nids alc882_adc_nids
8409#define alc262_adc_nids_alt alc882_adc_nids_alt
Takashi Iwai88c71a92008-02-14 17:27:17 +01008410#define alc262_capsrc_nids alc882_capsrc_nids
8411#define alc262_capsrc_nids_alt alc882_capsrc_nids_alt
Kailang Yangdf694da2005-12-05 19:42:22 +01008412
8413#define alc262_modes alc260_modes
Takashi Iwaic5f2ea02005-12-06 18:54:31 +01008414#define alc262_capture_source alc882_capture_source
Kailang Yangdf694da2005-12-05 19:42:22 +01008415
8416static struct snd_kcontrol_new alc262_base_mixer[] = {
8417 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
8418 HDA_CODEC_MUTE("Front Playback Switch", 0x14, 0x0, HDA_OUTPUT),
8419 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
8420 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
8421 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
8422 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
8423 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
8424 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Takashi Iwaicc69d122007-02-15 19:29:26 +01008425 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Kailang Yangdf694da2005-12-05 19:42:22 +01008426 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
8427 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
Takashi Iwaicc69d122007-02-15 19:29:26 +01008428 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
Kailang Yangdf694da2005-12-05 19:42:22 +01008429 /* HDA_CODEC_VOLUME("PC Beep Playback Volume", 0x0b, 0x05, HDA_INPUT),
Takashi Iwai31bffaa2008-02-27 16:10:44 +01008430 HDA_CODEC_MUTE("PC Beep Playback Switch", 0x0b, 0x05, HDA_INPUT), */
Kailang Yangdf694da2005-12-05 19:42:22 +01008431 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0D, 0x0, HDA_OUTPUT),
8432 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
8433 HDA_CODEC_VOLUME_MONO("Mono Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
8434 HDA_CODEC_MUTE_MONO("Mono Playback Switch", 0x16, 2, 0x0, HDA_OUTPUT),
Kailang Yangdf694da2005-12-05 19:42:22 +01008435 { } /* end */
Takashi Iwai834be882006-03-01 14:16:17 +01008436};
8437
Kailang Yangccc656c2006-10-17 12:32:26 +02008438static struct snd_kcontrol_new alc262_hippo1_mixer[] = {
8439 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
8440 HDA_CODEC_MUTE("Front Playback Switch", 0x14, 0x0, HDA_OUTPUT),
8441 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
8442 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
8443 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
8444 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
8445 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
8446 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Takashi Iwaicc69d122007-02-15 19:29:26 +01008447 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Kailang Yangccc656c2006-10-17 12:32:26 +02008448 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
8449 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
Takashi Iwaicc69d122007-02-15 19:29:26 +01008450 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
Kailang Yangccc656c2006-10-17 12:32:26 +02008451 /* HDA_CODEC_VOLUME("PC Beep Playback Volume", 0x0b, 0x05, HDA_INPUT),
Takashi Iwai31bffaa2008-02-27 16:10:44 +01008452 HDA_CODEC_MUTE("PC Beep Playback Switch", 0x0b, 0x05, HDA_INPUT), */
Kailang Yangccc656c2006-10-17 12:32:26 +02008453 /*HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0D, 0x0, HDA_OUTPUT),*/
8454 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
8455 { } /* end */
8456};
8457
Takashi Iwaice875f02008-01-28 18:17:43 +01008458/* update HP, line and mono-out pins according to the master switch */
8459static void alc262_hp_master_update(struct hda_codec *codec)
8460{
8461 struct alc_spec *spec = codec->spec;
8462 int val = spec->master_sw;
8463
8464 /* HP & line-out */
8465 snd_hda_codec_write_cache(codec, 0x1b, 0,
8466 AC_VERB_SET_PIN_WIDGET_CONTROL,
8467 val ? PIN_HP : 0);
8468 snd_hda_codec_write_cache(codec, 0x15, 0,
8469 AC_VERB_SET_PIN_WIDGET_CONTROL,
8470 val ? PIN_HP : 0);
8471 /* mono (speaker) depending on the HP jack sense */
8472 val = val && !spec->jack_present;
8473 snd_hda_codec_write_cache(codec, 0x16, 0,
8474 AC_VERB_SET_PIN_WIDGET_CONTROL,
8475 val ? PIN_OUT : 0);
8476}
8477
8478static void alc262_hp_bpc_automute(struct hda_codec *codec)
8479{
8480 struct alc_spec *spec = codec->spec;
8481 unsigned int presence;
8482 presence = snd_hda_codec_read(codec, 0x1b, 0,
8483 AC_VERB_GET_PIN_SENSE, 0);
8484 spec->jack_present = !!(presence & AC_PINSENSE_PRESENCE);
8485 alc262_hp_master_update(codec);
8486}
8487
8488static void alc262_hp_bpc_unsol_event(struct hda_codec *codec, unsigned int res)
8489{
8490 if ((res >> 26) != ALC880_HP_EVENT)
8491 return;
8492 alc262_hp_bpc_automute(codec);
8493}
8494
8495static void alc262_hp_wildwest_automute(struct hda_codec *codec)
8496{
8497 struct alc_spec *spec = codec->spec;
8498 unsigned int presence;
8499 presence = snd_hda_codec_read(codec, 0x15, 0,
8500 AC_VERB_GET_PIN_SENSE, 0);
8501 spec->jack_present = !!(presence & AC_PINSENSE_PRESENCE);
8502 alc262_hp_master_update(codec);
8503}
8504
8505static void alc262_hp_wildwest_unsol_event(struct hda_codec *codec,
8506 unsigned int res)
8507{
8508 if ((res >> 26) != ALC880_HP_EVENT)
8509 return;
8510 alc262_hp_wildwest_automute(codec);
8511}
8512
8513static int alc262_hp_master_sw_get(struct snd_kcontrol *kcontrol,
8514 struct snd_ctl_elem_value *ucontrol)
8515{
8516 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
8517 struct alc_spec *spec = codec->spec;
8518 *ucontrol->value.integer.value = spec->master_sw;
8519 return 0;
8520}
8521
8522static int alc262_hp_master_sw_put(struct snd_kcontrol *kcontrol,
8523 struct snd_ctl_elem_value *ucontrol)
8524{
8525 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
8526 struct alc_spec *spec = codec->spec;
8527 int val = !!*ucontrol->value.integer.value;
8528
8529 if (val == spec->master_sw)
8530 return 0;
8531 spec->master_sw = val;
8532 alc262_hp_master_update(codec);
8533 return 1;
8534}
8535
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008536static struct snd_kcontrol_new alc262_HP_BPC_mixer[] = {
Takashi Iwaice875f02008-01-28 18:17:43 +01008537 {
8538 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
8539 .name = "Master Playback Switch",
8540 .info = snd_ctl_boolean_mono_info,
8541 .get = alc262_hp_master_sw_get,
8542 .put = alc262_hp_master_sw_put,
8543 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008544 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
8545 HDA_CODEC_MUTE("Front Playback Switch", 0x15, 0x0, HDA_OUTPUT),
8546 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
Takashi Iwaice875f02008-01-28 18:17:43 +01008547 HDA_CODEC_VOLUME_MONO("Speaker Playback Volume", 0x0e, 2, 0x0,
8548 HDA_OUTPUT),
8549 HDA_CODEC_MUTE_MONO("Speaker Playback Switch", 0x16, 2, 0x0,
8550 HDA_OUTPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008551 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
8552 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Takashi Iwaicc69d122007-02-15 19:29:26 +01008553 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008554 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
8555 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
Takashi Iwaicc69d122007-02-15 19:29:26 +01008556 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008557 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
8558 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
8559 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
8560 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
8561 HDA_CODEC_VOLUME("PC Beep Playback Volume", 0x0b, 0x05, HDA_INPUT),
8562 HDA_CODEC_MUTE("PC Beep Playback Switch", 0x0b, 0x05, HDA_INPUT),
8563 HDA_CODEC_VOLUME("AUX IN Playback Volume", 0x0b, 0x06, HDA_INPUT),
8564 HDA_CODEC_MUTE("AUX IN Playback Switch", 0x0b, 0x06, HDA_INPUT),
8565 { } /* end */
8566};
8567
Kailang Yangcd7509a2007-01-26 18:33:17 +01008568static struct snd_kcontrol_new alc262_HP_BPC_WildWest_mixer[] = {
Takashi Iwaice875f02008-01-28 18:17:43 +01008569 {
8570 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
8571 .name = "Master Playback Switch",
8572 .info = snd_ctl_boolean_mono_info,
8573 .get = alc262_hp_master_sw_get,
8574 .put = alc262_hp_master_sw_put,
8575 },
Kailang Yangcd7509a2007-01-26 18:33:17 +01008576 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
8577 HDA_CODEC_MUTE("Front Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
8578 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
8579 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
Takashi Iwaice875f02008-01-28 18:17:43 +01008580 HDA_CODEC_VOLUME_MONO("Speaker Playback Volume", 0x0e, 2, 0x0,
8581 HDA_OUTPUT),
8582 HDA_CODEC_MUTE_MONO("Speaker Playback Switch", 0x16, 2, 0x0,
8583 HDA_OUTPUT),
Kailang Yangcd7509a2007-01-26 18:33:17 +01008584 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x02, HDA_INPUT),
8585 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x02, HDA_INPUT),
Takashi Iwaicc69d122007-02-15 19:29:26 +01008586 HDA_CODEC_VOLUME("Front Mic Boost", 0x1a, 0, HDA_INPUT),
Kailang Yangcd7509a2007-01-26 18:33:17 +01008587 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x01, HDA_INPUT),
8588 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x01, HDA_INPUT),
8589 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
8590 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
8591 HDA_CODEC_VOLUME("PC Beep Playback Volume", 0x0b, 0x05, HDA_INPUT),
8592 HDA_CODEC_MUTE("PC Beep Playback Switch", 0x0b, 0x05, HDA_INPUT),
8593 { } /* end */
8594};
8595
8596static struct snd_kcontrol_new alc262_HP_BPC_WildWest_option_mixer[] = {
8597 HDA_CODEC_VOLUME("Rear Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
8598 HDA_CODEC_MUTE("Rear Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Takashi Iwaicc69d122007-02-15 19:29:26 +01008599 HDA_CODEC_VOLUME("Rear Mic Boost", 0x18, 0, HDA_INPUT),
Kailang Yangcd7509a2007-01-26 18:33:17 +01008600 { } /* end */
8601};
8602
Kailang Yang66d2a9d2007-11-14 12:06:21 +01008603/* mute/unmute internal speaker according to the hp jack and mute state */
8604static void alc262_hp_t5735_automute(struct hda_codec *codec, int force)
8605{
8606 struct alc_spec *spec = codec->spec;
Kailang Yang66d2a9d2007-11-14 12:06:21 +01008607
8608 if (force || !spec->sense_updated) {
8609 unsigned int present;
8610 present = snd_hda_codec_read(codec, 0x15, 0,
8611 AC_VERB_GET_PIN_SENSE, 0);
Takashi Iwai4bb26132008-01-28 18:12:42 +01008612 spec->jack_present = (present & AC_PINSENSE_PRESENCE) != 0;
Kailang Yang66d2a9d2007-11-14 12:06:21 +01008613 spec->sense_updated = 1;
8614 }
Takashi Iwai4bb26132008-01-28 18:12:42 +01008615 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_OUTPUT, 0, HDA_AMP_MUTE,
8616 spec->jack_present ? HDA_AMP_MUTE : 0);
Kailang Yang66d2a9d2007-11-14 12:06:21 +01008617}
8618
8619static void alc262_hp_t5735_unsol_event(struct hda_codec *codec,
8620 unsigned int res)
8621{
8622 if ((res >> 26) != ALC880_HP_EVENT)
8623 return;
8624 alc262_hp_t5735_automute(codec, 1);
8625}
8626
8627static void alc262_hp_t5735_init_hook(struct hda_codec *codec)
8628{
8629 alc262_hp_t5735_automute(codec, 1);
8630}
8631
8632static struct snd_kcontrol_new alc262_hp_t5735_mixer[] = {
Takashi Iwai86cd9292008-01-28 18:09:56 +01008633 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
8634 HDA_CODEC_MUTE("Speaker Playback Switch", 0x14, 0x0, HDA_OUTPUT),
Kailang Yang66d2a9d2007-11-14 12:06:21 +01008635 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
8636 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
8637 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
8638 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
8639 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
8640 { } /* end */
8641};
8642
8643static struct hda_verb alc262_hp_t5735_verbs[] = {
8644 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
8645 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
8646
8647 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
8648 { }
8649};
8650
Kailang Yang8c427222008-01-10 13:03:59 +01008651static struct snd_kcontrol_new alc262_hp_rp5700_mixer[] = {
Takashi Iwaif2f48e12008-01-28 18:14:43 +01008652 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
8653 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
Takashi Iwai86cd9292008-01-28 18:09:56 +01008654 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0e, 0x0, HDA_OUTPUT),
8655 HDA_CODEC_MUTE("Speaker Playback Switch", 0x16, 0x0, HDA_OUTPUT),
Kailang Yang8c427222008-01-10 13:03:59 +01008656 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x01, HDA_INPUT),
8657 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x01, HDA_INPUT),
8658 { } /* end */
8659};
8660
8661static struct hda_verb alc262_hp_rp5700_verbs[] = {
8662 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
8663 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
8664 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
8665 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
8666 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
8667 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
8668 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
8669 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
8670 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x00 << 8))},
8671 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x00 << 8))},
8672 {}
8673};
8674
8675static struct hda_input_mux alc262_hp_rp5700_capture_source = {
8676 .num_items = 1,
8677 .items = {
8678 { "Line", 0x1 },
8679 },
8680};
8681
Takashi Iwai0724ea22007-08-23 00:31:43 +02008682/* bind hp and internal speaker mute (with plug check) */
8683static int alc262_sony_master_sw_put(struct snd_kcontrol *kcontrol,
8684 struct snd_ctl_elem_value *ucontrol)
8685{
8686 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
8687 long *valp = ucontrol->value.integer.value;
8688 int change;
8689
8690 /* change hp mute */
8691 change = snd_hda_codec_amp_update(codec, 0x15, 0, HDA_OUTPUT, 0,
8692 HDA_AMP_MUTE,
8693 valp[0] ? 0 : HDA_AMP_MUTE);
8694 change |= snd_hda_codec_amp_update(codec, 0x15, 1, HDA_OUTPUT, 0,
8695 HDA_AMP_MUTE,
8696 valp[1] ? 0 : HDA_AMP_MUTE);
8697 if (change) {
8698 /* change speaker according to HP jack state */
8699 struct alc_spec *spec = codec->spec;
8700 unsigned int mute;
8701 if (spec->jack_present)
8702 mute = HDA_AMP_MUTE;
8703 else
8704 mute = snd_hda_codec_amp_read(codec, 0x15, 0,
8705 HDA_OUTPUT, 0);
8706 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
8707 HDA_AMP_MUTE, mute);
8708 }
8709 return change;
8710}
Takashi Iwai5b319542007-07-26 11:49:22 +02008711
Kailang Yang272a5272007-05-14 11:00:38 +02008712static struct snd_kcontrol_new alc262_sony_mixer[] = {
Takashi Iwai0724ea22007-08-23 00:31:43 +02008713 HDA_CODEC_VOLUME("Master Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
8714 {
8715 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
8716 .name = "Master Playback Switch",
8717 .info = snd_hda_mixer_amp_switch_info,
8718 .get = snd_hda_mixer_amp_switch_get,
8719 .put = alc262_sony_master_sw_put,
8720 .private_value = HDA_COMPOSE_AMP_VAL(0x15, 3, 0, HDA_OUTPUT),
8721 },
Kailang Yang272a5272007-05-14 11:00:38 +02008722 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
8723 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
8724 HDA_CODEC_VOLUME("ATAPI Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
8725 HDA_CODEC_MUTE("ATAPI Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
8726 { } /* end */
8727};
8728
Kailang Yang83c34212007-07-05 11:43:05 +02008729static struct snd_kcontrol_new alc262_benq_t31_mixer[] = {
8730 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
8731 HDA_CODEC_MUTE("Front Playback Switch", 0x14, 0x0, HDA_OUTPUT),
8732 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
8733 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
8734 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
8735 HDA_CODEC_VOLUME("ATAPI Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
8736 HDA_CODEC_MUTE("ATAPI Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
8737 { } /* end */
8738};
Kailang Yang272a5272007-05-14 11:00:38 +02008739
Kailang Yangdf694da2005-12-05 19:42:22 +01008740#define alc262_capture_mixer alc882_capture_mixer
8741#define alc262_capture_alt_mixer alc882_capture_alt_mixer
8742
8743/*
8744 * generic initialization of ADC, input mixers and output mixers
8745 */
8746static struct hda_verb alc262_init_verbs[] = {
8747 /*
8748 * Unmute ADC0-2 and set the default input to mic-in
8749 */
8750 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
8751 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
8752 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
8753 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
8754 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
8755 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
8756
Takashi Iwaicb53c622007-08-10 17:21:45 +02008757 /* Mute input amps (CD, Line In, Mic 1 & Mic 2) of the analog-loopback
Kailang Yangdf694da2005-12-05 19:42:22 +01008758 * mixer widget
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02008759 * Note: PASD motherboards uses the Line In 2 as the input for
8760 * front panel mic (mic 2)
Kailang Yangdf694da2005-12-05 19:42:22 +01008761 */
8762 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +02008763 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
8764 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
8765 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
8766 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
8767 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Kailang Yangdf694da2005-12-05 19:42:22 +01008768
8769 /*
8770 * Set up output mixers (0x0c - 0x0e)
8771 */
8772 /* set vol=0 to output mixers */
8773 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
8774 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
8775 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
8776 /* set up input amps for analog loopback */
8777 /* Amp Indices: DAC = 0, mixer = 1 */
8778 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
8779 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
8780 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
8781 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
8782 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
8783 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
8784
8785 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40},
8786 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc0},
8787 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40},
8788 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
8789 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
8790 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
8791
8792 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
8793 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
8794 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
8795 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
8796 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
8797
8798 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
8799 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01},
8800
8801 /* FIXME: use matrix-type input source selection */
8802 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
8803 /* Input mixer1: unmute Mic, F-Mic, Line, CD inputs */
8804 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
8805 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
8806 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
8807 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
8808 /* Input mixer2 */
8809 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
8810 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
8811 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
8812 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
8813 /* Input mixer3 */
8814 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
8815 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
8816 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02008817 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
Kailang Yangdf694da2005-12-05 19:42:22 +01008818
8819 { }
8820};
8821
Kailang Yangccc656c2006-10-17 12:32:26 +02008822static struct hda_verb alc262_hippo_unsol_verbs[] = {
8823 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
8824 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
8825 {}
8826};
8827
8828static struct hda_verb alc262_hippo1_unsol_verbs[] = {
8829 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc0},
8830 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
8831 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
8832
8833 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
8834 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
8835 {}
8836};
8837
Kailang Yang272a5272007-05-14 11:00:38 +02008838static struct hda_verb alc262_sony_unsol_verbs[] = {
8839 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc0},
8840 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
8841 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24}, // Front Mic
8842
8843 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
8844 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Akio Idehara7b1e8792008-06-09 22:46:07 +09008845 {}
Kailang Yang272a5272007-05-14 11:00:38 +02008846};
8847
Kailang Yangccc656c2006-10-17 12:32:26 +02008848/* mute/unmute internal speaker according to the hp jack and mute state */
Takashi Iwai5b319542007-07-26 11:49:22 +02008849static void alc262_hippo_automute(struct hda_codec *codec)
Kailang Yangccc656c2006-10-17 12:32:26 +02008850{
8851 struct alc_spec *spec = codec->spec;
8852 unsigned int mute;
Takashi Iwai5b319542007-07-26 11:49:22 +02008853 unsigned int present;
Kailang Yangccc656c2006-10-17 12:32:26 +02008854
Takashi Iwai5b319542007-07-26 11:49:22 +02008855 /* need to execute and sync at first */
8856 snd_hda_codec_read(codec, 0x15, 0, AC_VERB_SET_PIN_SENSE, 0);
8857 present = snd_hda_codec_read(codec, 0x15, 0,
8858 AC_VERB_GET_PIN_SENSE, 0);
8859 spec->jack_present = (present & 0x80000000) != 0;
Kailang Yangccc656c2006-10-17 12:32:26 +02008860 if (spec->jack_present) {
8861 /* mute internal speaker */
Takashi Iwai47fd8302007-08-10 17:11:07 +02008862 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
8863 HDA_AMP_MUTE, HDA_AMP_MUTE);
Kailang Yangccc656c2006-10-17 12:32:26 +02008864 } else {
8865 /* unmute internal speaker if necessary */
8866 mute = snd_hda_codec_amp_read(codec, 0x15, 0, HDA_OUTPUT, 0);
Takashi Iwai47fd8302007-08-10 17:11:07 +02008867 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
8868 HDA_AMP_MUTE, mute);
Kailang Yangccc656c2006-10-17 12:32:26 +02008869 }
8870}
8871
8872/* unsolicited event for HP jack sensing */
8873static void alc262_hippo_unsol_event(struct hda_codec *codec,
8874 unsigned int res)
8875{
8876 if ((res >> 26) != ALC880_HP_EVENT)
8877 return;
Takashi Iwai5b319542007-07-26 11:49:22 +02008878 alc262_hippo_automute(codec);
Kailang Yangccc656c2006-10-17 12:32:26 +02008879}
8880
Takashi Iwai5b319542007-07-26 11:49:22 +02008881static void alc262_hippo1_automute(struct hda_codec *codec)
Kailang Yangccc656c2006-10-17 12:32:26 +02008882{
Kailang Yangccc656c2006-10-17 12:32:26 +02008883 unsigned int mute;
Takashi Iwai5b319542007-07-26 11:49:22 +02008884 unsigned int present;
Kailang Yangccc656c2006-10-17 12:32:26 +02008885
Takashi Iwai5b319542007-07-26 11:49:22 +02008886 snd_hda_codec_read(codec, 0x1b, 0, AC_VERB_SET_PIN_SENSE, 0);
8887 present = snd_hda_codec_read(codec, 0x1b, 0,
8888 AC_VERB_GET_PIN_SENSE, 0);
8889 present = (present & 0x80000000) != 0;
8890 if (present) {
Kailang Yangccc656c2006-10-17 12:32:26 +02008891 /* mute internal speaker */
Takashi Iwai47fd8302007-08-10 17:11:07 +02008892 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
8893 HDA_AMP_MUTE, HDA_AMP_MUTE);
Kailang Yangccc656c2006-10-17 12:32:26 +02008894 } else {
8895 /* unmute internal speaker if necessary */
8896 mute = snd_hda_codec_amp_read(codec, 0x1b, 0, HDA_OUTPUT, 0);
Takashi Iwai47fd8302007-08-10 17:11:07 +02008897 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
8898 HDA_AMP_MUTE, mute);
Kailang Yangccc656c2006-10-17 12:32:26 +02008899 }
8900}
8901
8902/* unsolicited event for HP jack sensing */
8903static void alc262_hippo1_unsol_event(struct hda_codec *codec,
8904 unsigned int res)
8905{
8906 if ((res >> 26) != ALC880_HP_EVENT)
8907 return;
Takashi Iwai5b319542007-07-26 11:49:22 +02008908 alc262_hippo1_automute(codec);
Kailang Yangccc656c2006-10-17 12:32:26 +02008909}
8910
Takashi Iwai834be882006-03-01 14:16:17 +01008911/*
8912 * fujitsu model
Tony Vroon5d9fab22008-03-14 17:09:18 +01008913 * 0x14 = headphone/spdif-out, 0x15 = internal speaker,
8914 * 0x1b = port replicator headphone out
Takashi Iwai834be882006-03-01 14:16:17 +01008915 */
8916
8917#define ALC_HP_EVENT 0x37
8918
8919static struct hda_verb alc262_fujitsu_unsol_verbs[] = {
8920 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC_HP_EVENT},
8921 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Tony Vroon5d9fab22008-03-14 17:09:18 +01008922 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC_HP_EVENT},
8923 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Takashi Iwai834be882006-03-01 14:16:17 +01008924 {}
8925};
8926
Jiang zhe0e31daf2008-03-20 12:12:39 +01008927static struct hda_verb alc262_lenovo_3000_unsol_verbs[] = {
8928 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC_HP_EVENT},
8929 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
8930 {}
8931};
8932
Takashi Iwai834be882006-03-01 14:16:17 +01008933static struct hda_input_mux alc262_fujitsu_capture_source = {
Takashi Iwai39d3ed32007-10-12 15:03:48 +02008934 .num_items = 3,
Takashi Iwai834be882006-03-01 14:16:17 +01008935 .items = {
8936 { "Mic", 0x0 },
Takashi Iwai39d3ed32007-10-12 15:03:48 +02008937 { "Int Mic", 0x1 },
Takashi Iwai834be882006-03-01 14:16:17 +01008938 { "CD", 0x4 },
8939 },
8940};
8941
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008942static struct hda_input_mux alc262_HP_capture_source = {
8943 .num_items = 5,
8944 .items = {
8945 { "Mic", 0x0 },
zhejiangaccbe492007-08-31 12:36:05 +02008946 { "Front Mic", 0x1 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008947 { "Line", 0x2 },
8948 { "CD", 0x4 },
8949 { "AUX IN", 0x6 },
8950 },
8951};
8952
zhejiangaccbe492007-08-31 12:36:05 +02008953static struct hda_input_mux alc262_HP_D7000_capture_source = {
8954 .num_items = 4,
8955 .items = {
8956 { "Mic", 0x0 },
8957 { "Front Mic", 0x2 },
8958 { "Line", 0x1 },
8959 { "CD", 0x4 },
8960 },
8961};
8962
Takashi Iwaiebc7a402008-05-20 09:23:05 +02008963/* mute/unmute internal speaker according to the hp jacks and mute state */
Takashi Iwai834be882006-03-01 14:16:17 +01008964static void alc262_fujitsu_automute(struct hda_codec *codec, int force)
8965{
8966 struct alc_spec *spec = codec->spec;
8967 unsigned int mute;
8968
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02008969 if (force || !spec->sense_updated) {
Takashi Iwaiebc7a402008-05-20 09:23:05 +02008970 unsigned int present;
Takashi Iwai834be882006-03-01 14:16:17 +01008971 /* need to execute and sync at first */
8972 snd_hda_codec_read(codec, 0x14, 0, AC_VERB_SET_PIN_SENSE, 0);
Takashi Iwaiebc7a402008-05-20 09:23:05 +02008973 /* check laptop HP jack */
8974 present = snd_hda_codec_read(codec, 0x14, 0,
8975 AC_VERB_GET_PIN_SENSE, 0);
8976 /* need to execute and sync at first */
8977 snd_hda_codec_read(codec, 0x1b, 0, AC_VERB_SET_PIN_SENSE, 0);
8978 /* check docking HP jack */
8979 present |= snd_hda_codec_read(codec, 0x1b, 0,
8980 AC_VERB_GET_PIN_SENSE, 0);
8981 if (present & AC_PINSENSE_PRESENCE)
8982 spec->jack_present = 1;
8983 else
8984 spec->jack_present = 0;
Takashi Iwai834be882006-03-01 14:16:17 +01008985 spec->sense_updated = 1;
8986 }
Takashi Iwaiebc7a402008-05-20 09:23:05 +02008987 /* unmute internal speaker only if both HPs are unplugged and
8988 * master switch is on
8989 */
8990 if (spec->jack_present)
8991 mute = HDA_AMP_MUTE;
8992 else
Takashi Iwai834be882006-03-01 14:16:17 +01008993 mute = snd_hda_codec_amp_read(codec, 0x14, 0, HDA_OUTPUT, 0);
Takashi Iwaiebc7a402008-05-20 09:23:05 +02008994 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
8995 HDA_AMP_MUTE, mute);
Takashi Iwai834be882006-03-01 14:16:17 +01008996}
8997
8998/* unsolicited event for HP jack sensing */
8999static void alc262_fujitsu_unsol_event(struct hda_codec *codec,
9000 unsigned int res)
9001{
9002 if ((res >> 26) != ALC_HP_EVENT)
9003 return;
9004 alc262_fujitsu_automute(codec, 1);
9005}
9006
Takashi Iwaiebc7a402008-05-20 09:23:05 +02009007static void alc262_fujitsu_init_hook(struct hda_codec *codec)
9008{
9009 alc262_fujitsu_automute(codec, 1);
9010}
9011
Takashi Iwai834be882006-03-01 14:16:17 +01009012/* bind volumes of both NID 0x0c and 0x0d */
Takashi Iwaicca3b372007-08-10 17:12:15 +02009013static struct hda_bind_ctls alc262_fujitsu_bind_master_vol = {
9014 .ops = &snd_hda_bind_vol,
9015 .values = {
9016 HDA_COMPOSE_AMP_VAL(0x0c, 3, 0, HDA_OUTPUT),
9017 HDA_COMPOSE_AMP_VAL(0x0d, 3, 0, HDA_OUTPUT),
9018 0
9019 },
9020};
Takashi Iwai834be882006-03-01 14:16:17 +01009021
Jiang zhe0e31daf2008-03-20 12:12:39 +01009022/* mute/unmute internal speaker according to the hp jack and mute state */
9023static void alc262_lenovo_3000_automute(struct hda_codec *codec, int force)
9024{
9025 struct alc_spec *spec = codec->spec;
9026 unsigned int mute;
9027
9028 if (force || !spec->sense_updated) {
9029 unsigned int present_int_hp;
9030 /* need to execute and sync at first */
9031 snd_hda_codec_read(codec, 0x1b, 0, AC_VERB_SET_PIN_SENSE, 0);
9032 present_int_hp = snd_hda_codec_read(codec, 0x1b, 0,
9033 AC_VERB_GET_PIN_SENSE, 0);
9034 spec->jack_present = (present_int_hp & 0x80000000) != 0;
9035 spec->sense_updated = 1;
9036 }
9037 if (spec->jack_present) {
9038 /* mute internal speaker */
9039 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
9040 HDA_AMP_MUTE, HDA_AMP_MUTE);
9041 snd_hda_codec_amp_stereo(codec, 0x16, HDA_OUTPUT, 0,
9042 HDA_AMP_MUTE, HDA_AMP_MUTE);
9043 } else {
9044 /* unmute internal speaker if necessary */
9045 mute = snd_hda_codec_amp_read(codec, 0x1b, 0, HDA_OUTPUT, 0);
9046 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
9047 HDA_AMP_MUTE, mute);
9048 snd_hda_codec_amp_stereo(codec, 0x16, HDA_OUTPUT, 0,
9049 HDA_AMP_MUTE, mute);
9050 }
9051}
9052
9053/* unsolicited event for HP jack sensing */
9054static void alc262_lenovo_3000_unsol_event(struct hda_codec *codec,
9055 unsigned int res)
9056{
9057 if ((res >> 26) != ALC_HP_EVENT)
9058 return;
9059 alc262_lenovo_3000_automute(codec, 1);
9060}
9061
Takashi Iwai834be882006-03-01 14:16:17 +01009062/* bind hp and internal speaker mute (with plug check) */
9063static int alc262_fujitsu_master_sw_put(struct snd_kcontrol *kcontrol,
9064 struct snd_ctl_elem_value *ucontrol)
9065{
9066 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
9067 long *valp = ucontrol->value.integer.value;
9068 int change;
9069
Tony Vroon5d9fab22008-03-14 17:09:18 +01009070 change = snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
9071 HDA_AMP_MUTE,
9072 valp ? 0 : HDA_AMP_MUTE);
9073 change |= snd_hda_codec_amp_stereo(codec, 0x1b, HDA_OUTPUT, 0,
9074 HDA_AMP_MUTE,
9075 valp ? 0 : HDA_AMP_MUTE);
9076
Takashi Iwai82beb8f2007-08-10 17:09:26 +02009077 if (change)
9078 alc262_fujitsu_automute(codec, 0);
Takashi Iwai834be882006-03-01 14:16:17 +01009079 return change;
9080}
9081
9082static struct snd_kcontrol_new alc262_fujitsu_mixer[] = {
Takashi Iwaicca3b372007-08-10 17:12:15 +02009083 HDA_BIND_VOL("Master Playback Volume", &alc262_fujitsu_bind_master_vol),
Takashi Iwai834be882006-03-01 14:16:17 +01009084 {
9085 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
9086 .name = "Master Playback Switch",
9087 .info = snd_hda_mixer_amp_switch_info,
9088 .get = snd_hda_mixer_amp_switch_get,
9089 .put = alc262_fujitsu_master_sw_put,
9090 .private_value = HDA_COMPOSE_AMP_VAL(0x14, 3, 0, HDA_OUTPUT),
9091 },
9092 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
9093 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
Tony Vroon06a9c302008-04-14 13:31:45 +02009094 HDA_CODEC_VOLUME("PC Speaker Volume", 0x0b, 0x05, HDA_INPUT),
9095 HDA_CODEC_MUTE("PC Speaker Switch", 0x0b, 0x05, HDA_INPUT),
Takashi Iwai834be882006-03-01 14:16:17 +01009096 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
9097 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
9098 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Takashi Iwai39d3ed32007-10-12 15:03:48 +02009099 HDA_CODEC_VOLUME("Int Mic Boost", 0x19, 0, HDA_INPUT),
9100 HDA_CODEC_VOLUME("Int Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
9101 HDA_CODEC_MUTE("Int Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Takashi Iwai834be882006-03-01 14:16:17 +01009102 { } /* end */
9103};
9104
Jiang zhe0e31daf2008-03-20 12:12:39 +01009105/* bind hp and internal speaker mute (with plug check) */
9106static int alc262_lenovo_3000_master_sw_put(struct snd_kcontrol *kcontrol,
9107 struct snd_ctl_elem_value *ucontrol)
9108{
9109 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
9110 long *valp = ucontrol->value.integer.value;
9111 int change;
9112
9113 change = snd_hda_codec_amp_stereo(codec, 0x1b, HDA_OUTPUT, 0,
9114 HDA_AMP_MUTE,
9115 valp ? 0 : HDA_AMP_MUTE);
9116
9117 if (change)
9118 alc262_lenovo_3000_automute(codec, 0);
9119 return change;
9120}
9121
9122static struct snd_kcontrol_new alc262_lenovo_3000_mixer[] = {
9123 HDA_BIND_VOL("Master Playback Volume", &alc262_fujitsu_bind_master_vol),
9124 {
9125 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
9126 .name = "Master Playback Switch",
9127 .info = snd_hda_mixer_amp_switch_info,
9128 .get = snd_hda_mixer_amp_switch_get,
9129 .put = alc262_lenovo_3000_master_sw_put,
9130 .private_value = HDA_COMPOSE_AMP_VAL(0x1b, 3, 0, HDA_OUTPUT),
9131 },
9132 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
9133 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
9134 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
9135 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
9136 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
9137 HDA_CODEC_VOLUME("Int Mic Boost", 0x19, 0, HDA_INPUT),
9138 HDA_CODEC_VOLUME("Int Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
9139 HDA_CODEC_MUTE("Int Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
9140 { } /* end */
9141};
9142
Takashi Iwai304dcaa2006-07-25 14:51:16 +02009143/* additional init verbs for Benq laptops */
9144static struct hda_verb alc262_EAPD_verbs[] = {
9145 {0x20, AC_VERB_SET_COEF_INDEX, 0x07},
9146 {0x20, AC_VERB_SET_PROC_COEF, 0x3070},
9147 {}
9148};
9149
Kailang Yang83c34212007-07-05 11:43:05 +02009150static struct hda_verb alc262_benq_t31_EAPD_verbs[] = {
9151 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
9152 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
9153
9154 {0x20, AC_VERB_SET_COEF_INDEX, 0x07},
9155 {0x20, AC_VERB_SET_PROC_COEF, 0x3050},
9156 {}
9157};
9158
Tobin Davisf651b502007-10-26 12:40:47 +02009159/* Samsung Q1 Ultra Vista model setup */
9160static struct snd_kcontrol_new alc262_ultra_mixer[] = {
Takashi Iwaibb9f76c2008-03-12 12:51:09 +01009161 HDA_CODEC_VOLUME("Master Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
9162 HDA_BIND_MUTE("Master Playback Switch", 0x0c, 2, HDA_INPUT),
Tobin Davisf651b502007-10-26 12:40:47 +02009163 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
9164 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
9165 HDA_CODEC_VOLUME("Mic Boost", 0x19, 0, HDA_INPUT),
Takashi Iwaibb9f76c2008-03-12 12:51:09 +01009166 HDA_CODEC_VOLUME("Headphone Mic Boost", 0x15, 0, HDA_INPUT),
Tobin Davisf651b502007-10-26 12:40:47 +02009167 { } /* end */
9168};
9169
9170static struct hda_verb alc262_ultra_verbs[] = {
Takashi Iwaibb9f76c2008-03-12 12:51:09 +01009171 /* output mixer */
9172 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
9173 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
9174 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
9175 /* speaker */
9176 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
9177 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
9178 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9179 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
9180 /* HP */
Tobin Davisf651b502007-10-26 12:40:47 +02009181 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Takashi Iwaibb9f76c2008-03-12 12:51:09 +01009182 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
9183 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9184 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
9185 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
9186 /* internal mic */
9187 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
9188 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
9189 /* ADC, choose mic */
9190 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
9191 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
9192 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
9193 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
9194 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
9195 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
9196 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(5)},
9197 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(6)},
9198 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(7)},
9199 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(8)},
Tobin Davisf651b502007-10-26 12:40:47 +02009200 {}
9201};
9202
Tobin Davisf651b502007-10-26 12:40:47 +02009203/* mute/unmute internal speaker according to the hp jack and mute state */
9204static void alc262_ultra_automute(struct hda_codec *codec)
9205{
9206 struct alc_spec *spec = codec->spec;
9207 unsigned int mute;
Tobin Davisf651b502007-10-26 12:40:47 +02009208
Takashi Iwaibb9f76c2008-03-12 12:51:09 +01009209 mute = 0;
9210 /* auto-mute only when HP is used as HP */
9211 if (!spec->cur_mux[0]) {
9212 unsigned int present;
9213 /* need to execute and sync at first */
9214 snd_hda_codec_read(codec, 0x15, 0, AC_VERB_SET_PIN_SENSE, 0);
9215 present = snd_hda_codec_read(codec, 0x15, 0,
9216 AC_VERB_GET_PIN_SENSE, 0);
9217 spec->jack_present = (present & AC_PINSENSE_PRESENCE) != 0;
9218 if (spec->jack_present)
9219 mute = HDA_AMP_MUTE;
Tobin Davisf651b502007-10-26 12:40:47 +02009220 }
Takashi Iwaibb9f76c2008-03-12 12:51:09 +01009221 /* mute/unmute internal speaker */
9222 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
9223 HDA_AMP_MUTE, mute);
9224 /* mute/unmute HP */
9225 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
9226 HDA_AMP_MUTE, mute ? 0 : HDA_AMP_MUTE);
Tobin Davisf651b502007-10-26 12:40:47 +02009227}
9228
9229/* unsolicited event for HP jack sensing */
9230static void alc262_ultra_unsol_event(struct hda_codec *codec,
9231 unsigned int res)
9232{
9233 if ((res >> 26) != ALC880_HP_EVENT)
9234 return;
9235 alc262_ultra_automute(codec);
9236}
9237
Takashi Iwaibb9f76c2008-03-12 12:51:09 +01009238static struct hda_input_mux alc262_ultra_capture_source = {
9239 .num_items = 2,
9240 .items = {
9241 { "Mic", 0x1 },
9242 { "Headphone", 0x7 },
9243 },
9244};
9245
9246static int alc262_ultra_mux_enum_put(struct snd_kcontrol *kcontrol,
9247 struct snd_ctl_elem_value *ucontrol)
9248{
9249 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
9250 struct alc_spec *spec = codec->spec;
9251 int ret;
9252
9253 ret = alc882_mux_enum_put(kcontrol, ucontrol);
9254 if (!ret)
9255 return 0;
9256 /* reprogram the HP pin as mic or HP according to the input source */
9257 snd_hda_codec_write_cache(codec, 0x15, 0,
9258 AC_VERB_SET_PIN_WIDGET_CONTROL,
9259 spec->cur_mux[0] ? PIN_VREF80 : PIN_HP);
9260 alc262_ultra_automute(codec); /* mute/unmute HP */
9261 return ret;
9262}
9263
9264static struct snd_kcontrol_new alc262_ultra_capture_mixer[] = {
9265 HDA_CODEC_VOLUME("Capture Volume", 0x07, 0x0, HDA_INPUT),
9266 HDA_CODEC_MUTE("Capture Switch", 0x07, 0x0, HDA_INPUT),
9267 {
9268 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
9269 .name = "Capture Source",
9270 .info = alc882_mux_enum_info,
9271 .get = alc882_mux_enum_get,
9272 .put = alc262_ultra_mux_enum_put,
9273 },
9274 { } /* end */
9275};
9276
Kailang Yangdf694da2005-12-05 19:42:22 +01009277/* add playback controls from the parsed DAC table */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009278static int alc262_auto_create_multi_out_ctls(struct alc_spec *spec,
9279 const struct auto_pin_cfg *cfg)
Kailang Yangdf694da2005-12-05 19:42:22 +01009280{
9281 hda_nid_t nid;
9282 int err;
9283
9284 spec->multiout.num_dacs = 1; /* only use one dac */
9285 spec->multiout.dac_nids = spec->private_dac_nids;
9286 spec->multiout.dac_nids[0] = 2;
9287
9288 nid = cfg->line_out_pins[0];
9289 if (nid) {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009290 err = add_control(spec, ALC_CTL_WIDGET_VOL,
9291 "Front Playback Volume",
9292 HDA_COMPOSE_AMP_VAL(0x0c, 3, 0, HDA_OUTPUT));
9293 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +01009294 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009295 err = add_control(spec, ALC_CTL_WIDGET_MUTE,
9296 "Front Playback Switch",
9297 HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_OUTPUT));
9298 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +01009299 return err;
9300 }
9301
Takashi Iwai82bc9552006-03-21 11:24:42 +01009302 nid = cfg->speaker_pins[0];
Kailang Yangdf694da2005-12-05 19:42:22 +01009303 if (nid) {
9304 if (nid == 0x16) {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009305 err = add_control(spec, ALC_CTL_WIDGET_VOL,
9306 "Speaker Playback Volume",
9307 HDA_COMPOSE_AMP_VAL(0x0e, 2, 0,
9308 HDA_OUTPUT));
9309 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +01009310 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009311 err = add_control(spec, ALC_CTL_WIDGET_MUTE,
9312 "Speaker Playback Switch",
9313 HDA_COMPOSE_AMP_VAL(nid, 2, 0,
9314 HDA_OUTPUT));
9315 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +01009316 return err;
9317 } else {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009318 err = add_control(spec, ALC_CTL_WIDGET_MUTE,
9319 "Speaker Playback Switch",
9320 HDA_COMPOSE_AMP_VAL(nid, 3, 0,
9321 HDA_OUTPUT));
9322 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +01009323 return err;
9324 }
9325 }
Takashi Iwaieb06ed82006-09-20 17:10:27 +02009326 nid = cfg->hp_pins[0];
Kailang Yangdf694da2005-12-05 19:42:22 +01009327 if (nid) {
9328 /* spec->multiout.hp_nid = 2; */
9329 if (nid == 0x16) {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009330 err = add_control(spec, ALC_CTL_WIDGET_VOL,
9331 "Headphone Playback Volume",
9332 HDA_COMPOSE_AMP_VAL(0x0e, 2, 0,
9333 HDA_OUTPUT));
9334 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +01009335 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009336 err = add_control(spec, ALC_CTL_WIDGET_MUTE,
9337 "Headphone Playback Switch",
9338 HDA_COMPOSE_AMP_VAL(nid, 2, 0,
9339 HDA_OUTPUT));
9340 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +01009341 return err;
9342 } else {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009343 err = add_control(spec, ALC_CTL_WIDGET_MUTE,
9344 "Headphone Playback Switch",
9345 HDA_COMPOSE_AMP_VAL(nid, 3, 0,
9346 HDA_OUTPUT));
9347 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +01009348 return err;
9349 }
9350 }
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009351 return 0;
Kailang Yangdf694da2005-12-05 19:42:22 +01009352}
9353
9354/* identical with ALC880 */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009355#define alc262_auto_create_analog_input_ctls \
9356 alc880_auto_create_analog_input_ctls
Kailang Yangdf694da2005-12-05 19:42:22 +01009357
9358/*
9359 * generic initialization of ADC, input mixers and output mixers
9360 */
9361static struct hda_verb alc262_volume_init_verbs[] = {
9362 /*
9363 * Unmute ADC0-2 and set the default input to mic-in
9364 */
9365 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
9366 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9367 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
9368 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9369 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
9370 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9371
Takashi Iwaicb53c622007-08-10 17:21:45 +02009372 /* Mute input amps (CD, Line In, Mic 1 & Mic 2) of the analog-loopback
Kailang Yangdf694da2005-12-05 19:42:22 +01009373 * mixer widget
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009374 * Note: PASD motherboards uses the Line In 2 as the input for
9375 * front panel mic (mic 2)
Kailang Yangdf694da2005-12-05 19:42:22 +01009376 */
9377 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +02009378 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
9379 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
9380 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
9381 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
9382 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Kailang Yangdf694da2005-12-05 19:42:22 +01009383
9384 /*
9385 * Set up output mixers (0x0c - 0x0f)
9386 */
9387 /* set vol=0 to output mixers */
9388 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
9389 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
9390 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
9391
9392 /* set up input amps for analog loopback */
9393 /* Amp Indices: DAC = 0, mixer = 1 */
9394 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9395 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
9396 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9397 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
9398 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9399 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
9400
9401 /* FIXME: use matrix-type input source selection */
9402 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
9403 /* Input mixer1: unmute Mic, F-Mic, Line, CD inputs */
9404 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
9405 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
9406 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
9407 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
9408 /* Input mixer2 */
9409 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
9410 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
9411 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
9412 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
9413 /* Input mixer3 */
9414 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
9415 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
9416 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
9417 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
9418
9419 { }
9420};
9421
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009422static struct hda_verb alc262_HP_BPC_init_verbs[] = {
9423 /*
9424 * Unmute ADC0-2 and set the default input to mic-in
9425 */
9426 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
9427 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9428 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
9429 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9430 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
9431 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9432
Takashi Iwaicb53c622007-08-10 17:21:45 +02009433 /* Mute input amps (CD, Line In, Mic 1 & Mic 2) of the analog-loopback
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009434 * mixer widget
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009435 * Note: PASD motherboards uses the Line In 2 as the input for
9436 * front panel mic (mic 2)
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009437 */
9438 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +02009439 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
9440 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
9441 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
9442 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
9443 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
9444 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(5)},
9445 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(6)},
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009446
9447 /*
9448 * Set up output mixers (0x0c - 0x0e)
9449 */
9450 /* set vol=0 to output mixers */
9451 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
9452 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
9453 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
9454
9455 /* set up input amps for analog loopback */
9456 /* Amp Indices: DAC = 0, mixer = 1 */
9457 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9458 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
9459 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9460 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
9461 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9462 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
9463
Takashi Iwaice875f02008-01-28 18:17:43 +01009464 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009465 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
9466 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
9467
9468 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
9469 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
9470
9471 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
9472 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
9473
9474 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
9475 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
9476 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
9477 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
9478 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
9479
9480 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, 0x7023 },
9481 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000 },
9482 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000 },
9483 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, 0x7023 },
9484 {0x1c, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000 },
9485 {0x1d, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000 },
9486
9487
9488 /* FIXME: use matrix-type input source selection */
9489 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
9490 /* Input mixer1: unmute Mic, F-Mic, Line, CD inputs */
9491 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
9492 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x03 << 8))},
9493 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x02 << 8))},
9494 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x04 << 8))},
9495 /* Input mixer2 */
9496 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
9497 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x03 << 8))},
9498 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x02 << 8))},
9499 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x04 << 8))},
9500 /* Input mixer3 */
9501 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
9502 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x03 << 8))},
9503 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x02 << 8))},
9504 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x04 << 8))},
9505
Takashi Iwaice875f02008-01-28 18:17:43 +01009506 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
9507
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009508 { }
9509};
9510
Kailang Yangcd7509a2007-01-26 18:33:17 +01009511static struct hda_verb alc262_HP_BPC_WildWest_init_verbs[] = {
9512 /*
9513 * Unmute ADC0-2 and set the default input to mic-in
9514 */
9515 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
9516 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9517 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
9518 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9519 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
9520 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9521
Takashi Iwaicb53c622007-08-10 17:21:45 +02009522 /* Mute input amps (CD, Line In, Mic 1 & Mic 2) of the analog-loopback
Kailang Yangcd7509a2007-01-26 18:33:17 +01009523 * mixer widget
9524 * Note: PASD motherboards uses the Line In 2 as the input for front
9525 * panel mic (mic 2)
9526 */
9527 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +02009528 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
9529 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
9530 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
9531 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
9532 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
9533 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(5)},
9534 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(6)},
9535 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(7)},
Kailang Yangcd7509a2007-01-26 18:33:17 +01009536 /*
9537 * Set up output mixers (0x0c - 0x0e)
9538 */
9539 /* set vol=0 to output mixers */
9540 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
9541 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
9542 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
9543
9544 /* set up input amps for analog loopback */
9545 /* Amp Indices: DAC = 0, mixer = 1 */
9546 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9547 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
9548 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9549 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
9550 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
9551 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
9552
9553
9554 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP }, /* HP */
9555 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT }, /* Mono */
9556 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 }, /* rear MIC */
9557 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN }, /* Line in */
9558 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 }, /* Front MIC */
9559 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT }, /* Line out */
9560 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN }, /* CD in */
9561
9562 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
9563 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
9564
9565 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
9566 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01},
9567
9568 /* {0x14, AC_VERB_SET_AMP_GAIN_MUTE, 0x7023 }, */
9569 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000 },
9570 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000 },
9571 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, 0x7023 },
9572 {0x1c, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000 },
9573 {0x1d, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000 },
9574
9575 /* FIXME: use matrix-type input source selection */
9576 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
9577 /* Input mixer1: unmute Mic, F-Mic, Line, CD inputs */
9578 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))}, /*rear MIC*/
9579 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))}, /*Line in*/
9580 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x02 << 8))}, /*F MIC*/
9581 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x03 << 8))}, /*Front*/
9582 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x04 << 8))}, /*CD*/
9583 /* {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x06 << 8))}, */
9584 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x07 << 8))}, /*HP*/
9585 /* Input mixer2 */
9586 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
9587 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
9588 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x02 << 8))},
9589 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x03 << 8))},
9590 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x04 << 8))},
9591 /* {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x06 << 8))}, */
9592 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x07 << 8))},
9593 /* Input mixer3 */
9594 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
9595 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
9596 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x02 << 8))},
9597 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x03 << 8))},
9598 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x04 << 8))},
9599 /* {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x06 << 8))}, */
9600 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x07 << 8))},
9601
Takashi Iwaice875f02008-01-28 18:17:43 +01009602 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
9603
Kailang Yangcd7509a2007-01-26 18:33:17 +01009604 { }
9605};
9606
Takashi Iwaicb53c622007-08-10 17:21:45 +02009607#ifdef CONFIG_SND_HDA_POWER_SAVE
9608#define alc262_loopbacks alc880_loopbacks
9609#endif
9610
Kailang Yangdf694da2005-12-05 19:42:22 +01009611/* pcm configuration: identiacal with ALC880 */
9612#define alc262_pcm_analog_playback alc880_pcm_analog_playback
9613#define alc262_pcm_analog_capture alc880_pcm_analog_capture
9614#define alc262_pcm_digital_playback alc880_pcm_digital_playback
9615#define alc262_pcm_digital_capture alc880_pcm_digital_capture
9616
9617/*
9618 * BIOS auto configuration
9619 */
9620static int alc262_parse_auto_config(struct hda_codec *codec)
9621{
9622 struct alc_spec *spec = codec->spec;
9623 int err;
9624 static hda_nid_t alc262_ignore[] = { 0x1d, 0 };
9625
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009626 err = snd_hda_parse_pin_def_config(codec, &spec->autocfg,
9627 alc262_ignore);
9628 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +01009629 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009630 if (!spec->autocfg.line_outs)
Kailang Yangdf694da2005-12-05 19:42:22 +01009631 return 0; /* can't find valid BIOS pin config */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009632 err = alc262_auto_create_multi_out_ctls(spec, &spec->autocfg);
9633 if (err < 0)
9634 return err;
9635 err = alc262_auto_create_analog_input_ctls(spec, &spec->autocfg);
9636 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +01009637 return err;
9638
9639 spec->multiout.max_channels = spec->multiout.num_dacs * 2;
9640
9641 if (spec->autocfg.dig_out_pin)
9642 spec->multiout.dig_out_nid = ALC262_DIGOUT_NID;
9643 if (spec->autocfg.dig_in_pin)
9644 spec->dig_in_nid = ALC262_DIGIN_NID;
9645
9646 if (spec->kctl_alloc)
9647 spec->mixers[spec->num_mixers++] = spec->kctl_alloc;
9648
9649 spec->init_verbs[spec->num_init_verbs++] = alc262_volume_init_verbs;
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02009650 spec->num_mux_defs = 1;
Kailang Yangdf694da2005-12-05 19:42:22 +01009651 spec->input_mux = &spec->private_imux;
9652
Takashi Iwai776e1842007-08-29 15:07:11 +02009653 err = alc_auto_add_mic_boost(codec);
9654 if (err < 0)
9655 return err;
9656
Kailang Yangdf694da2005-12-05 19:42:22 +01009657 return 1;
9658}
9659
9660#define alc262_auto_init_multi_out alc882_auto_init_multi_out
9661#define alc262_auto_init_hp_out alc882_auto_init_hp_out
9662#define alc262_auto_init_analog_input alc882_auto_init_analog_input
9663
9664
9665/* init callback for auto-configuration model -- overriding the default init */
Takashi Iwaiae6b8132006-03-03 16:47:17 +01009666static void alc262_auto_init(struct hda_codec *codec)
Kailang Yangdf694da2005-12-05 19:42:22 +01009667{
Takashi Iwaif6c7e542008-02-12 18:32:23 +01009668 struct alc_spec *spec = codec->spec;
Kailang Yangdf694da2005-12-05 19:42:22 +01009669 alc262_auto_init_multi_out(codec);
9670 alc262_auto_init_hp_out(codec);
9671 alc262_auto_init_analog_input(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +01009672 if (spec->unsol_event)
9673 alc_sku_automute(codec);
Kailang Yangdf694da2005-12-05 19:42:22 +01009674}
9675
9676/*
9677 * configuration and preset
9678 */
Takashi Iwaif5fcc132006-11-24 17:07:44 +01009679static const char *alc262_models[ALC262_MODEL_LAST] = {
9680 [ALC262_BASIC] = "basic",
9681 [ALC262_HIPPO] = "hippo",
9682 [ALC262_HIPPO_1] = "hippo_1",
9683 [ALC262_FUJITSU] = "fujitsu",
9684 [ALC262_HP_BPC] = "hp-bpc",
Kailang Yangcd7509a2007-01-26 18:33:17 +01009685 [ALC262_HP_BPC_D7000_WL]= "hp-bpc-d7000",
Takashi Iwai61dc35d2007-11-14 12:26:44 +01009686 [ALC262_HP_TC_T5735] = "hp-tc-t5735",
Kailang Yang8c427222008-01-10 13:03:59 +01009687 [ALC262_HP_RP5700] = "hp-rp5700",
Takashi Iwaif5fcc132006-11-24 17:07:44 +01009688 [ALC262_BENQ_ED8] = "benq",
Takashi Iwai0f405022007-07-06 12:24:11 +02009689 [ALC262_BENQ_T31] = "benq-t31",
9690 [ALC262_SONY_ASSAMD] = "sony-assamd",
Tobin Davisf651b502007-10-26 12:40:47 +02009691 [ALC262_ULTRA] = "ultra",
Jiang zhe0e31daf2008-03-20 12:12:39 +01009692 [ALC262_LENOVO_3000] = "lenovo-3000",
Takashi Iwaif5fcc132006-11-24 17:07:44 +01009693 [ALC262_AUTO] = "auto",
9694};
9695
9696static struct snd_pci_quirk alc262_cfg_tbl[] = {
9697 SND_PCI_QUIRK(0x1002, 0x437b, "Hippo", ALC262_HIPPO),
9698 SND_PCI_QUIRK(0x103c, 0x12fe, "HP xw9400", ALC262_HP_BPC),
Kailang Yang7d87de22007-06-05 12:17:21 +02009699 SND_PCI_QUIRK(0x103c, 0x12ff, "HP xw4550", ALC262_HP_BPC),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01009700 SND_PCI_QUIRK(0x103c, 0x1306, "HP xw8600", ALC262_HP_BPC),
9701 SND_PCI_QUIRK(0x103c, 0x1307, "HP xw6600", ALC262_HP_BPC),
Kailang Yang7d87de22007-06-05 12:17:21 +02009702 SND_PCI_QUIRK(0x103c, 0x1308, "HP xw4600", ALC262_HP_BPC),
Takashi Iwaib98f9332007-11-07 14:18:01 +01009703 SND_PCI_QUIRK(0x103c, 0x1309, "HP xw4*00", ALC262_HP_BPC),
Takashi Iwaib98f9332007-11-07 14:18:01 +01009704 SND_PCI_QUIRK(0x103c, 0x130a, "HP xw6*00", ALC262_HP_BPC),
Takashi Iwaib98f9332007-11-07 14:18:01 +01009705 SND_PCI_QUIRK(0x103c, 0x130b, "HP xw8*00", ALC262_HP_BPC),
Kailang Yangcd7509a2007-01-26 18:33:17 +01009706 SND_PCI_QUIRK(0x103c, 0x2800, "HP D7000", ALC262_HP_BPC_D7000_WL),
Kailang Yangcd7509a2007-01-26 18:33:17 +01009707 SND_PCI_QUIRK(0x103c, 0x2801, "HP D7000", ALC262_HP_BPC_D7000_WF),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01009708 SND_PCI_QUIRK(0x103c, 0x2802, "HP D7000", ALC262_HP_BPC_D7000_WL),
Kailang Yangcd7509a2007-01-26 18:33:17 +01009709 SND_PCI_QUIRK(0x103c, 0x2803, "HP D7000", ALC262_HP_BPC_D7000_WF),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01009710 SND_PCI_QUIRK(0x103c, 0x2804, "HP D7000", ALC262_HP_BPC_D7000_WL),
Kailang Yangcd7509a2007-01-26 18:33:17 +01009711 SND_PCI_QUIRK(0x103c, 0x2805, "HP D7000", ALC262_HP_BPC_D7000_WF),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01009712 SND_PCI_QUIRK(0x103c, 0x2806, "HP D7000", ALC262_HP_BPC_D7000_WL),
Kailang Yangcd7509a2007-01-26 18:33:17 +01009713 SND_PCI_QUIRK(0x103c, 0x2807, "HP D7000", ALC262_HP_BPC_D7000_WF),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01009714 SND_PCI_QUIRK(0x103c, 0x280c, "HP xw4400", ALC262_HP_BPC),
9715 SND_PCI_QUIRK(0x103c, 0x3014, "HP xw6400", ALC262_HP_BPC),
9716 SND_PCI_QUIRK(0x103c, 0x3015, "HP xw8400", ALC262_HP_BPC),
Kailang Yang66d2a9d2007-11-14 12:06:21 +01009717 SND_PCI_QUIRK(0x103c, 0x302f, "HP Thin Client T5735",
9718 ALC262_HP_TC_T5735),
Kailang Yang8c427222008-01-10 13:03:59 +01009719 SND_PCI_QUIRK(0x103c, 0x2817, "HP RP5700", ALC262_HP_RP5700),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01009720 SND_PCI_QUIRK(0x104d, 0x1f00, "Sony ASSAMD", ALC262_SONY_ASSAMD),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01009721 SND_PCI_QUIRK(0x104d, 0x8203, "Sony UX-90", ALC262_HIPPO),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01009722 SND_PCI_QUIRK(0x104d, 0x820f, "Sony ASSAMD", ALC262_SONY_ASSAMD),
9723 SND_PCI_QUIRK(0x104d, 0x900e, "Sony ASSAMD", ALC262_SONY_ASSAMD),
9724 SND_PCI_QUIRK(0x104d, 0x9015, "Sony 0x9015", ALC262_SONY_ASSAMD),
Akio Idehara36ca6e12008-06-09 22:57:40 +09009725 SND_PCI_QUIRK(0x1179, 0x0001, "Toshiba dynabook SS RX1",
9726 ALC262_SONY_ASSAMD),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01009727 SND_PCI_QUIRK(0x10cf, 0x1397, "Fujitsu", ALC262_FUJITSU),
Tony Vroon3f1eeae2008-02-25 16:44:13 +01009728 SND_PCI_QUIRK(0x10cf, 0x142d, "Fujitsu Lifebook E8410", ALC262_FUJITSU),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01009729 SND_PCI_QUIRK(0x144d, 0xc032, "Samsung Q1 Ultra", ALC262_ULTRA),
Takashi Iwaibb9f76c2008-03-12 12:51:09 +01009730 SND_PCI_QUIRK(0x144d, 0xc039, "Samsung Q1U EL", ALC262_ULTRA),
Jiang zhe0e31daf2008-03-20 12:12:39 +01009731 SND_PCI_QUIRK(0x17aa, 0x384e, "Lenovo 3000 y410", ALC262_LENOVO_3000),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01009732 SND_PCI_QUIRK(0x17ff, 0x0560, "Benq ED8", ALC262_BENQ_ED8),
Kailang Yang83c34212007-07-05 11:43:05 +02009733 SND_PCI_QUIRK(0x17ff, 0x058d, "Benq T31-16", ALC262_BENQ_T31),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01009734 SND_PCI_QUIRK(0x17ff, 0x058f, "Benq Hippo", ALC262_HIPPO_1),
Kailang Yangdf694da2005-12-05 19:42:22 +01009735 {}
9736};
9737
9738static struct alc_config_preset alc262_presets[] = {
9739 [ALC262_BASIC] = {
9740 .mixers = { alc262_base_mixer },
9741 .init_verbs = { alc262_init_verbs },
9742 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
9743 .dac_nids = alc262_dac_nids,
9744 .hp_nid = 0x03,
9745 .num_channel_mode = ARRAY_SIZE(alc262_modes),
9746 .channel_mode = alc262_modes,
Takashi Iwaia3bcba32005-12-06 19:05:29 +01009747 .input_mux = &alc262_capture_source,
Kailang Yangdf694da2005-12-05 19:42:22 +01009748 },
Kailang Yangccc656c2006-10-17 12:32:26 +02009749 [ALC262_HIPPO] = {
9750 .mixers = { alc262_base_mixer },
9751 .init_verbs = { alc262_init_verbs, alc262_hippo_unsol_verbs},
9752 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
9753 .dac_nids = alc262_dac_nids,
9754 .hp_nid = 0x03,
9755 .dig_out_nid = ALC262_DIGOUT_NID,
9756 .num_channel_mode = ARRAY_SIZE(alc262_modes),
9757 .channel_mode = alc262_modes,
9758 .input_mux = &alc262_capture_source,
9759 .unsol_event = alc262_hippo_unsol_event,
Takashi Iwai5b319542007-07-26 11:49:22 +02009760 .init_hook = alc262_hippo_automute,
Kailang Yangccc656c2006-10-17 12:32:26 +02009761 },
9762 [ALC262_HIPPO_1] = {
9763 .mixers = { alc262_hippo1_mixer },
9764 .init_verbs = { alc262_init_verbs, alc262_hippo1_unsol_verbs},
9765 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
9766 .dac_nids = alc262_dac_nids,
9767 .hp_nid = 0x02,
9768 .dig_out_nid = ALC262_DIGOUT_NID,
9769 .num_channel_mode = ARRAY_SIZE(alc262_modes),
9770 .channel_mode = alc262_modes,
9771 .input_mux = &alc262_capture_source,
9772 .unsol_event = alc262_hippo1_unsol_event,
Takashi Iwai5b319542007-07-26 11:49:22 +02009773 .init_hook = alc262_hippo1_automute,
Kailang Yangccc656c2006-10-17 12:32:26 +02009774 },
Takashi Iwai834be882006-03-01 14:16:17 +01009775 [ALC262_FUJITSU] = {
9776 .mixers = { alc262_fujitsu_mixer },
Takashi Iwai39d3ed32007-10-12 15:03:48 +02009777 .init_verbs = { alc262_init_verbs, alc262_EAPD_verbs,
9778 alc262_fujitsu_unsol_verbs },
Takashi Iwai834be882006-03-01 14:16:17 +01009779 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
9780 .dac_nids = alc262_dac_nids,
9781 .hp_nid = 0x03,
9782 .dig_out_nid = ALC262_DIGOUT_NID,
9783 .num_channel_mode = ARRAY_SIZE(alc262_modes),
9784 .channel_mode = alc262_modes,
9785 .input_mux = &alc262_fujitsu_capture_source,
Takashi Iwaiae6b8132006-03-03 16:47:17 +01009786 .unsol_event = alc262_fujitsu_unsol_event,
Takashi Iwaiebc7a402008-05-20 09:23:05 +02009787 .init_hook = alc262_fujitsu_init_hook,
Takashi Iwai834be882006-03-01 14:16:17 +01009788 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009789 [ALC262_HP_BPC] = {
9790 .mixers = { alc262_HP_BPC_mixer },
9791 .init_verbs = { alc262_HP_BPC_init_verbs },
9792 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
9793 .dac_nids = alc262_dac_nids,
9794 .hp_nid = 0x03,
9795 .num_channel_mode = ARRAY_SIZE(alc262_modes),
9796 .channel_mode = alc262_modes,
9797 .input_mux = &alc262_HP_capture_source,
Takashi Iwaice875f02008-01-28 18:17:43 +01009798 .unsol_event = alc262_hp_bpc_unsol_event,
9799 .init_hook = alc262_hp_bpc_automute,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009800 },
Kailang Yangcd7509a2007-01-26 18:33:17 +01009801 [ALC262_HP_BPC_D7000_WF] = {
9802 .mixers = { alc262_HP_BPC_WildWest_mixer },
9803 .init_verbs = { alc262_HP_BPC_WildWest_init_verbs },
9804 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
9805 .dac_nids = alc262_dac_nids,
9806 .hp_nid = 0x03,
9807 .num_channel_mode = ARRAY_SIZE(alc262_modes),
9808 .channel_mode = alc262_modes,
zhejiangaccbe492007-08-31 12:36:05 +02009809 .input_mux = &alc262_HP_D7000_capture_source,
Takashi Iwaice875f02008-01-28 18:17:43 +01009810 .unsol_event = alc262_hp_wildwest_unsol_event,
9811 .init_hook = alc262_hp_wildwest_automute,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009812 },
Kailang Yangcd7509a2007-01-26 18:33:17 +01009813 [ALC262_HP_BPC_D7000_WL] = {
9814 .mixers = { alc262_HP_BPC_WildWest_mixer,
9815 alc262_HP_BPC_WildWest_option_mixer },
9816 .init_verbs = { alc262_HP_BPC_WildWest_init_verbs },
9817 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
9818 .dac_nids = alc262_dac_nids,
9819 .hp_nid = 0x03,
9820 .num_channel_mode = ARRAY_SIZE(alc262_modes),
9821 .channel_mode = alc262_modes,
zhejiangaccbe492007-08-31 12:36:05 +02009822 .input_mux = &alc262_HP_D7000_capture_source,
Takashi Iwaice875f02008-01-28 18:17:43 +01009823 .unsol_event = alc262_hp_wildwest_unsol_event,
9824 .init_hook = alc262_hp_wildwest_automute,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009825 },
Kailang Yang66d2a9d2007-11-14 12:06:21 +01009826 [ALC262_HP_TC_T5735] = {
9827 .mixers = { alc262_hp_t5735_mixer },
9828 .init_verbs = { alc262_init_verbs, alc262_hp_t5735_verbs },
9829 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
9830 .dac_nids = alc262_dac_nids,
9831 .hp_nid = 0x03,
9832 .num_channel_mode = ARRAY_SIZE(alc262_modes),
9833 .channel_mode = alc262_modes,
9834 .input_mux = &alc262_capture_source,
9835 .unsol_event = alc262_hp_t5735_unsol_event,
9836 .init_hook = alc262_hp_t5735_init_hook,
Kailang Yang8c427222008-01-10 13:03:59 +01009837 },
9838 [ALC262_HP_RP5700] = {
9839 .mixers = { alc262_hp_rp5700_mixer },
9840 .init_verbs = { alc262_init_verbs, alc262_hp_rp5700_verbs },
9841 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
9842 .dac_nids = alc262_dac_nids,
9843 .num_channel_mode = ARRAY_SIZE(alc262_modes),
9844 .channel_mode = alc262_modes,
9845 .input_mux = &alc262_hp_rp5700_capture_source,
Kailang Yang66d2a9d2007-11-14 12:06:21 +01009846 },
Takashi Iwai304dcaa2006-07-25 14:51:16 +02009847 [ALC262_BENQ_ED8] = {
9848 .mixers = { alc262_base_mixer },
9849 .init_verbs = { alc262_init_verbs, alc262_EAPD_verbs },
9850 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
9851 .dac_nids = alc262_dac_nids,
9852 .hp_nid = 0x03,
9853 .num_channel_mode = ARRAY_SIZE(alc262_modes),
9854 .channel_mode = alc262_modes,
9855 .input_mux = &alc262_capture_source,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009856 },
Kailang Yang272a5272007-05-14 11:00:38 +02009857 [ALC262_SONY_ASSAMD] = {
9858 .mixers = { alc262_sony_mixer },
9859 .init_verbs = { alc262_init_verbs, alc262_sony_unsol_verbs},
9860 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
9861 .dac_nids = alc262_dac_nids,
9862 .hp_nid = 0x02,
9863 .num_channel_mode = ARRAY_SIZE(alc262_modes),
9864 .channel_mode = alc262_modes,
9865 .input_mux = &alc262_capture_source,
9866 .unsol_event = alc262_hippo_unsol_event,
Takashi Iwai5b319542007-07-26 11:49:22 +02009867 .init_hook = alc262_hippo_automute,
Kailang Yang83c34212007-07-05 11:43:05 +02009868 },
9869 [ALC262_BENQ_T31] = {
9870 .mixers = { alc262_benq_t31_mixer },
9871 .init_verbs = { alc262_init_verbs, alc262_benq_t31_EAPD_verbs, alc262_hippo_unsol_verbs },
9872 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
9873 .dac_nids = alc262_dac_nids,
9874 .hp_nid = 0x03,
9875 .num_channel_mode = ARRAY_SIZE(alc262_modes),
9876 .channel_mode = alc262_modes,
9877 .input_mux = &alc262_capture_source,
9878 .unsol_event = alc262_hippo_unsol_event,
Takashi Iwai5b319542007-07-26 11:49:22 +02009879 .init_hook = alc262_hippo_automute,
Kailang Yang272a5272007-05-14 11:00:38 +02009880 },
Tobin Davisf651b502007-10-26 12:40:47 +02009881 [ALC262_ULTRA] = {
Takashi Iwaibb9f76c2008-03-12 12:51:09 +01009882 .mixers = { alc262_ultra_mixer, alc262_ultra_capture_mixer },
9883 .init_verbs = { alc262_ultra_verbs },
Tobin Davisf651b502007-10-26 12:40:47 +02009884 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
9885 .dac_nids = alc262_dac_nids,
Tobin Davisf651b502007-10-26 12:40:47 +02009886 .num_channel_mode = ARRAY_SIZE(alc262_modes),
9887 .channel_mode = alc262_modes,
9888 .input_mux = &alc262_ultra_capture_source,
Takashi Iwaibb9f76c2008-03-12 12:51:09 +01009889 .adc_nids = alc262_adc_nids, /* ADC0 */
9890 .capsrc_nids = alc262_capsrc_nids,
9891 .num_adc_nids = 1, /* single ADC */
Tobin Davisf651b502007-10-26 12:40:47 +02009892 .unsol_event = alc262_ultra_unsol_event,
9893 .init_hook = alc262_ultra_automute,
9894 },
Jiang zhe0e31daf2008-03-20 12:12:39 +01009895 [ALC262_LENOVO_3000] = {
9896 .mixers = { alc262_lenovo_3000_mixer },
9897 .init_verbs = { alc262_init_verbs, alc262_EAPD_verbs,
9898 alc262_lenovo_3000_unsol_verbs },
9899 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
9900 .dac_nids = alc262_dac_nids,
9901 .hp_nid = 0x03,
9902 .dig_out_nid = ALC262_DIGOUT_NID,
9903 .num_channel_mode = ARRAY_SIZE(alc262_modes),
9904 .channel_mode = alc262_modes,
9905 .input_mux = &alc262_fujitsu_capture_source,
9906 .unsol_event = alc262_lenovo_3000_unsol_event,
9907 },
Kailang Yangdf694da2005-12-05 19:42:22 +01009908};
9909
9910static int patch_alc262(struct hda_codec *codec)
9911{
9912 struct alc_spec *spec;
9913 int board_config;
9914 int err;
9915
Robert P. J. Daydc041e02006-12-19 14:44:15 +01009916 spec = kzalloc(sizeof(*spec), GFP_KERNEL);
Kailang Yangdf694da2005-12-05 19:42:22 +01009917 if (spec == NULL)
9918 return -ENOMEM;
9919
9920 codec->spec = spec;
9921#if 0
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009922 /* pshou 07/11/05 set a zero PCM sample to DAC when FIFO is
9923 * under-run
9924 */
Kailang Yangdf694da2005-12-05 19:42:22 +01009925 {
9926 int tmp;
9927 snd_hda_codec_write(codec, 0x1a, 0, AC_VERB_SET_COEF_INDEX, 7);
9928 tmp = snd_hda_codec_read(codec, 0x20, 0, AC_VERB_GET_PROC_COEF, 0);
9929 snd_hda_codec_write(codec, 0x1a, 0, AC_VERB_SET_COEF_INDEX, 7);
9930 snd_hda_codec_write(codec, 0x1a, 0, AC_VERB_SET_PROC_COEF, tmp | 0x80);
9931 }
9932#endif
9933
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +02009934 alc_fix_pll_init(codec, 0x20, 0x0a, 10);
9935
Takashi Iwaif5fcc132006-11-24 17:07:44 +01009936 board_config = snd_hda_check_board_config(codec, ALC262_MODEL_LAST,
9937 alc262_models,
9938 alc262_cfg_tbl);
Kailang Yangcd7509a2007-01-26 18:33:17 +01009939
Takashi Iwaif5fcc132006-11-24 17:07:44 +01009940 if (board_config < 0) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009941 printk(KERN_INFO "hda_codec: Unknown model for ALC262, "
9942 "trying auto-probe from BIOS...\n");
Kailang Yangdf694da2005-12-05 19:42:22 +01009943 board_config = ALC262_AUTO;
9944 }
9945
9946 if (board_config == ALC262_AUTO) {
9947 /* automatic parse from the BIOS config */
9948 err = alc262_parse_auto_config(codec);
9949 if (err < 0) {
9950 alc_free(codec);
9951 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009952 } else if (!err) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009953 printk(KERN_INFO
9954 "hda_codec: Cannot set up configuration "
9955 "from BIOS. Using base mode...\n");
Kailang Yangdf694da2005-12-05 19:42:22 +01009956 board_config = ALC262_BASIC;
9957 }
9958 }
9959
9960 if (board_config != ALC262_AUTO)
9961 setup_preset(spec, &alc262_presets[board_config]);
9962
9963 spec->stream_name_analog = "ALC262 Analog";
9964 spec->stream_analog_playback = &alc262_pcm_analog_playback;
9965 spec->stream_analog_capture = &alc262_pcm_analog_capture;
9966
9967 spec->stream_name_digital = "ALC262 Digital";
9968 spec->stream_digital_playback = &alc262_pcm_digital_playback;
9969 spec->stream_digital_capture = &alc262_pcm_digital_capture;
9970
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009971 if (!spec->adc_nids && spec->input_mux) {
Kailang Yangdf694da2005-12-05 19:42:22 +01009972 /* check whether NID 0x07 is valid */
Takashi Iwai4a471b72005-12-07 13:56:29 +01009973 unsigned int wcap = get_wcaps(codec, 0x07);
9974
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009975 /* get type */
9976 wcap = (wcap & AC_WCAP_TYPE) >> AC_WCAP_TYPE_SHIFT;
Kailang Yangdf694da2005-12-05 19:42:22 +01009977 if (wcap != AC_WID_AUD_IN) {
9978 spec->adc_nids = alc262_adc_nids_alt;
9979 spec->num_adc_nids = ARRAY_SIZE(alc262_adc_nids_alt);
Takashi Iwai88c71a92008-02-14 17:27:17 +01009980 spec->capsrc_nids = alc262_capsrc_nids_alt;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009981 spec->mixers[spec->num_mixers] =
9982 alc262_capture_alt_mixer;
Kailang Yangdf694da2005-12-05 19:42:22 +01009983 spec->num_mixers++;
9984 } else {
9985 spec->adc_nids = alc262_adc_nids;
9986 spec->num_adc_nids = ARRAY_SIZE(alc262_adc_nids);
Takashi Iwai88c71a92008-02-14 17:27:17 +01009987 spec->capsrc_nids = alc262_capsrc_nids;
Kailang Yangdf694da2005-12-05 19:42:22 +01009988 spec->mixers[spec->num_mixers] = alc262_capture_mixer;
9989 spec->num_mixers++;
9990 }
9991 }
9992
Takashi Iwai2134ea42008-01-10 16:53:55 +01009993 spec->vmaster_nid = 0x0c;
9994
Kailang Yangdf694da2005-12-05 19:42:22 +01009995 codec->patch_ops = alc_patch_ops;
9996 if (board_config == ALC262_AUTO)
Takashi Iwaiae6b8132006-03-03 16:47:17 +01009997 spec->init_hook = alc262_auto_init;
Takashi Iwaicb53c622007-08-10 17:21:45 +02009998#ifdef CONFIG_SND_HDA_POWER_SAVE
9999 if (!spec->loopback.amplist)
10000 spec->loopback.amplist = alc262_loopbacks;
10001#endif
Takashi Iwai834be882006-03-01 14:16:17 +010010002
Kailang Yangdf694da2005-12-05 19:42:22 +010010003 return 0;
10004}
10005
Kailang Yangdf694da2005-12-05 19:42:22 +010010006/*
Kailang Yanga361d842007-06-05 12:30:55 +020010007 * ALC268 channel source setting (2 channel)
10008 */
10009#define ALC268_DIGOUT_NID ALC880_DIGOUT_NID
10010#define alc268_modes alc260_modes
10011
10012static hda_nid_t alc268_dac_nids[2] = {
10013 /* front, hp */
10014 0x02, 0x03
10015};
10016
10017static hda_nid_t alc268_adc_nids[2] = {
10018 /* ADC0-1 */
10019 0x08, 0x07
10020};
10021
10022static hda_nid_t alc268_adc_nids_alt[1] = {
10023 /* ADC0 */
10024 0x08
10025};
10026
Takashi Iwaie1406342008-02-11 18:32:32 +010010027static hda_nid_t alc268_capsrc_nids[2] = { 0x23, 0x24 };
10028
Kailang Yanga361d842007-06-05 12:30:55 +020010029static struct snd_kcontrol_new alc268_base_mixer[] = {
10030 /* output mixer control */
10031 HDA_CODEC_VOLUME("Front Playback Volume", 0x2, 0x0, HDA_OUTPUT),
10032 HDA_CODEC_MUTE("Front Playback Switch", 0x14, 0x0, HDA_OUTPUT),
10033 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x3, 0x0, HDA_OUTPUT),
10034 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
Takashi Iwai33bf17a2007-08-21 11:51:42 +020010035 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
10036 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
10037 HDA_CODEC_VOLUME("Line In Boost", 0x1a, 0, HDA_INPUT),
Kailang Yanga361d842007-06-05 12:30:55 +020010038 { }
10039};
10040
Takashi Iwaiaef9d312008-02-19 13:16:41 +010010041/* bind Beep switches of both NID 0x0f and 0x10 */
10042static struct hda_bind_ctls alc268_bind_beep_sw = {
10043 .ops = &snd_hda_bind_sw,
10044 .values = {
10045 HDA_COMPOSE_AMP_VAL(0x0f, 3, 1, HDA_INPUT),
10046 HDA_COMPOSE_AMP_VAL(0x10, 3, 1, HDA_INPUT),
10047 0
10048 },
10049};
10050
10051static struct snd_kcontrol_new alc268_beep_mixer[] = {
10052 HDA_CODEC_VOLUME("Beep Playback Volume", 0x1d, 0x0, HDA_INPUT),
10053 HDA_BIND_SW("Beep Playback Switch", &alc268_bind_beep_sw),
10054 { }
10055};
10056
Kailang Yangd1a991a2007-08-15 16:21:59 +020010057static struct hda_verb alc268_eapd_verbs[] = {
10058 {0x14, AC_VERB_SET_EAPD_BTLENABLE, 2},
10059 {0x15, AC_VERB_SET_EAPD_BTLENABLE, 2},
10060 { }
10061};
10062
Takashi Iwaid2738092007-08-16 14:59:45 +020010063/* Toshiba specific */
10064#define alc268_toshiba_automute alc262_hippo_automute
10065
10066static struct hda_verb alc268_toshiba_verbs[] = {
10067 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
10068 { } /* end */
10069};
10070
10071/* Acer specific */
Takashi Iwai889c4392007-08-23 18:56:52 +020010072/* bind volumes of both NID 0x02 and 0x03 */
Takashi Iwai6bc96852007-08-17 09:02:12 +020010073static struct hda_bind_ctls alc268_acer_bind_master_vol = {
10074 .ops = &snd_hda_bind_vol,
10075 .values = {
10076 HDA_COMPOSE_AMP_VAL(0x02, 3, 0, HDA_OUTPUT),
10077 HDA_COMPOSE_AMP_VAL(0x03, 3, 0, HDA_OUTPUT),
10078 0
10079 },
10080};
10081
Takashi Iwai889c4392007-08-23 18:56:52 +020010082/* mute/unmute internal speaker according to the hp jack and mute state */
10083static void alc268_acer_automute(struct hda_codec *codec, int force)
10084{
10085 struct alc_spec *spec = codec->spec;
10086 unsigned int mute;
10087
10088 if (force || !spec->sense_updated) {
10089 unsigned int present;
10090 present = snd_hda_codec_read(codec, 0x14, 0,
10091 AC_VERB_GET_PIN_SENSE, 0);
10092 spec->jack_present = (present & 0x80000000) != 0;
10093 spec->sense_updated = 1;
10094 }
10095 if (spec->jack_present)
10096 mute = HDA_AMP_MUTE; /* mute internal speaker */
10097 else /* unmute internal speaker if necessary */
10098 mute = snd_hda_codec_amp_read(codec, 0x14, 0, HDA_OUTPUT, 0);
10099 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
10100 HDA_AMP_MUTE, mute);
10101}
10102
10103
10104/* bind hp and internal speaker mute (with plug check) */
10105static int alc268_acer_master_sw_put(struct snd_kcontrol *kcontrol,
10106 struct snd_ctl_elem_value *ucontrol)
10107{
10108 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
10109 long *valp = ucontrol->value.integer.value;
10110 int change;
10111
10112 change = snd_hda_codec_amp_update(codec, 0x14, 0, HDA_OUTPUT, 0,
10113 HDA_AMP_MUTE,
10114 valp[0] ? 0 : HDA_AMP_MUTE);
10115 change |= snd_hda_codec_amp_update(codec, 0x14, 1, HDA_OUTPUT, 0,
10116 HDA_AMP_MUTE,
10117 valp[1] ? 0 : HDA_AMP_MUTE);
10118 if (change)
10119 alc268_acer_automute(codec, 0);
10120 return change;
10121}
Takashi Iwaid2738092007-08-16 14:59:45 +020010122
10123static struct snd_kcontrol_new alc268_acer_mixer[] = {
10124 /* output mixer control */
10125 HDA_BIND_VOL("Master Playback Volume", &alc268_acer_bind_master_vol),
10126 {
10127 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
10128 .name = "Master Playback Switch",
10129 .info = snd_hda_mixer_amp_switch_info,
10130 .get = snd_hda_mixer_amp_switch_get,
10131 .put = alc268_acer_master_sw_put,
10132 .private_value = HDA_COMPOSE_AMP_VAL(0x14, 3, 0, HDA_OUTPUT),
10133 },
Takashi Iwai33bf17a2007-08-21 11:51:42 +020010134 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
10135 HDA_CODEC_VOLUME("Internal Mic Boost", 0x19, 0, HDA_INPUT),
10136 HDA_CODEC_VOLUME("Line In Boost", 0x1a, 0, HDA_INPUT),
Takashi Iwaid2738092007-08-16 14:59:45 +020010137 { }
10138};
10139
10140static struct hda_verb alc268_acer_verbs[] = {
Takashi Iwai0ccb5412008-03-06 16:58:35 +010010141 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN}, /* internal dmic? */
10142 {0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Takashi Iwaid2738092007-08-16 14:59:45 +020010143 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
10144 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwai0ccb5412008-03-06 16:58:35 +010010145 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
10146 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Takashi Iwaid2738092007-08-16 14:59:45 +020010147
10148 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
10149 { }
10150};
10151
10152/* unsolicited event for HP jack sensing */
10153static void alc268_toshiba_unsol_event(struct hda_codec *codec,
10154 unsigned int res)
10155{
Takashi Iwai889c4392007-08-23 18:56:52 +020010156 if ((res >> 26) != ALC880_HP_EVENT)
Takashi Iwaid2738092007-08-16 14:59:45 +020010157 return;
10158 alc268_toshiba_automute(codec);
10159}
10160
10161static void alc268_acer_unsol_event(struct hda_codec *codec,
10162 unsigned int res)
10163{
Takashi Iwai889c4392007-08-23 18:56:52 +020010164 if ((res >> 26) != ALC880_HP_EVENT)
Takashi Iwaid2738092007-08-16 14:59:45 +020010165 return;
10166 alc268_acer_automute(codec, 1);
10167}
10168
Takashi Iwai889c4392007-08-23 18:56:52 +020010169static void alc268_acer_init_hook(struct hda_codec *codec)
10170{
10171 alc268_acer_automute(codec, 1);
10172}
10173
Takashi Iwai3866f0b2008-01-15 12:37:42 +010010174static struct snd_kcontrol_new alc268_dell_mixer[] = {
10175 /* output mixer control */
10176 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x02, 0x0, HDA_OUTPUT),
10177 HDA_CODEC_MUTE("Speaker Playback Switch", 0x14, 0x0, HDA_OUTPUT),
10178 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x03, 0x0, HDA_OUTPUT),
10179 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
10180 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
10181 HDA_CODEC_VOLUME("Internal Mic Boost", 0x19, 0, HDA_INPUT),
10182 { }
10183};
10184
10185static struct hda_verb alc268_dell_verbs[] = {
10186 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
10187 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
10188 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
10189 { }
10190};
10191
10192/* mute/unmute internal speaker according to the hp jack and mute state */
10193static void alc268_dell_automute(struct hda_codec *codec)
10194{
10195 unsigned int present;
10196 unsigned int mute;
10197
10198 present = snd_hda_codec_read(codec, 0x15, 0, AC_VERB_GET_PIN_SENSE, 0);
10199 if (present & 0x80000000)
10200 mute = HDA_AMP_MUTE;
10201 else
10202 mute = snd_hda_codec_amp_read(codec, 0x15, 0, HDA_OUTPUT, 0);
10203 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
10204 HDA_AMP_MUTE, mute);
10205}
10206
10207static void alc268_dell_unsol_event(struct hda_codec *codec,
10208 unsigned int res)
10209{
10210 if ((res >> 26) != ALC880_HP_EVENT)
10211 return;
10212 alc268_dell_automute(codec);
10213}
10214
10215#define alc268_dell_init_hook alc268_dell_automute
10216
Herton Ronaldo Krzesinskieb5a6622008-04-14 13:46:28 +020010217static struct snd_kcontrol_new alc267_quanta_il1_mixer[] = {
10218 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x2, 0x0, HDA_OUTPUT),
10219 HDA_CODEC_MUTE("Speaker Playback Switch", 0x14, 0x0, HDA_OUTPUT),
10220 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x3, 0x0, HDA_OUTPUT),
10221 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
10222 HDA_CODEC_VOLUME("Mic Capture Volume", 0x23, 0x0, HDA_OUTPUT),
10223 HDA_BIND_MUTE("Mic Capture Switch", 0x23, 2, HDA_OUTPUT),
10224 HDA_CODEC_VOLUME("Ext Mic Boost", 0x18, 0, HDA_INPUT),
10225 HDA_CODEC_VOLUME("Int Mic Boost", 0x19, 0, HDA_INPUT),
10226 { }
10227};
10228
10229static struct hda_verb alc267_quanta_il1_verbs[] = {
10230 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
10231 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_MIC_EVENT | AC_USRSP_EN},
10232 { }
10233};
10234
10235static void alc267_quanta_il1_hp_automute(struct hda_codec *codec)
10236{
10237 unsigned int present;
10238
10239 present = snd_hda_codec_read(codec, 0x15, 0, AC_VERB_GET_PIN_SENSE, 0)
10240 & AC_PINSENSE_PRESENCE;
10241 snd_hda_codec_write(codec, 0x14, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
10242 present ? 0 : PIN_OUT);
10243}
10244
10245static void alc267_quanta_il1_mic_automute(struct hda_codec *codec)
10246{
10247 unsigned int present;
10248
10249 present = snd_hda_codec_read(codec, 0x18, 0,
10250 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
10251 snd_hda_codec_write(codec, 0x23, 0,
10252 AC_VERB_SET_CONNECT_SEL,
10253 present ? 0x00 : 0x01);
10254}
10255
10256static void alc267_quanta_il1_automute(struct hda_codec *codec)
10257{
10258 alc267_quanta_il1_hp_automute(codec);
10259 alc267_quanta_il1_mic_automute(codec);
10260}
10261
10262static void alc267_quanta_il1_unsol_event(struct hda_codec *codec,
10263 unsigned int res)
10264{
10265 switch (res >> 26) {
10266 case ALC880_HP_EVENT:
10267 alc267_quanta_il1_hp_automute(codec);
10268 break;
10269 case ALC880_MIC_EVENT:
10270 alc267_quanta_il1_mic_automute(codec);
10271 break;
10272 }
10273}
10274
Kailang Yanga361d842007-06-05 12:30:55 +020010275/*
10276 * generic initialization of ADC, input mixers and output mixers
10277 */
10278static struct hda_verb alc268_base_init_verbs[] = {
10279 /* Unmute DAC0-1 and set vol = 0 */
10280 {0x02, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
10281 {0x02, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10282 {0x02, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
10283 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
10284 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10285 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
10286
10287 /*
10288 * Set up output mixers (0x0c - 0x0e)
10289 */
10290 /* set vol=0 to output mixers */
10291 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10292 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
10293 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
10294 {0x0e, AC_VERB_SET_CONNECT_SEL, 0x00},
10295
10296 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10297 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10298
10299 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40},
10300 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc0},
10301 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40},
10302 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
10303 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
10304 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
10305 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
10306 {0x1d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
10307
10308 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
10309 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
10310 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
10311 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
10312 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
10313 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
10314 {0x1c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
Takashi Iwaiaef9d312008-02-19 13:16:41 +010010315
10316 /* set PCBEEP vol = 0, mute connections */
10317 {0x1d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10318 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
10319 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Kailang Yanga361d842007-06-05 12:30:55 +020010320
Jiang Zhea9b3aa82007-12-20 13:13:13 +010010321 /* Unmute Selector 23h,24h and set the default input to mic-in */
10322
10323 {0x23, AC_VERB_SET_CONNECT_SEL, 0x00},
10324 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
10325 {0x24, AC_VERB_SET_CONNECT_SEL, 0x00},
10326 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Kailang Yanga361d842007-06-05 12:30:55 +020010327
Kailang Yanga361d842007-06-05 12:30:55 +020010328 { }
10329};
10330
10331/*
10332 * generic initialization of ADC, input mixers and output mixers
10333 */
10334static struct hda_verb alc268_volume_init_verbs[] = {
10335 /* set output DAC */
10336 {0x02, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10337 {0x02, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
10338 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10339 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
10340
10341 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
10342 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
10343 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
10344 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
10345 {0x1d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
10346
10347 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
10348 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10349 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
10350 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10351 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10352
10353 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
10354 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
10355 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
10356 {0x1c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
10357
Takashi Iwaiaef9d312008-02-19 13:16:41 +010010358 /* set PCBEEP vol = 0, mute connections */
10359 {0x1d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10360 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
10361 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Kailang Yanga361d842007-06-05 12:30:55 +020010362
10363 { }
10364};
10365
10366#define alc268_mux_enum_info alc_mux_enum_info
10367#define alc268_mux_enum_get alc_mux_enum_get
Takashi Iwaie1406342008-02-11 18:32:32 +010010368#define alc268_mux_enum_put alc_mux_enum_put
Kailang Yanga361d842007-06-05 12:30:55 +020010369
10370static struct snd_kcontrol_new alc268_capture_alt_mixer[] = {
10371 HDA_CODEC_VOLUME("Capture Volume", 0x23, 0x0, HDA_OUTPUT),
10372 HDA_CODEC_MUTE("Capture Switch", 0x23, 0x0, HDA_OUTPUT),
10373 {
10374 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
10375 /* The multiple "Capture Source" controls confuse alsamixer
10376 * So call somewhat different..
Kailang Yanga361d842007-06-05 12:30:55 +020010377 */
10378 /* .name = "Capture Source", */
10379 .name = "Input Source",
10380 .count = 1,
10381 .info = alc268_mux_enum_info,
10382 .get = alc268_mux_enum_get,
10383 .put = alc268_mux_enum_put,
10384 },
10385 { } /* end */
10386};
10387
10388static struct snd_kcontrol_new alc268_capture_mixer[] = {
10389 HDA_CODEC_VOLUME("Capture Volume", 0x23, 0x0, HDA_OUTPUT),
10390 HDA_CODEC_MUTE("Capture Switch", 0x23, 0x0, HDA_OUTPUT),
10391 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x24, 0x0, HDA_OUTPUT),
10392 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x24, 0x0, HDA_OUTPUT),
10393 {
10394 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
10395 /* The multiple "Capture Source" controls confuse alsamixer
10396 * So call somewhat different..
Kailang Yanga361d842007-06-05 12:30:55 +020010397 */
10398 /* .name = "Capture Source", */
10399 .name = "Input Source",
10400 .count = 2,
10401 .info = alc268_mux_enum_info,
10402 .get = alc268_mux_enum_get,
10403 .put = alc268_mux_enum_put,
10404 },
10405 { } /* end */
10406};
10407
10408static struct hda_input_mux alc268_capture_source = {
10409 .num_items = 4,
10410 .items = {
10411 { "Mic", 0x0 },
10412 { "Front Mic", 0x1 },
10413 { "Line", 0x2 },
10414 { "CD", 0x3 },
10415 },
10416};
10417
Takashi Iwai0ccb5412008-03-06 16:58:35 +010010418static struct hda_input_mux alc268_acer_capture_source = {
10419 .num_items = 3,
10420 .items = {
10421 { "Mic", 0x0 },
10422 { "Internal Mic", 0x6 },
10423 { "Line", 0x2 },
10424 },
10425};
10426
Jonathan Woithe86c53bd2008-01-08 12:33:19 +010010427#ifdef CONFIG_SND_DEBUG
10428static struct snd_kcontrol_new alc268_test_mixer[] = {
Jonathan Woithe86c53bd2008-01-08 12:33:19 +010010429 /* Volume widgets */
10430 HDA_CODEC_VOLUME("LOUT1 Playback Volume", 0x02, 0x0, HDA_OUTPUT),
10431 HDA_CODEC_VOLUME("LOUT2 Playback Volume", 0x03, 0x0, HDA_OUTPUT),
10432 HDA_BIND_MUTE_MONO("Mono sum Playback Switch", 0x0e, 1, 2, HDA_INPUT),
10433 HDA_BIND_MUTE("LINE-OUT sum Playback Switch", 0x0f, 2, HDA_INPUT),
10434 HDA_BIND_MUTE("HP-OUT sum Playback Switch", 0x10, 2, HDA_INPUT),
10435 HDA_BIND_MUTE("LINE-OUT Playback Switch", 0x14, 2, HDA_OUTPUT),
10436 HDA_BIND_MUTE("HP-OUT Playback Switch", 0x15, 2, HDA_OUTPUT),
10437 HDA_BIND_MUTE("Mono Playback Switch", 0x16, 2, HDA_OUTPUT),
10438 HDA_CODEC_VOLUME("MIC1 Capture Volume", 0x18, 0x0, HDA_INPUT),
10439 HDA_BIND_MUTE("MIC1 Capture Switch", 0x18, 2, HDA_OUTPUT),
10440 HDA_CODEC_VOLUME("MIC2 Capture Volume", 0x19, 0x0, HDA_INPUT),
10441 HDA_CODEC_VOLUME("LINE1 Capture Volume", 0x1a, 0x0, HDA_INPUT),
10442 HDA_BIND_MUTE("LINE1 Capture Switch", 0x1a, 2, HDA_OUTPUT),
Takashi Iwaif0747ee2008-01-15 11:41:41 +010010443 /* The below appears problematic on some hardwares */
10444 /*HDA_CODEC_VOLUME("PCBEEP Playback Volume", 0x1d, 0x0, HDA_INPUT),*/
Jonathan Woithe86c53bd2008-01-08 12:33:19 +010010445 HDA_CODEC_VOLUME("PCM-IN1 Capture Volume", 0x23, 0x0, HDA_OUTPUT),
10446 HDA_BIND_MUTE("PCM-IN1 Capture Switch", 0x23, 2, HDA_OUTPUT),
10447 HDA_CODEC_VOLUME("PCM-IN2 Capture Volume", 0x24, 0x0, HDA_OUTPUT),
10448 HDA_BIND_MUTE("PCM-IN2 Capture Switch", 0x24, 2, HDA_OUTPUT),
10449
10450 /* Modes for retasking pin widgets */
10451 ALC_PIN_MODE("LINE-OUT pin mode", 0x14, ALC_PIN_DIR_INOUT),
10452 ALC_PIN_MODE("HP-OUT pin mode", 0x15, ALC_PIN_DIR_INOUT),
10453 ALC_PIN_MODE("MIC1 pin mode", 0x18, ALC_PIN_DIR_INOUT),
10454 ALC_PIN_MODE("LINE1 pin mode", 0x1a, ALC_PIN_DIR_INOUT),
10455
10456 /* Controls for GPIO pins, assuming they are configured as outputs */
10457 ALC_GPIO_DATA_SWITCH("GPIO pin 0", 0x01, 0x01),
10458 ALC_GPIO_DATA_SWITCH("GPIO pin 1", 0x01, 0x02),
10459 ALC_GPIO_DATA_SWITCH("GPIO pin 2", 0x01, 0x04),
10460 ALC_GPIO_DATA_SWITCH("GPIO pin 3", 0x01, 0x08),
10461
10462 /* Switches to allow the digital SPDIF output pin to be enabled.
10463 * The ALC268 does not have an SPDIF input.
10464 */
10465 ALC_SPDIF_CTRL_SWITCH("SPDIF Playback Switch", 0x06, 0x01),
10466
10467 /* A switch allowing EAPD to be enabled. Some laptops seem to use
10468 * this output to turn on an external amplifier.
10469 */
10470 ALC_EAPD_CTRL_SWITCH("LINE-OUT EAPD Enable Switch", 0x0f, 0x02),
10471 ALC_EAPD_CTRL_SWITCH("HP-OUT EAPD Enable Switch", 0x10, 0x02),
10472
10473 { } /* end */
10474};
10475#endif
10476
Kailang Yanga361d842007-06-05 12:30:55 +020010477/* create input playback/capture controls for the given pin */
10478static int alc268_new_analog_output(struct alc_spec *spec, hda_nid_t nid,
10479 const char *ctlname, int idx)
10480{
10481 char name[32];
10482 int err;
10483
10484 sprintf(name, "%s Playback Volume", ctlname);
10485 if (nid == 0x14) {
10486 err = add_control(spec, ALC_CTL_WIDGET_VOL, name,
10487 HDA_COMPOSE_AMP_VAL(0x02, 3, idx,
10488 HDA_OUTPUT));
10489 if (err < 0)
10490 return err;
10491 } else if (nid == 0x15) {
10492 err = add_control(spec, ALC_CTL_WIDGET_VOL, name,
10493 HDA_COMPOSE_AMP_VAL(0x03, 3, idx,
10494 HDA_OUTPUT));
10495 if (err < 0)
10496 return err;
10497 } else
10498 return -1;
10499 sprintf(name, "%s Playback Switch", ctlname);
10500 err = add_control(spec, ALC_CTL_WIDGET_MUTE, name,
10501 HDA_COMPOSE_AMP_VAL(nid, 3, idx, HDA_OUTPUT));
10502 if (err < 0)
10503 return err;
10504 return 0;
10505}
10506
10507/* add playback controls from the parsed DAC table */
10508static int alc268_auto_create_multi_out_ctls(struct alc_spec *spec,
10509 const struct auto_pin_cfg *cfg)
10510{
10511 hda_nid_t nid;
10512 int err;
10513
10514 spec->multiout.num_dacs = 2; /* only use one dac */
10515 spec->multiout.dac_nids = spec->private_dac_nids;
10516 spec->multiout.dac_nids[0] = 2;
10517 spec->multiout.dac_nids[1] = 3;
10518
10519 nid = cfg->line_out_pins[0];
10520 if (nid)
10521 alc268_new_analog_output(spec, nid, "Front", 0);
10522
10523 nid = cfg->speaker_pins[0];
10524 if (nid == 0x1d) {
10525 err = add_control(spec, ALC_CTL_WIDGET_VOL,
10526 "Speaker Playback Volume",
10527 HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_INPUT));
10528 if (err < 0)
10529 return err;
10530 }
10531 nid = cfg->hp_pins[0];
10532 if (nid)
10533 alc268_new_analog_output(spec, nid, "Headphone", 0);
10534
10535 nid = cfg->line_out_pins[1] | cfg->line_out_pins[2];
10536 if (nid == 0x16) {
10537 err = add_control(spec, ALC_CTL_WIDGET_MUTE,
10538 "Mono Playback Switch",
10539 HDA_COMPOSE_AMP_VAL(nid, 2, 0, HDA_INPUT));
10540 if (err < 0)
10541 return err;
10542 }
10543 return 0;
10544}
10545
10546/* create playback/capture controls for input pins */
10547static int alc268_auto_create_analog_input_ctls(struct alc_spec *spec,
10548 const struct auto_pin_cfg *cfg)
10549{
10550 struct hda_input_mux *imux = &spec->private_imux;
10551 int i, idx1;
10552
10553 for (i = 0; i < AUTO_PIN_LAST; i++) {
10554 switch(cfg->input_pins[i]) {
10555 case 0x18:
10556 idx1 = 0; /* Mic 1 */
10557 break;
10558 case 0x19:
10559 idx1 = 1; /* Mic 2 */
10560 break;
10561 case 0x1a:
10562 idx1 = 2; /* Line In */
10563 break;
10564 case 0x1c:
10565 idx1 = 3; /* CD */
10566 break;
Takashi Iwai7194cae2008-03-06 16:58:17 +010010567 case 0x12:
10568 case 0x13:
10569 idx1 = 6; /* digital mics */
10570 break;
Kailang Yanga361d842007-06-05 12:30:55 +020010571 default:
10572 continue;
10573 }
10574 imux->items[imux->num_items].label = auto_pin_cfg_labels[i];
10575 imux->items[imux->num_items].index = idx1;
10576 imux->num_items++;
10577 }
10578 return 0;
10579}
10580
10581static void alc268_auto_init_mono_speaker_out(struct hda_codec *codec)
10582{
10583 struct alc_spec *spec = codec->spec;
10584 hda_nid_t speaker_nid = spec->autocfg.speaker_pins[0];
10585 hda_nid_t hp_nid = spec->autocfg.hp_pins[0];
10586 hda_nid_t line_nid = spec->autocfg.line_out_pins[0];
10587 unsigned int dac_vol1, dac_vol2;
10588
10589 if (speaker_nid) {
10590 snd_hda_codec_write(codec, speaker_nid, 0,
10591 AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
10592 snd_hda_codec_write(codec, 0x0f, 0,
10593 AC_VERB_SET_AMP_GAIN_MUTE,
10594 AMP_IN_UNMUTE(1));
10595 snd_hda_codec_write(codec, 0x10, 0,
10596 AC_VERB_SET_AMP_GAIN_MUTE,
10597 AMP_IN_UNMUTE(1));
10598 } else {
10599 snd_hda_codec_write(codec, 0x0f, 0,
10600 AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1));
10601 snd_hda_codec_write(codec, 0x10, 0,
10602 AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1));
10603 }
10604
10605 dac_vol1 = dac_vol2 = 0xb000 | 0x40; /* set max volume */
10606 if (line_nid == 0x14)
10607 dac_vol2 = AMP_OUT_ZERO;
10608 else if (line_nid == 0x15)
10609 dac_vol1 = AMP_OUT_ZERO;
10610 if (hp_nid == 0x14)
10611 dac_vol2 = AMP_OUT_ZERO;
10612 else if (hp_nid == 0x15)
10613 dac_vol1 = AMP_OUT_ZERO;
10614 if (line_nid != 0x16 || hp_nid != 0x16 ||
10615 spec->autocfg.line_out_pins[1] != 0x16 ||
10616 spec->autocfg.line_out_pins[2] != 0x16)
10617 dac_vol1 = dac_vol2 = AMP_OUT_ZERO;
10618
10619 snd_hda_codec_write(codec, 0x02, 0,
10620 AC_VERB_SET_AMP_GAIN_MUTE, dac_vol1);
10621 snd_hda_codec_write(codec, 0x03, 0,
10622 AC_VERB_SET_AMP_GAIN_MUTE, dac_vol2);
10623}
10624
10625/* pcm configuration: identiacal with ALC880 */
10626#define alc268_pcm_analog_playback alc880_pcm_analog_playback
10627#define alc268_pcm_analog_capture alc880_pcm_analog_capture
Takashi Iwai63300792008-01-24 15:31:36 +010010628#define alc268_pcm_analog_alt_capture alc880_pcm_analog_alt_capture
Kailang Yanga361d842007-06-05 12:30:55 +020010629#define alc268_pcm_digital_playback alc880_pcm_digital_playback
10630
10631/*
10632 * BIOS auto configuration
10633 */
10634static int alc268_parse_auto_config(struct hda_codec *codec)
10635{
10636 struct alc_spec *spec = codec->spec;
10637 int err;
10638 static hda_nid_t alc268_ignore[] = { 0 };
10639
10640 err = snd_hda_parse_pin_def_config(codec, &spec->autocfg,
10641 alc268_ignore);
10642 if (err < 0)
10643 return err;
10644 if (!spec->autocfg.line_outs)
10645 return 0; /* can't find valid BIOS pin config */
10646
10647 err = alc268_auto_create_multi_out_ctls(spec, &spec->autocfg);
10648 if (err < 0)
10649 return err;
10650 err = alc268_auto_create_analog_input_ctls(spec, &spec->autocfg);
10651 if (err < 0)
10652 return err;
10653
10654 spec->multiout.max_channels = 2;
10655
10656 /* digital only support output */
10657 if (spec->autocfg.dig_out_pin)
10658 spec->multiout.dig_out_nid = ALC268_DIGOUT_NID;
10659
10660 if (spec->kctl_alloc)
10661 spec->mixers[spec->num_mixers++] = spec->kctl_alloc;
10662
Takashi Iwaiaef9d312008-02-19 13:16:41 +010010663 if (spec->autocfg.speaker_pins[0] != 0x1d)
10664 spec->mixers[spec->num_mixers++] = alc268_beep_mixer;
10665
Kailang Yanga361d842007-06-05 12:30:55 +020010666 spec->init_verbs[spec->num_init_verbs++] = alc268_volume_init_verbs;
10667 spec->num_mux_defs = 1;
10668 spec->input_mux = &spec->private_imux;
10669
Takashi Iwai776e1842007-08-29 15:07:11 +020010670 err = alc_auto_add_mic_boost(codec);
10671 if (err < 0)
10672 return err;
10673
Kailang Yanga361d842007-06-05 12:30:55 +020010674 return 1;
10675}
10676
10677#define alc268_auto_init_multi_out alc882_auto_init_multi_out
10678#define alc268_auto_init_hp_out alc882_auto_init_hp_out
10679#define alc268_auto_init_analog_input alc882_auto_init_analog_input
10680
10681/* init callback for auto-configuration model -- overriding the default init */
10682static void alc268_auto_init(struct hda_codec *codec)
10683{
Takashi Iwaif6c7e542008-02-12 18:32:23 +010010684 struct alc_spec *spec = codec->spec;
Kailang Yanga361d842007-06-05 12:30:55 +020010685 alc268_auto_init_multi_out(codec);
10686 alc268_auto_init_hp_out(codec);
10687 alc268_auto_init_mono_speaker_out(codec);
10688 alc268_auto_init_analog_input(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +010010689 if (spec->unsol_event)
10690 alc_sku_automute(codec);
Kailang Yanga361d842007-06-05 12:30:55 +020010691}
10692
10693/*
10694 * configuration and preset
10695 */
10696static const char *alc268_models[ALC268_MODEL_LAST] = {
Herton Ronaldo Krzesinskieb5a6622008-04-14 13:46:28 +020010697 [ALC267_QUANTA_IL1] = "quanta-il1",
Kailang Yanga361d842007-06-05 12:30:55 +020010698 [ALC268_3ST] = "3stack",
Takashi Iwai983f8ae2007-08-15 16:44:04 +020010699 [ALC268_TOSHIBA] = "toshiba",
Takashi Iwaid2738092007-08-16 14:59:45 +020010700 [ALC268_ACER] = "acer",
Takashi Iwai3866f0b2008-01-15 12:37:42 +010010701 [ALC268_DELL] = "dell",
Mirco Tischlerf12462c2008-02-04 12:33:59 +010010702 [ALC268_ZEPTO] = "zepto",
Jonathan Woithe86c53bd2008-01-08 12:33:19 +010010703#ifdef CONFIG_SND_DEBUG
10704 [ALC268_TEST] = "test",
10705#endif
Kailang Yanga361d842007-06-05 12:30:55 +020010706 [ALC268_AUTO] = "auto",
10707};
10708
10709static struct snd_pci_quirk alc268_cfg_tbl[] = {
Takashi Iwaia0b8f7d2008-04-22 19:39:49 +020010710 SND_PCI_QUIRK(0x1025, 0x011e, "Acer Aspire 5720z", ALC268_ACER),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010010711 SND_PCI_QUIRK(0x1025, 0x0126, "Acer", ALC268_ACER),
Andy Shevchenkodafc8352008-01-25 11:53:50 +010010712 SND_PCI_QUIRK(0x1025, 0x012e, "Acer Aspire 5310", ALC268_ACER),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010010713 SND_PCI_QUIRK(0x1025, 0x0130, "Acer Extensa 5210", ALC268_ACER),
Andy Shevchenko29a52c22008-01-24 17:29:00 +010010714 SND_PCI_QUIRK(0x1025, 0x0136, "Acer Aspire 5315", ALC268_ACER),
Takashi Iwai3866f0b2008-01-15 12:37:42 +010010715 SND_PCI_QUIRK(0x1028, 0x0253, "Dell OEM", ALC268_DELL),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010010716 SND_PCI_QUIRK(0x103c, 0x30cc, "TOSHIBA", ALC268_TOSHIBA),
Kailang Yanga361d842007-06-05 12:30:55 +020010717 SND_PCI_QUIRK(0x1043, 0x1205, "ASUS W7J", ALC268_3ST),
Kailang Yangd1a991a2007-08-15 16:21:59 +020010718 SND_PCI_QUIRK(0x1179, 0xff10, "TOSHIBA A205", ALC268_TOSHIBA),
Takashi Iwai8e7f00f2007-09-07 10:58:58 +020010719 SND_PCI_QUIRK(0x1179, 0xff50, "TOSHIBA A305", ALC268_TOSHIBA),
Tony Vroon378bd6a2008-06-04 12:08:30 +020010720 SND_PCI_QUIRK(0x14c0, 0x0025, "COMPAL IFL90/JFL-92", ALC268_TOSHIBA),
Takashi Iwaib875bf32007-09-06 15:00:27 +020010721 SND_PCI_QUIRK(0x152d, 0x0763, "Diverse (CPR2000)", ALC268_ACER),
Herton Ronaldo Krzesinskieb5a6622008-04-14 13:46:28 +020010722 SND_PCI_QUIRK(0x152d, 0x0771, "Quanta IL1", ALC267_QUANTA_IL1),
Mirco Tischlerf12462c2008-02-04 12:33:59 +010010723 SND_PCI_QUIRK(0x1170, 0x0040, "ZEPTO", ALC268_ZEPTO),
Kailang Yanga361d842007-06-05 12:30:55 +020010724 {}
10725};
10726
10727static struct alc_config_preset alc268_presets[] = {
Herton Ronaldo Krzesinskieb5a6622008-04-14 13:46:28 +020010728 [ALC267_QUANTA_IL1] = {
10729 .mixers = { alc267_quanta_il1_mixer },
10730 .init_verbs = { alc268_base_init_verbs, alc268_eapd_verbs,
10731 alc267_quanta_il1_verbs },
10732 .num_dacs = ARRAY_SIZE(alc268_dac_nids),
10733 .dac_nids = alc268_dac_nids,
10734 .num_adc_nids = ARRAY_SIZE(alc268_adc_nids_alt),
10735 .adc_nids = alc268_adc_nids_alt,
10736 .hp_nid = 0x03,
10737 .num_channel_mode = ARRAY_SIZE(alc268_modes),
10738 .channel_mode = alc268_modes,
10739 .input_mux = &alc268_capture_source,
10740 .unsol_event = alc267_quanta_il1_unsol_event,
10741 .init_hook = alc267_quanta_il1_automute,
10742 },
Kailang Yanga361d842007-06-05 12:30:55 +020010743 [ALC268_3ST] = {
Takashi Iwaiaef9d312008-02-19 13:16:41 +010010744 .mixers = { alc268_base_mixer, alc268_capture_alt_mixer,
10745 alc268_beep_mixer },
Kailang Yanga361d842007-06-05 12:30:55 +020010746 .init_verbs = { alc268_base_init_verbs },
10747 .num_dacs = ARRAY_SIZE(alc268_dac_nids),
10748 .dac_nids = alc268_dac_nids,
10749 .num_adc_nids = ARRAY_SIZE(alc268_adc_nids_alt),
10750 .adc_nids = alc268_adc_nids_alt,
Takashi Iwaie1406342008-02-11 18:32:32 +010010751 .capsrc_nids = alc268_capsrc_nids,
Kailang Yanga361d842007-06-05 12:30:55 +020010752 .hp_nid = 0x03,
10753 .dig_out_nid = ALC268_DIGOUT_NID,
10754 .num_channel_mode = ARRAY_SIZE(alc268_modes),
10755 .channel_mode = alc268_modes,
10756 .input_mux = &alc268_capture_source,
10757 },
Kailang Yangd1a991a2007-08-15 16:21:59 +020010758 [ALC268_TOSHIBA] = {
Takashi Iwaiaef9d312008-02-19 13:16:41 +010010759 .mixers = { alc268_base_mixer, alc268_capture_alt_mixer,
10760 alc268_beep_mixer },
Takashi Iwaid2738092007-08-16 14:59:45 +020010761 .init_verbs = { alc268_base_init_verbs, alc268_eapd_verbs,
10762 alc268_toshiba_verbs },
Kailang Yangd1a991a2007-08-15 16:21:59 +020010763 .num_dacs = ARRAY_SIZE(alc268_dac_nids),
10764 .dac_nids = alc268_dac_nids,
10765 .num_adc_nids = ARRAY_SIZE(alc268_adc_nids_alt),
10766 .adc_nids = alc268_adc_nids_alt,
Takashi Iwaie1406342008-02-11 18:32:32 +010010767 .capsrc_nids = alc268_capsrc_nids,
Kailang Yangd1a991a2007-08-15 16:21:59 +020010768 .hp_nid = 0x03,
10769 .num_channel_mode = ARRAY_SIZE(alc268_modes),
10770 .channel_mode = alc268_modes,
10771 .input_mux = &alc268_capture_source,
Takashi Iwaid2738092007-08-16 14:59:45 +020010772 .unsol_event = alc268_toshiba_unsol_event,
10773 .init_hook = alc268_toshiba_automute,
10774 },
10775 [ALC268_ACER] = {
Takashi Iwaiaef9d312008-02-19 13:16:41 +010010776 .mixers = { alc268_acer_mixer, alc268_capture_alt_mixer,
10777 alc268_beep_mixer },
Takashi Iwaid2738092007-08-16 14:59:45 +020010778 .init_verbs = { alc268_base_init_verbs, alc268_eapd_verbs,
10779 alc268_acer_verbs },
10780 .num_dacs = ARRAY_SIZE(alc268_dac_nids),
10781 .dac_nids = alc268_dac_nids,
10782 .num_adc_nids = ARRAY_SIZE(alc268_adc_nids_alt),
10783 .adc_nids = alc268_adc_nids_alt,
Takashi Iwaie1406342008-02-11 18:32:32 +010010784 .capsrc_nids = alc268_capsrc_nids,
Takashi Iwaid2738092007-08-16 14:59:45 +020010785 .hp_nid = 0x02,
10786 .num_channel_mode = ARRAY_SIZE(alc268_modes),
10787 .channel_mode = alc268_modes,
Takashi Iwai0ccb5412008-03-06 16:58:35 +010010788 .input_mux = &alc268_acer_capture_source,
Takashi Iwaid2738092007-08-16 14:59:45 +020010789 .unsol_event = alc268_acer_unsol_event,
Takashi Iwai889c4392007-08-23 18:56:52 +020010790 .init_hook = alc268_acer_init_hook,
Kailang Yangd1a991a2007-08-15 16:21:59 +020010791 },
Takashi Iwai3866f0b2008-01-15 12:37:42 +010010792 [ALC268_DELL] = {
Takashi Iwaiaef9d312008-02-19 13:16:41 +010010793 .mixers = { alc268_dell_mixer, alc268_beep_mixer },
Takashi Iwai3866f0b2008-01-15 12:37:42 +010010794 .init_verbs = { alc268_base_init_verbs, alc268_eapd_verbs,
10795 alc268_dell_verbs },
10796 .num_dacs = ARRAY_SIZE(alc268_dac_nids),
10797 .dac_nids = alc268_dac_nids,
10798 .hp_nid = 0x02,
10799 .num_channel_mode = ARRAY_SIZE(alc268_modes),
10800 .channel_mode = alc268_modes,
10801 .unsol_event = alc268_dell_unsol_event,
10802 .init_hook = alc268_dell_init_hook,
10803 .input_mux = &alc268_capture_source,
10804 },
Mirco Tischlerf12462c2008-02-04 12:33:59 +010010805 [ALC268_ZEPTO] = {
Takashi Iwaiaef9d312008-02-19 13:16:41 +010010806 .mixers = { alc268_base_mixer, alc268_capture_alt_mixer,
10807 alc268_beep_mixer },
Mirco Tischlerf12462c2008-02-04 12:33:59 +010010808 .init_verbs = { alc268_base_init_verbs, alc268_eapd_verbs,
10809 alc268_toshiba_verbs },
10810 .num_dacs = ARRAY_SIZE(alc268_dac_nids),
10811 .dac_nids = alc268_dac_nids,
10812 .num_adc_nids = ARRAY_SIZE(alc268_adc_nids_alt),
10813 .adc_nids = alc268_adc_nids_alt,
Takashi Iwaie1406342008-02-11 18:32:32 +010010814 .capsrc_nids = alc268_capsrc_nids,
Mirco Tischlerf12462c2008-02-04 12:33:59 +010010815 .hp_nid = 0x03,
10816 .dig_out_nid = ALC268_DIGOUT_NID,
10817 .num_channel_mode = ARRAY_SIZE(alc268_modes),
10818 .channel_mode = alc268_modes,
10819 .input_mux = &alc268_capture_source,
10820 .unsol_event = alc268_toshiba_unsol_event,
10821 .init_hook = alc268_toshiba_automute
10822 },
Jonathan Woithe86c53bd2008-01-08 12:33:19 +010010823#ifdef CONFIG_SND_DEBUG
10824 [ALC268_TEST] = {
10825 .mixers = { alc268_test_mixer, alc268_capture_mixer },
10826 .init_verbs = { alc268_base_init_verbs, alc268_eapd_verbs,
10827 alc268_volume_init_verbs },
10828 .num_dacs = ARRAY_SIZE(alc268_dac_nids),
10829 .dac_nids = alc268_dac_nids,
10830 .num_adc_nids = ARRAY_SIZE(alc268_adc_nids_alt),
10831 .adc_nids = alc268_adc_nids_alt,
Takashi Iwaie1406342008-02-11 18:32:32 +010010832 .capsrc_nids = alc268_capsrc_nids,
Jonathan Woithe86c53bd2008-01-08 12:33:19 +010010833 .hp_nid = 0x03,
10834 .dig_out_nid = ALC268_DIGOUT_NID,
10835 .num_channel_mode = ARRAY_SIZE(alc268_modes),
10836 .channel_mode = alc268_modes,
10837 .input_mux = &alc268_capture_source,
10838 },
10839#endif
Kailang Yanga361d842007-06-05 12:30:55 +020010840};
10841
10842static int patch_alc268(struct hda_codec *codec)
10843{
10844 struct alc_spec *spec;
10845 int board_config;
10846 int err;
10847
10848 spec = kcalloc(1, sizeof(*spec), GFP_KERNEL);
10849 if (spec == NULL)
10850 return -ENOMEM;
10851
10852 codec->spec = spec;
10853
10854 board_config = snd_hda_check_board_config(codec, ALC268_MODEL_LAST,
10855 alc268_models,
10856 alc268_cfg_tbl);
10857
10858 if (board_config < 0 || board_config >= ALC268_MODEL_LAST) {
10859 printk(KERN_INFO "hda_codec: Unknown model for ALC268, "
10860 "trying auto-probe from BIOS...\n");
10861 board_config = ALC268_AUTO;
10862 }
10863
10864 if (board_config == ALC268_AUTO) {
10865 /* automatic parse from the BIOS config */
10866 err = alc268_parse_auto_config(codec);
10867 if (err < 0) {
10868 alc_free(codec);
10869 return err;
10870 } else if (!err) {
10871 printk(KERN_INFO
10872 "hda_codec: Cannot set up configuration "
10873 "from BIOS. Using base mode...\n");
10874 board_config = ALC268_3ST;
10875 }
10876 }
10877
10878 if (board_config != ALC268_AUTO)
10879 setup_preset(spec, &alc268_presets[board_config]);
10880
Kailang Yang2f893282008-05-27 12:14:47 +020010881 if (codec->vendor_id == 0x10ec0267) {
10882 spec->stream_name_analog = "ALC267 Analog";
10883 spec->stream_name_digital = "ALC267 Digital";
10884 } else {
10885 spec->stream_name_analog = "ALC268 Analog";
10886 spec->stream_name_digital = "ALC268 Digital";
10887 }
10888
Kailang Yanga361d842007-06-05 12:30:55 +020010889 spec->stream_analog_playback = &alc268_pcm_analog_playback;
10890 spec->stream_analog_capture = &alc268_pcm_analog_capture;
Takashi Iwai63300792008-01-24 15:31:36 +010010891 spec->stream_analog_alt_capture = &alc268_pcm_analog_alt_capture;
Kailang Yanga361d842007-06-05 12:30:55 +020010892
Kailang Yanga361d842007-06-05 12:30:55 +020010893 spec->stream_digital_playback = &alc268_pcm_digital_playback;
10894
Takashi Iwaiaef9d312008-02-19 13:16:41 +010010895 if (!query_amp_caps(codec, 0x1d, HDA_INPUT))
10896 /* override the amp caps for beep generator */
10897 snd_hda_override_amp_caps(codec, 0x1d, HDA_INPUT,
10898 (0x0c << AC_AMPCAP_OFFSET_SHIFT) |
10899 (0x0c << AC_AMPCAP_NUM_STEPS_SHIFT) |
10900 (0x07 << AC_AMPCAP_STEP_SIZE_SHIFT) |
10901 (0 << AC_AMPCAP_MUTE_SHIFT));
10902
Takashi Iwai3866f0b2008-01-15 12:37:42 +010010903 if (!spec->adc_nids && spec->input_mux) {
10904 /* check whether NID 0x07 is valid */
10905 unsigned int wcap = get_wcaps(codec, 0x07);
Takashi Iwai85860c02008-02-19 15:00:15 +010010906 int i;
Kailang Yanga361d842007-06-05 12:30:55 +020010907
Takashi Iwai3866f0b2008-01-15 12:37:42 +010010908 /* get type */
10909 wcap = (wcap & AC_WCAP_TYPE) >> AC_WCAP_TYPE_SHIFT;
Takashi Iwai67ebcb02008-02-19 15:03:57 +010010910 if (wcap != AC_WID_AUD_IN || spec->input_mux->num_items == 1) {
Takashi Iwai3866f0b2008-01-15 12:37:42 +010010911 spec->adc_nids = alc268_adc_nids_alt;
10912 spec->num_adc_nids = ARRAY_SIZE(alc268_adc_nids_alt);
10913 spec->mixers[spec->num_mixers] =
Kailang Yanga361d842007-06-05 12:30:55 +020010914 alc268_capture_alt_mixer;
Takashi Iwai3866f0b2008-01-15 12:37:42 +010010915 spec->num_mixers++;
10916 } else {
10917 spec->adc_nids = alc268_adc_nids;
10918 spec->num_adc_nids = ARRAY_SIZE(alc268_adc_nids);
10919 spec->mixers[spec->num_mixers] =
10920 alc268_capture_mixer;
10921 spec->num_mixers++;
Kailang Yanga361d842007-06-05 12:30:55 +020010922 }
Takashi Iwaie1406342008-02-11 18:32:32 +010010923 spec->capsrc_nids = alc268_capsrc_nids;
Takashi Iwai85860c02008-02-19 15:00:15 +010010924 /* set default input source */
10925 for (i = 0; i < spec->num_adc_nids; i++)
10926 snd_hda_codec_write_cache(codec, alc268_capsrc_nids[i],
10927 0, AC_VERB_SET_CONNECT_SEL,
10928 spec->input_mux->items[0].index);
Kailang Yanga361d842007-06-05 12:30:55 +020010929 }
Takashi Iwai2134ea42008-01-10 16:53:55 +010010930
10931 spec->vmaster_nid = 0x02;
10932
Kailang Yanga361d842007-06-05 12:30:55 +020010933 codec->patch_ops = alc_patch_ops;
10934 if (board_config == ALC268_AUTO)
10935 spec->init_hook = alc268_auto_init;
10936
10937 return 0;
10938}
10939
10940/*
Kailang Yangf6a92242007-12-13 16:52:54 +010010941 * ALC269 channel source setting (2 channel)
10942 */
10943#define ALC269_DIGOUT_NID ALC880_DIGOUT_NID
10944
10945#define alc269_dac_nids alc260_dac_nids
10946
10947static hda_nid_t alc269_adc_nids[1] = {
10948 /* ADC1 */
10949 0x07,
10950};
10951
10952#define alc269_modes alc260_modes
10953#define alc269_capture_source alc880_lg_lw_capture_source
10954
10955static struct snd_kcontrol_new alc269_base_mixer[] = {
10956 HDA_CODEC_VOLUME("Front Playback Volume", 0x02, 0x0, HDA_OUTPUT),
10957 HDA_CODEC_MUTE("Front Playback Switch", 0x14, 0x0, HDA_OUTPUT),
10958 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
10959 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
10960 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
10961 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
10962 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
10963 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
10964 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
10965 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
10966 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
10967 HDA_CODEC_MUTE_MONO("Mono Playback Switch", 0x16, 2, 0x0, HDA_OUTPUT),
10968 { } /* end */
10969};
10970
10971/* capture mixer elements */
10972static struct snd_kcontrol_new alc269_capture_mixer[] = {
10973 HDA_CODEC_VOLUME("Capture Volume", 0x07, 0x0, HDA_INPUT),
10974 HDA_CODEC_MUTE("Capture Switch", 0x07, 0x0, HDA_INPUT),
10975 {
10976 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
10977 /* The multiple "Capture Source" controls confuse alsamixer
10978 * So call somewhat different..
Kailang Yangf6a92242007-12-13 16:52:54 +010010979 */
10980 /* .name = "Capture Source", */
10981 .name = "Input Source",
10982 .count = 1,
10983 .info = alc_mux_enum_info,
10984 .get = alc_mux_enum_get,
10985 .put = alc_mux_enum_put,
10986 },
10987 { } /* end */
10988};
10989
10990/*
10991 * generic initialization of ADC, input mixers and output mixers
10992 */
10993static struct hda_verb alc269_init_verbs[] = {
10994 /*
10995 * Unmute ADC0 and set the default input to mic-in
10996 */
10997 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10998
10999 /* Mute input amps (PCBeep, Line In, Mic 1 & Mic 2) of the
11000 * analog-loopback mixer widget
11001 * Note: PASD motherboards uses the Line In 2 as the input for
11002 * front panel mic (mic 2)
11003 */
11004 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
11005 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
11006 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
11007 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
11008 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
11009 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
11010
11011 /*
11012 * Set up output mixers (0x0c - 0x0e)
11013 */
11014 /* set vol=0 to output mixers */
11015 {0x02, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
11016 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
11017
11018 /* set up input amps for analog loopback */
11019 /* Amp Indices: DAC = 0, mixer = 1 */
11020 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11021 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11022 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11023 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11024 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11025 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11026
11027 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
11028 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
11029 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
11030 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
11031 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
11032 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
11033 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
11034
11035 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11036 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11037 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
11038 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
11039 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
11040 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
11041 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
11042
11043 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
11044 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
11045
11046 /* FIXME: use matrix-type input source selection */
11047 /* Mixer elements: 0x18, 19, 1a, 1b, 1d, 0b */
11048 /* Input mixer1: unmute Mic, F-Mic, Line, CD inputs */
11049 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11050 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
11051 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
11052 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
11053
11054 /* set EAPD */
11055 {0x14, AC_VERB_SET_EAPD_BTLENABLE, 2},
11056 {0x15, AC_VERB_SET_EAPD_BTLENABLE, 2},
11057 { }
11058};
11059
11060/* add playback controls from the parsed DAC table */
11061static int alc269_auto_create_multi_out_ctls(struct alc_spec *spec,
11062 const struct auto_pin_cfg *cfg)
11063{
11064 hda_nid_t nid;
11065 int err;
11066
11067 spec->multiout.num_dacs = 1; /* only use one dac */
11068 spec->multiout.dac_nids = spec->private_dac_nids;
11069 spec->multiout.dac_nids[0] = 2;
11070
11071 nid = cfg->line_out_pins[0];
11072 if (nid) {
11073 err = add_control(spec, ALC_CTL_WIDGET_VOL,
11074 "Front Playback Volume",
11075 HDA_COMPOSE_AMP_VAL(0x02, 3, 0, HDA_OUTPUT));
11076 if (err < 0)
11077 return err;
11078 err = add_control(spec, ALC_CTL_WIDGET_MUTE,
11079 "Front Playback Switch",
11080 HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_OUTPUT));
11081 if (err < 0)
11082 return err;
11083 }
11084
11085 nid = cfg->speaker_pins[0];
11086 if (nid) {
11087 if (!cfg->line_out_pins[0]) {
11088 err = add_control(spec, ALC_CTL_WIDGET_VOL,
11089 "Speaker Playback Volume",
11090 HDA_COMPOSE_AMP_VAL(0x02, 3, 0,
11091 HDA_OUTPUT));
11092 if (err < 0)
11093 return err;
11094 }
11095 if (nid == 0x16) {
11096 err = add_control(spec, ALC_CTL_WIDGET_MUTE,
11097 "Speaker Playback Switch",
11098 HDA_COMPOSE_AMP_VAL(nid, 2, 0,
11099 HDA_OUTPUT));
11100 if (err < 0)
11101 return err;
11102 } else {
11103 err = add_control(spec, ALC_CTL_WIDGET_MUTE,
11104 "Speaker Playback Switch",
11105 HDA_COMPOSE_AMP_VAL(nid, 3, 0,
11106 HDA_OUTPUT));
11107 if (err < 0)
11108 return err;
11109 }
11110 }
11111 nid = cfg->hp_pins[0];
11112 if (nid) {
11113 /* spec->multiout.hp_nid = 2; */
11114 if (!cfg->line_out_pins[0] && !cfg->speaker_pins[0]) {
11115 err = add_control(spec, ALC_CTL_WIDGET_VOL,
11116 "Headphone Playback Volume",
11117 HDA_COMPOSE_AMP_VAL(0x02, 3, 0,
11118 HDA_OUTPUT));
11119 if (err < 0)
11120 return err;
11121 }
11122 if (nid == 0x16) {
11123 err = add_control(spec, ALC_CTL_WIDGET_MUTE,
11124 "Headphone Playback Switch",
11125 HDA_COMPOSE_AMP_VAL(nid, 2, 0,
11126 HDA_OUTPUT));
11127 if (err < 0)
11128 return err;
11129 } else {
11130 err = add_control(spec, ALC_CTL_WIDGET_MUTE,
11131 "Headphone Playback Switch",
11132 HDA_COMPOSE_AMP_VAL(nid, 3, 0,
11133 HDA_OUTPUT));
11134 if (err < 0)
11135 return err;
11136 }
11137 }
11138 return 0;
11139}
11140
11141#define alc269_auto_create_analog_input_ctls \
11142 alc880_auto_create_analog_input_ctls
11143
11144#ifdef CONFIG_SND_HDA_POWER_SAVE
11145#define alc269_loopbacks alc880_loopbacks
11146#endif
11147
11148/* pcm configuration: identiacal with ALC880 */
11149#define alc269_pcm_analog_playback alc880_pcm_analog_playback
11150#define alc269_pcm_analog_capture alc880_pcm_analog_capture
11151#define alc269_pcm_digital_playback alc880_pcm_digital_playback
11152#define alc269_pcm_digital_capture alc880_pcm_digital_capture
11153
11154/*
11155 * BIOS auto configuration
11156 */
11157static int alc269_parse_auto_config(struct hda_codec *codec)
11158{
11159 struct alc_spec *spec = codec->spec;
11160 int err;
11161 static hda_nid_t alc269_ignore[] = { 0x1d, 0 };
11162
11163 err = snd_hda_parse_pin_def_config(codec, &spec->autocfg,
11164 alc269_ignore);
11165 if (err < 0)
11166 return err;
11167
11168 err = alc269_auto_create_multi_out_ctls(spec, &spec->autocfg);
11169 if (err < 0)
11170 return err;
11171 err = alc269_auto_create_analog_input_ctls(spec, &spec->autocfg);
11172 if (err < 0)
11173 return err;
11174
11175 spec->multiout.max_channels = spec->multiout.num_dacs * 2;
11176
11177 if (spec->autocfg.dig_out_pin)
11178 spec->multiout.dig_out_nid = ALC269_DIGOUT_NID;
11179
11180 if (spec->kctl_alloc)
11181 spec->mixers[spec->num_mixers++] = spec->kctl_alloc;
11182
11183 spec->init_verbs[spec->num_init_verbs++] = alc269_init_verbs;
11184 spec->num_mux_defs = 1;
11185 spec->input_mux = &spec->private_imux;
11186
11187 err = alc_auto_add_mic_boost(codec);
11188 if (err < 0)
11189 return err;
11190
11191 return 1;
11192}
11193
11194#define alc269_auto_init_multi_out alc882_auto_init_multi_out
11195#define alc269_auto_init_hp_out alc882_auto_init_hp_out
11196#define alc269_auto_init_analog_input alc882_auto_init_analog_input
11197
11198
11199/* init callback for auto-configuration model -- overriding the default init */
11200static void alc269_auto_init(struct hda_codec *codec)
11201{
Takashi Iwaif6c7e542008-02-12 18:32:23 +010011202 struct alc_spec *spec = codec->spec;
Kailang Yangf6a92242007-12-13 16:52:54 +010011203 alc269_auto_init_multi_out(codec);
11204 alc269_auto_init_hp_out(codec);
11205 alc269_auto_init_analog_input(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +010011206 if (spec->unsol_event)
11207 alc_sku_automute(codec);
Kailang Yangf6a92242007-12-13 16:52:54 +010011208}
11209
11210/*
11211 * configuration and preset
11212 */
11213static const char *alc269_models[ALC269_MODEL_LAST] = {
11214 [ALC269_BASIC] = "basic",
11215};
11216
11217static struct snd_pci_quirk alc269_cfg_tbl[] = {
11218 {}
11219};
11220
11221static struct alc_config_preset alc269_presets[] = {
11222 [ALC269_BASIC] = {
11223 .mixers = { alc269_base_mixer },
11224 .init_verbs = { alc269_init_verbs },
11225 .num_dacs = ARRAY_SIZE(alc269_dac_nids),
11226 .dac_nids = alc269_dac_nids,
11227 .hp_nid = 0x03,
11228 .num_channel_mode = ARRAY_SIZE(alc269_modes),
11229 .channel_mode = alc269_modes,
11230 .input_mux = &alc269_capture_source,
11231 },
11232};
11233
11234static int patch_alc269(struct hda_codec *codec)
11235{
11236 struct alc_spec *spec;
11237 int board_config;
11238 int err;
11239
11240 spec = kzalloc(sizeof(*spec), GFP_KERNEL);
11241 if (spec == NULL)
11242 return -ENOMEM;
11243
11244 codec->spec = spec;
11245
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +020011246 alc_fix_pll_init(codec, 0x20, 0x04, 15);
11247
Kailang Yangf6a92242007-12-13 16:52:54 +010011248 board_config = snd_hda_check_board_config(codec, ALC269_MODEL_LAST,
11249 alc269_models,
11250 alc269_cfg_tbl);
11251
11252 if (board_config < 0) {
11253 printk(KERN_INFO "hda_codec: Unknown model for ALC269, "
11254 "trying auto-probe from BIOS...\n");
11255 board_config = ALC269_AUTO;
11256 }
11257
11258 if (board_config == ALC269_AUTO) {
11259 /* automatic parse from the BIOS config */
11260 err = alc269_parse_auto_config(codec);
11261 if (err < 0) {
11262 alc_free(codec);
11263 return err;
11264 } else if (!err) {
11265 printk(KERN_INFO
11266 "hda_codec: Cannot set up configuration "
11267 "from BIOS. Using base mode...\n");
11268 board_config = ALC269_BASIC;
11269 }
11270 }
11271
11272 if (board_config != ALC269_AUTO)
11273 setup_preset(spec, &alc269_presets[board_config]);
11274
11275 spec->stream_name_analog = "ALC269 Analog";
11276 spec->stream_analog_playback = &alc269_pcm_analog_playback;
11277 spec->stream_analog_capture = &alc269_pcm_analog_capture;
11278
11279 spec->stream_name_digital = "ALC269 Digital";
11280 spec->stream_digital_playback = &alc269_pcm_digital_playback;
11281 spec->stream_digital_capture = &alc269_pcm_digital_capture;
11282
11283 spec->adc_nids = alc269_adc_nids;
11284 spec->num_adc_nids = ARRAY_SIZE(alc269_adc_nids);
11285 spec->mixers[spec->num_mixers] = alc269_capture_mixer;
11286 spec->num_mixers++;
11287
11288 codec->patch_ops = alc_patch_ops;
11289 if (board_config == ALC269_AUTO)
11290 spec->init_hook = alc269_auto_init;
11291#ifdef CONFIG_SND_HDA_POWER_SAVE
11292 if (!spec->loopback.amplist)
11293 spec->loopback.amplist = alc269_loopbacks;
11294#endif
11295
11296 return 0;
11297}
11298
11299/*
Kailang Yangdf694da2005-12-05 19:42:22 +010011300 * ALC861 channel source setting (2/6 channel selection for 3-stack)
11301 */
11302
11303/*
11304 * set the path ways for 2 channel output
11305 * need to set the codec line out and mic 1 pin widgets to inputs
11306 */
11307static struct hda_verb alc861_threestack_ch2_init[] = {
11308 /* set pin widget 1Ah (line in) for input */
11309 { 0x0c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011310 /* set pin widget 18h (mic1/2) for input, for mic also enable
11311 * the vref
11312 */
Kailang Yangdf694da2005-12-05 19:42:22 +010011313 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
11314
Takashi Iwai9c7f8522006-06-28 15:08:22 +020011315 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0xb00c },
11316#if 0
11317 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8)) }, /*mic*/
11318 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x02 << 8)) }, /*line-in*/
11319#endif
Kailang Yangdf694da2005-12-05 19:42:22 +010011320 { } /* end */
11321};
11322/*
11323 * 6ch mode
11324 * need to set the codec line out and mic 1 pin widgets to outputs
11325 */
11326static struct hda_verb alc861_threestack_ch6_init[] = {
11327 /* set pin widget 1Ah (line in) for output (Back Surround)*/
11328 { 0x0c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
11329 /* set pin widget 18h (mic1) for output (CLFE)*/
11330 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
11331
11332 { 0x0c, AC_VERB_SET_CONNECT_SEL, 0x00 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +020011333 { 0x0d, AC_VERB_SET_CONNECT_SEL, 0x00 },
Kailang Yangdf694da2005-12-05 19:42:22 +010011334
Takashi Iwai9c7f8522006-06-28 15:08:22 +020011335 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0xb080 },
11336#if 0
11337 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x01 << 8)) }, /*mic*/
11338 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8)) }, /*line in*/
11339#endif
Kailang Yangdf694da2005-12-05 19:42:22 +010011340 { } /* end */
11341};
11342
11343static struct hda_channel_mode alc861_threestack_modes[2] = {
11344 { 2, alc861_threestack_ch2_init },
11345 { 6, alc861_threestack_ch6_init },
11346};
Takashi Iwai22309c32006-08-09 16:57:28 +020011347/* Set mic1 as input and unmute the mixer */
11348static struct hda_verb alc861_uniwill_m31_ch2_init[] = {
11349 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
11350 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x01 << 8)) }, /*mic*/
11351 { } /* end */
11352};
11353/* Set mic1 as output and mute mixer */
11354static struct hda_verb alc861_uniwill_m31_ch4_init[] = {
11355 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
11356 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8)) }, /*mic*/
11357 { } /* end */
11358};
11359
11360static struct hda_channel_mode alc861_uniwill_m31_modes[2] = {
11361 { 2, alc861_uniwill_m31_ch2_init },
11362 { 4, alc861_uniwill_m31_ch4_init },
11363};
Kailang Yangdf694da2005-12-05 19:42:22 +010011364
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020011365/* Set mic1 and line-in as input and unmute the mixer */
11366static struct hda_verb alc861_asus_ch2_init[] = {
11367 /* set pin widget 1Ah (line in) for input */
11368 { 0x0c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011369 /* set pin widget 18h (mic1/2) for input, for mic also enable
11370 * the vref
11371 */
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020011372 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
11373
11374 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0xb00c },
11375#if 0
11376 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8)) }, /*mic*/
11377 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x02 << 8)) }, /*line-in*/
11378#endif
11379 { } /* end */
11380};
11381/* Set mic1 nad line-in as output and mute mixer */
11382static struct hda_verb alc861_asus_ch6_init[] = {
11383 /* set pin widget 1Ah (line in) for output (Back Surround)*/
11384 { 0x0c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
11385 /* { 0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE }, */
11386 /* set pin widget 18h (mic1) for output (CLFE)*/
11387 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
11388 /* { 0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE }, */
11389 { 0x0c, AC_VERB_SET_CONNECT_SEL, 0x00 },
11390 { 0x0d, AC_VERB_SET_CONNECT_SEL, 0x00 },
11391
11392 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0xb080 },
11393#if 0
11394 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x01 << 8)) }, /*mic*/
11395 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8)) }, /*line in*/
11396#endif
11397 { } /* end */
11398};
11399
11400static struct hda_channel_mode alc861_asus_modes[2] = {
11401 { 2, alc861_asus_ch2_init },
11402 { 6, alc861_asus_ch6_init },
11403};
11404
Kailang Yangdf694da2005-12-05 19:42:22 +010011405/* patch-ALC861 */
11406
11407static struct snd_kcontrol_new alc861_base_mixer[] = {
11408 /* output mixer control */
11409 HDA_CODEC_MUTE("Front Playback Switch", 0x03, 0x0, HDA_OUTPUT),
11410 HDA_CODEC_MUTE("Surround Playback Switch", 0x06, 0x0, HDA_OUTPUT),
11411 HDA_CODEC_MUTE_MONO("Center Playback Switch", 0x05, 1, 0x0, HDA_OUTPUT),
11412 HDA_CODEC_MUTE_MONO("LFE Playback Switch", 0x05, 2, 0x0, HDA_OUTPUT),
11413 HDA_CODEC_MUTE("Side Playback Switch", 0x04, 0x0, HDA_OUTPUT),
11414
11415 /*Input mixer control */
11416 /* HDA_CODEC_VOLUME("Input Playback Volume", 0x15, 0x0, HDA_OUTPUT),
11417 HDA_CODEC_MUTE("Input Playback Switch", 0x15, 0x0, HDA_OUTPUT), */
11418 HDA_CODEC_VOLUME("CD Playback Volume", 0x15, 0x0, HDA_INPUT),
11419 HDA_CODEC_MUTE("CD Playback Switch", 0x15, 0x0, HDA_INPUT),
11420 HDA_CODEC_VOLUME("Line Playback Volume", 0x15, 0x02, HDA_INPUT),
11421 HDA_CODEC_MUTE("Line Playback Switch", 0x15, 0x02, HDA_INPUT),
11422 HDA_CODEC_VOLUME("Mic Playback Volume", 0x15, 0x01, HDA_INPUT),
11423 HDA_CODEC_MUTE("Mic Playback Switch", 0x15, 0x01, HDA_INPUT),
11424 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x10, 0x01, HDA_OUTPUT),
11425 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1a, 0x03, HDA_INPUT),
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011426
Kailang Yangdf694da2005-12-05 19:42:22 +010011427 /* Capture mixer control */
11428 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
11429 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
11430 {
11431 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
11432 .name = "Capture Source",
11433 .count = 1,
11434 .info = alc_mux_enum_info,
11435 .get = alc_mux_enum_get,
11436 .put = alc_mux_enum_put,
11437 },
11438 { } /* end */
11439};
11440
11441static struct snd_kcontrol_new alc861_3ST_mixer[] = {
11442 /* output mixer control */
11443 HDA_CODEC_MUTE("Front Playback Switch", 0x03, 0x0, HDA_OUTPUT),
11444 HDA_CODEC_MUTE("Surround Playback Switch", 0x06, 0x0, HDA_OUTPUT),
11445 HDA_CODEC_MUTE_MONO("Center Playback Switch", 0x05, 1, 0x0, HDA_OUTPUT),
11446 HDA_CODEC_MUTE_MONO("LFE Playback Switch", 0x05, 2, 0x0, HDA_OUTPUT),
11447 /*HDA_CODEC_MUTE("Side Playback Switch", 0x04, 0x0, HDA_OUTPUT), */
11448
11449 /* Input mixer control */
11450 /* HDA_CODEC_VOLUME("Input Playback Volume", 0x15, 0x0, HDA_OUTPUT),
11451 HDA_CODEC_MUTE("Input Playback Switch", 0x15, 0x0, HDA_OUTPUT), */
11452 HDA_CODEC_VOLUME("CD Playback Volume", 0x15, 0x0, HDA_INPUT),
11453 HDA_CODEC_MUTE("CD Playback Switch", 0x15, 0x0, HDA_INPUT),
11454 HDA_CODEC_VOLUME("Line Playback Volume", 0x15, 0x02, HDA_INPUT),
11455 HDA_CODEC_MUTE("Line Playback Switch", 0x15, 0x02, HDA_INPUT),
11456 HDA_CODEC_VOLUME("Mic Playback Volume", 0x15, 0x01, HDA_INPUT),
11457 HDA_CODEC_MUTE("Mic Playback Switch", 0x15, 0x01, HDA_INPUT),
11458 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x10, 0x01, HDA_OUTPUT),
11459 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1a, 0x03, HDA_INPUT),
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011460
Kailang Yangdf694da2005-12-05 19:42:22 +010011461 /* Capture mixer control */
11462 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
11463 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
11464 {
11465 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
11466 .name = "Capture Source",
11467 .count = 1,
11468 .info = alc_mux_enum_info,
11469 .get = alc_mux_enum_get,
11470 .put = alc_mux_enum_put,
11471 },
11472 {
11473 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
11474 .name = "Channel Mode",
11475 .info = alc_ch_mode_info,
11476 .get = alc_ch_mode_get,
11477 .put = alc_ch_mode_put,
11478 .private_value = ARRAY_SIZE(alc861_threestack_modes),
11479 },
11480 { } /* end */
Tobin Davisa53d1ae2006-10-17 12:00:28 +020011481};
11482
Takashi Iwaid1d985f2006-11-23 19:27:12 +010011483static struct snd_kcontrol_new alc861_toshiba_mixer[] = {
Tobin Davisa53d1ae2006-10-17 12:00:28 +020011484 /* output mixer control */
11485 HDA_CODEC_MUTE("Master Playback Switch", 0x03, 0x0, HDA_OUTPUT),
11486 HDA_CODEC_VOLUME("Mic Playback Volume", 0x15, 0x01, HDA_INPUT),
11487 HDA_CODEC_MUTE("Mic Playback Switch", 0x15, 0x01, HDA_INPUT),
11488
11489 /*Capture mixer control */
11490 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
11491 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
11492 {
11493 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
11494 .name = "Capture Source",
11495 .count = 1,
11496 .info = alc_mux_enum_info,
11497 .get = alc_mux_enum_get,
11498 .put = alc_mux_enum_put,
11499 },
11500
11501 { } /* end */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011502};
Tobin Davisa53d1ae2006-10-17 12:00:28 +020011503
Takashi Iwai22309c32006-08-09 16:57:28 +020011504static struct snd_kcontrol_new alc861_uniwill_m31_mixer[] = {
11505 /* output mixer control */
11506 HDA_CODEC_MUTE("Front Playback Switch", 0x03, 0x0, HDA_OUTPUT),
11507 HDA_CODEC_MUTE("Surround Playback Switch", 0x06, 0x0, HDA_OUTPUT),
11508 HDA_CODEC_MUTE_MONO("Center Playback Switch", 0x05, 1, 0x0, HDA_OUTPUT),
11509 HDA_CODEC_MUTE_MONO("LFE Playback Switch", 0x05, 2, 0x0, HDA_OUTPUT),
11510 /*HDA_CODEC_MUTE("Side Playback Switch", 0x04, 0x0, HDA_OUTPUT), */
11511
11512 /* Input mixer control */
11513 /* HDA_CODEC_VOLUME("Input Playback Volume", 0x15, 0x0, HDA_OUTPUT),
11514 HDA_CODEC_MUTE("Input Playback Switch", 0x15, 0x0, HDA_OUTPUT), */
11515 HDA_CODEC_VOLUME("CD Playback Volume", 0x15, 0x0, HDA_INPUT),
11516 HDA_CODEC_MUTE("CD Playback Switch", 0x15, 0x0, HDA_INPUT),
11517 HDA_CODEC_VOLUME("Line Playback Volume", 0x15, 0x02, HDA_INPUT),
11518 HDA_CODEC_MUTE("Line Playback Switch", 0x15, 0x02, HDA_INPUT),
11519 HDA_CODEC_VOLUME("Mic Playback Volume", 0x15, 0x01, HDA_INPUT),
11520 HDA_CODEC_MUTE("Mic Playback Switch", 0x15, 0x01, HDA_INPUT),
11521 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x10, 0x01, HDA_OUTPUT),
11522 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1a, 0x03, HDA_INPUT),
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011523
Takashi Iwai22309c32006-08-09 16:57:28 +020011524 /* Capture mixer control */
11525 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
11526 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
11527 {
11528 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
11529 .name = "Capture Source",
11530 .count = 1,
11531 .info = alc_mux_enum_info,
11532 .get = alc_mux_enum_get,
11533 .put = alc_mux_enum_put,
11534 },
11535 {
11536 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
11537 .name = "Channel Mode",
11538 .info = alc_ch_mode_info,
11539 .get = alc_ch_mode_get,
11540 .put = alc_ch_mode_put,
11541 .private_value = ARRAY_SIZE(alc861_uniwill_m31_modes),
11542 },
11543 { } /* end */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011544};
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020011545
11546static struct snd_kcontrol_new alc861_asus_mixer[] = {
11547 /* output mixer control */
11548 HDA_CODEC_MUTE("Front Playback Switch", 0x03, 0x0, HDA_OUTPUT),
11549 HDA_CODEC_MUTE("Surround Playback Switch", 0x06, 0x0, HDA_OUTPUT),
11550 HDA_CODEC_MUTE_MONO("Center Playback Switch", 0x05, 1, 0x0, HDA_OUTPUT),
11551 HDA_CODEC_MUTE_MONO("LFE Playback Switch", 0x05, 2, 0x0, HDA_OUTPUT),
11552 HDA_CODEC_MUTE("Side Playback Switch", 0x04, 0x0, HDA_OUTPUT),
11553
11554 /* Input mixer control */
11555 HDA_CODEC_VOLUME("Input Playback Volume", 0x15, 0x0, HDA_OUTPUT),
11556 HDA_CODEC_MUTE("Input Playback Switch", 0x15, 0x0, HDA_OUTPUT),
11557 HDA_CODEC_VOLUME("CD Playback Volume", 0x15, 0x0, HDA_INPUT),
11558 HDA_CODEC_MUTE("CD Playback Switch", 0x15, 0x0, HDA_INPUT),
11559 HDA_CODEC_VOLUME("Line Playback Volume", 0x15, 0x02, HDA_INPUT),
11560 HDA_CODEC_MUTE("Line Playback Switch", 0x15, 0x02, HDA_INPUT),
11561 HDA_CODEC_VOLUME("Mic Playback Volume", 0x15, 0x01, HDA_INPUT),
11562 HDA_CODEC_MUTE("Mic Playback Switch", 0x15, 0x01, HDA_INPUT),
11563 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x10, 0x01, HDA_OUTPUT),
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011564 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1a, 0x03, HDA_OUTPUT),
11565
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020011566 /* Capture mixer control */
11567 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
11568 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
11569 {
11570 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
11571 .name = "Capture Source",
11572 .count = 1,
11573 .info = alc_mux_enum_info,
11574 .get = alc_mux_enum_get,
11575 .put = alc_mux_enum_put,
11576 },
11577 {
11578 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
11579 .name = "Channel Mode",
11580 .info = alc_ch_mode_info,
11581 .get = alc_ch_mode_get,
11582 .put = alc_ch_mode_put,
11583 .private_value = ARRAY_SIZE(alc861_asus_modes),
11584 },
11585 { }
Takashi Iwai56bb0ca2006-11-22 11:52:52 +010011586};
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020011587
Takashi Iwai56bb0ca2006-11-22 11:52:52 +010011588/* additional mixer */
Takashi Iwaid1d985f2006-11-23 19:27:12 +010011589static struct snd_kcontrol_new alc861_asus_laptop_mixer[] = {
Takashi Iwai56bb0ca2006-11-22 11:52:52 +010011590 HDA_CODEC_VOLUME("CD Playback Volume", 0x15, 0x0, HDA_INPUT),
11591 HDA_CODEC_MUTE("CD Playback Switch", 0x15, 0x0, HDA_INPUT),
11592 HDA_CODEC_VOLUME("PC Beep Playback Volume", 0x23, 0x0, HDA_OUTPUT),
11593 HDA_CODEC_MUTE("PC Beep Playback Switch", 0x23, 0x0, HDA_OUTPUT),
11594 { }
11595};
11596
Kailang Yangdf694da2005-12-05 19:42:22 +010011597/*
11598 * generic initialization of ADC, input mixers and output mixers
11599 */
11600static struct hda_verb alc861_base_init_verbs[] = {
11601 /*
11602 * Unmute ADC0 and set the default input to mic-in
11603 */
11604 /* port-A for surround (rear panel) */
11605 { 0x0e, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
11606 { 0x0e, AC_VERB_SET_CONNECT_SEL, 0x00 },
11607 /* port-B for mic-in (rear panel) with vref */
11608 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
11609 /* port-C for line-in (rear panel) */
11610 { 0x0c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
11611 /* port-D for Front */
11612 { 0x0b, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
11613 { 0x0b, AC_VERB_SET_CONNECT_SEL, 0x00 },
11614 /* port-E for HP out (front panel) */
11615 { 0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc0 },
11616 /* route front PCM to HP */
Takashi Iwai9dece1d2006-11-16 17:12:49 +010011617 { 0x0f, AC_VERB_SET_CONNECT_SEL, 0x00 },
Kailang Yangdf694da2005-12-05 19:42:22 +010011618 /* port-F for mic-in (front panel) with vref */
11619 { 0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
11620 /* port-G for CLFE (rear panel) */
11621 { 0x1f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
11622 { 0x1f, AC_VERB_SET_CONNECT_SEL, 0x00 },
11623 /* port-H for side (rear panel) */
11624 { 0x20, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
11625 { 0x20, AC_VERB_SET_CONNECT_SEL, 0x00 },
11626 /* CD-in */
11627 { 0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
11628 /* route front mic to ADC1*/
11629 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
11630 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11631
11632 /* Unmute DAC0~3 & spdif out*/
11633 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11634 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11635 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11636 {0x06, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11637 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11638
11639 /* Unmute Mixer 14 (mic) 1c (Line in)*/
11640 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11641 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11642 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11643 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11644
11645 /* Unmute Stereo Mixer 15 */
11646 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11647 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11648 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011649 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0xb00c}, /* Output 0~12 step */
Kailang Yangdf694da2005-12-05 19:42:22 +010011650
11651 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11652 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11653 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11654 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11655 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11656 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11657 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11658 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011659 /* hp used DAC 3 (Front) */
11660 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(3)},
Kailang Yangdf694da2005-12-05 19:42:22 +010011661 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
11662
11663 { }
11664};
11665
11666static struct hda_verb alc861_threestack_init_verbs[] = {
11667 /*
11668 * Unmute ADC0 and set the default input to mic-in
11669 */
11670 /* port-A for surround (rear panel) */
11671 { 0x0e, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
11672 /* port-B for mic-in (rear panel) with vref */
11673 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
11674 /* port-C for line-in (rear panel) */
11675 { 0x0c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
11676 /* port-D for Front */
11677 { 0x0b, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
11678 { 0x0b, AC_VERB_SET_CONNECT_SEL, 0x00 },
11679 /* port-E for HP out (front panel) */
11680 { 0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc0 },
11681 /* route front PCM to HP */
Takashi Iwai9dece1d2006-11-16 17:12:49 +010011682 { 0x0f, AC_VERB_SET_CONNECT_SEL, 0x00 },
Kailang Yangdf694da2005-12-05 19:42:22 +010011683 /* port-F for mic-in (front panel) with vref */
11684 { 0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
11685 /* port-G for CLFE (rear panel) */
11686 { 0x1f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
11687 /* port-H for side (rear panel) */
11688 { 0x20, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
11689 /* CD-in */
11690 { 0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
11691 /* route front mic to ADC1*/
11692 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
11693 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11694 /* Unmute DAC0~3 & spdif out*/
11695 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11696 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11697 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11698 {0x06, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11699 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11700
11701 /* Unmute Mixer 14 (mic) 1c (Line in)*/
11702 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11703 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11704 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11705 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11706
11707 /* Unmute Stereo Mixer 15 */
11708 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11709 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11710 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011711 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0xb00c}, /* Output 0~12 step */
Kailang Yangdf694da2005-12-05 19:42:22 +010011712
11713 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11714 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11715 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11716 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11717 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11718 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11719 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11720 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011721 /* hp used DAC 3 (Front) */
11722 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(3)},
Kailang Yangdf694da2005-12-05 19:42:22 +010011723 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
11724 { }
11725};
Takashi Iwai22309c32006-08-09 16:57:28 +020011726
11727static struct hda_verb alc861_uniwill_m31_init_verbs[] = {
11728 /*
11729 * Unmute ADC0 and set the default input to mic-in
11730 */
11731 /* port-A for surround (rear panel) */
11732 { 0x0e, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
11733 /* port-B for mic-in (rear panel) with vref */
11734 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
11735 /* port-C for line-in (rear panel) */
11736 { 0x0c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
11737 /* port-D for Front */
11738 { 0x0b, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
11739 { 0x0b, AC_VERB_SET_CONNECT_SEL, 0x00 },
11740 /* port-E for HP out (front panel) */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011741 /* this has to be set to VREF80 */
11742 { 0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
Takashi Iwai22309c32006-08-09 16:57:28 +020011743 /* route front PCM to HP */
Takashi Iwai9dece1d2006-11-16 17:12:49 +010011744 { 0x0f, AC_VERB_SET_CONNECT_SEL, 0x00 },
Takashi Iwai22309c32006-08-09 16:57:28 +020011745 /* port-F for mic-in (front panel) with vref */
11746 { 0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
11747 /* port-G for CLFE (rear panel) */
11748 { 0x1f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
11749 /* port-H for side (rear panel) */
11750 { 0x20, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
11751 /* CD-in */
11752 { 0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
11753 /* route front mic to ADC1*/
11754 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
11755 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11756 /* Unmute DAC0~3 & spdif out*/
11757 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11758 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11759 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11760 {0x06, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11761 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11762
11763 /* Unmute Mixer 14 (mic) 1c (Line in)*/
11764 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11765 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11766 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11767 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11768
11769 /* Unmute Stereo Mixer 15 */
11770 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11771 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11772 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011773 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0xb00c}, /* Output 0~12 step */
Takashi Iwai22309c32006-08-09 16:57:28 +020011774
11775 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11776 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11777 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11778 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11779 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11780 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11781 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11782 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011783 /* hp used DAC 3 (Front) */
11784 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(3)},
Takashi Iwai22309c32006-08-09 16:57:28 +020011785 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
11786 { }
11787};
11788
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020011789static struct hda_verb alc861_asus_init_verbs[] = {
11790 /*
11791 * Unmute ADC0 and set the default input to mic-in
11792 */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011793 /* port-A for surround (rear panel)
11794 * according to codec#0 this is the HP jack
11795 */
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020011796 { 0x0e, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc0 }, /* was 0x00 */
11797 /* route front PCM to HP */
11798 { 0x0e, AC_VERB_SET_CONNECT_SEL, 0x01 },
11799 /* port-B for mic-in (rear panel) with vref */
11800 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
11801 /* port-C for line-in (rear panel) */
11802 { 0x0c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
11803 /* port-D for Front */
11804 { 0x0b, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
11805 { 0x0b, AC_VERB_SET_CONNECT_SEL, 0x00 },
11806 /* port-E for HP out (front panel) */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011807 /* this has to be set to VREF80 */
11808 { 0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020011809 /* route front PCM to HP */
Takashi Iwai9dece1d2006-11-16 17:12:49 +010011810 { 0x0f, AC_VERB_SET_CONNECT_SEL, 0x00 },
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020011811 /* port-F for mic-in (front panel) with vref */
11812 { 0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
11813 /* port-G for CLFE (rear panel) */
11814 { 0x1f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
11815 /* port-H for side (rear panel) */
11816 { 0x20, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
11817 /* CD-in */
11818 { 0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
11819 /* route front mic to ADC1*/
11820 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
11821 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11822 /* Unmute DAC0~3 & spdif out*/
11823 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11824 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11825 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11826 {0x06, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11827 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11828 /* Unmute Mixer 14 (mic) 1c (Line in)*/
11829 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11830 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11831 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11832 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11833
11834 /* Unmute Stereo Mixer 15 */
11835 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11836 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11837 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011838 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0xb00c}, /* Output 0~12 step */
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020011839
11840 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11841 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11842 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11843 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11844 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11845 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11846 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11847 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011848 /* hp used DAC 3 (Front) */
11849 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(3)},
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020011850 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
11851 { }
11852};
11853
Takashi Iwai56bb0ca2006-11-22 11:52:52 +010011854/* additional init verbs for ASUS laptops */
11855static struct hda_verb alc861_asus_laptop_init_verbs[] = {
11856 { 0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x45 }, /* HP-out */
11857 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2) }, /* mute line-in */
11858 { }
11859};
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020011860
Kailang Yangdf694da2005-12-05 19:42:22 +010011861/*
11862 * generic initialization of ADC, input mixers and output mixers
11863 */
11864static struct hda_verb alc861_auto_init_verbs[] = {
11865 /*
11866 * Unmute ADC0 and set the default input to mic-in
11867 */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011868 /* {0x08, AC_VERB_SET_CONNECT_SEL, 0x00}, */
Kailang Yangdf694da2005-12-05 19:42:22 +010011869 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11870
11871 /* Unmute DAC0~3 & spdif out*/
11872 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
11873 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
11874 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
11875 {0x06, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
11876 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
11877
11878 /* Unmute Mixer 14 (mic) 1c (Line in)*/
11879 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11880 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11881 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11882 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11883
11884 /* Unmute Stereo Mixer 15 */
11885 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11886 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11887 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
11888 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0xb00c},
11889
11890 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11891 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11892 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11893 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11894 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11895 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11896 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11897 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11898
11899 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
11900 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011901 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
11902 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(3)},
Kailang Yangdf694da2005-12-05 19:42:22 +010011903 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
11904 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011905 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
11906 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(3)},
Kailang Yangdf694da2005-12-05 19:42:22 +010011907
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011908 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00}, /* set Mic 1 */
Kailang Yangdf694da2005-12-05 19:42:22 +010011909
11910 { }
11911};
11912
Tobin Davisa53d1ae2006-10-17 12:00:28 +020011913static struct hda_verb alc861_toshiba_init_verbs[] = {
11914 {0x0f, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011915
Tobin Davisa53d1ae2006-10-17 12:00:28 +020011916 { }
11917};
11918
11919/* toggle speaker-output according to the hp-jack state */
11920static void alc861_toshiba_automute(struct hda_codec *codec)
11921{
11922 unsigned int present;
11923
11924 present = snd_hda_codec_read(codec, 0x0f, 0,
11925 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +020011926 snd_hda_codec_amp_stereo(codec, 0x16, HDA_INPUT, 0,
11927 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
11928 snd_hda_codec_amp_stereo(codec, 0x1a, HDA_INPUT, 3,
11929 HDA_AMP_MUTE, present ? 0 : HDA_AMP_MUTE);
Tobin Davisa53d1ae2006-10-17 12:00:28 +020011930}
11931
11932static void alc861_toshiba_unsol_event(struct hda_codec *codec,
11933 unsigned int res)
11934{
Tobin Davisa53d1ae2006-10-17 12:00:28 +020011935 if ((res >> 26) == ALC880_HP_EVENT)
11936 alc861_toshiba_automute(codec);
11937}
11938
Kailang Yangdf694da2005-12-05 19:42:22 +010011939/* pcm configuration: identiacal with ALC880 */
11940#define alc861_pcm_analog_playback alc880_pcm_analog_playback
11941#define alc861_pcm_analog_capture alc880_pcm_analog_capture
11942#define alc861_pcm_digital_playback alc880_pcm_digital_playback
11943#define alc861_pcm_digital_capture alc880_pcm_digital_capture
11944
11945
11946#define ALC861_DIGOUT_NID 0x07
11947
11948static struct hda_channel_mode alc861_8ch_modes[1] = {
11949 { 8, NULL }
11950};
11951
11952static hda_nid_t alc861_dac_nids[4] = {
11953 /* front, surround, clfe, side */
11954 0x03, 0x06, 0x05, 0x04
11955};
11956
Takashi Iwai9c7f8522006-06-28 15:08:22 +020011957static hda_nid_t alc660_dac_nids[3] = {
11958 /* front, clfe, surround */
11959 0x03, 0x05, 0x06
11960};
11961
Kailang Yangdf694da2005-12-05 19:42:22 +010011962static hda_nid_t alc861_adc_nids[1] = {
11963 /* ADC0-2 */
11964 0x08,
11965};
11966
11967static struct hda_input_mux alc861_capture_source = {
11968 .num_items = 5,
11969 .items = {
11970 { "Mic", 0x0 },
11971 { "Front Mic", 0x3 },
11972 { "Line", 0x1 },
11973 { "CD", 0x4 },
11974 { "Mixer", 0x5 },
11975 },
11976};
11977
11978/* fill in the dac_nids table from the parsed pin configuration */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011979static int alc861_auto_fill_dac_nids(struct alc_spec *spec,
11980 const struct auto_pin_cfg *cfg)
Kailang Yangdf694da2005-12-05 19:42:22 +010011981{
11982 int i;
11983 hda_nid_t nid;
11984
11985 spec->multiout.dac_nids = spec->private_dac_nids;
11986 for (i = 0; i < cfg->line_outs; i++) {
11987 nid = cfg->line_out_pins[i];
11988 if (nid) {
11989 if (i >= ARRAY_SIZE(alc861_dac_nids))
11990 continue;
11991 spec->multiout.dac_nids[i] = alc861_dac_nids[i];
11992 }
11993 }
11994 spec->multiout.num_dacs = cfg->line_outs;
11995 return 0;
11996}
11997
11998/* add playback controls from the parsed DAC table */
11999static int alc861_auto_create_multi_out_ctls(struct alc_spec *spec,
12000 const struct auto_pin_cfg *cfg)
12001{
12002 char name[32];
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012003 static const char *chname[4] = {
12004 "Front", "Surround", NULL /*CLFE*/, "Side"
12005 };
Kailang Yangdf694da2005-12-05 19:42:22 +010012006 hda_nid_t nid;
12007 int i, idx, err;
12008
12009 for (i = 0; i < cfg->line_outs; i++) {
12010 nid = spec->multiout.dac_nids[i];
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012011 if (!nid)
Kailang Yangdf694da2005-12-05 19:42:22 +010012012 continue;
12013 if (nid == 0x05) {
12014 /* Center/LFE */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012015 err = add_control(spec, ALC_CTL_BIND_MUTE,
12016 "Center Playback Switch",
12017 HDA_COMPOSE_AMP_VAL(nid, 1, 0,
12018 HDA_OUTPUT));
12019 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +010012020 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012021 err = add_control(spec, ALC_CTL_BIND_MUTE,
12022 "LFE Playback Switch",
12023 HDA_COMPOSE_AMP_VAL(nid, 2, 0,
12024 HDA_OUTPUT));
12025 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +010012026 return err;
12027 } else {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012028 for (idx = 0; idx < ARRAY_SIZE(alc861_dac_nids) - 1;
12029 idx++)
Kailang Yangdf694da2005-12-05 19:42:22 +010012030 if (nid == alc861_dac_nids[idx])
12031 break;
12032 sprintf(name, "%s Playback Switch", chname[idx]);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012033 err = add_control(spec, ALC_CTL_BIND_MUTE, name,
12034 HDA_COMPOSE_AMP_VAL(nid, 3, 0,
12035 HDA_OUTPUT));
12036 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +010012037 return err;
12038 }
12039 }
12040 return 0;
12041}
12042
12043static int alc861_auto_create_hp_ctls(struct alc_spec *spec, hda_nid_t pin)
12044{
12045 int err;
12046 hda_nid_t nid;
12047
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012048 if (!pin)
Kailang Yangdf694da2005-12-05 19:42:22 +010012049 return 0;
12050
12051 if ((pin >= 0x0b && pin <= 0x10) || pin == 0x1f || pin == 0x20) {
12052 nid = 0x03;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012053 err = add_control(spec, ALC_CTL_WIDGET_MUTE,
12054 "Headphone Playback Switch",
12055 HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_OUTPUT));
12056 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +010012057 return err;
12058 spec->multiout.hp_nid = nid;
12059 }
12060 return 0;
12061}
12062
12063/* create playback/capture controls for input pins */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012064static int alc861_auto_create_analog_input_ctls(struct alc_spec *spec,
12065 const struct auto_pin_cfg *cfg)
Kailang Yangdf694da2005-12-05 19:42:22 +010012066{
Kailang Yangdf694da2005-12-05 19:42:22 +010012067 struct hda_input_mux *imux = &spec->private_imux;
12068 int i, err, idx, idx1;
12069
12070 for (i = 0; i < AUTO_PIN_LAST; i++) {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012071 switch (cfg->input_pins[i]) {
Kailang Yangdf694da2005-12-05 19:42:22 +010012072 case 0x0c:
12073 idx1 = 1;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012074 idx = 2; /* Line In */
Kailang Yangdf694da2005-12-05 19:42:22 +010012075 break;
12076 case 0x0f:
12077 idx1 = 2;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012078 idx = 2; /* Line In */
Kailang Yangdf694da2005-12-05 19:42:22 +010012079 break;
12080 case 0x0d:
12081 idx1 = 0;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012082 idx = 1; /* Mic In */
Kailang Yangdf694da2005-12-05 19:42:22 +010012083 break;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012084 case 0x10:
Kailang Yangdf694da2005-12-05 19:42:22 +010012085 idx1 = 3;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012086 idx = 1; /* Mic In */
Kailang Yangdf694da2005-12-05 19:42:22 +010012087 break;
12088 case 0x11:
12089 idx1 = 4;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012090 idx = 0; /* CD */
Kailang Yangdf694da2005-12-05 19:42:22 +010012091 break;
12092 default:
12093 continue;
12094 }
12095
Takashi Iwai4a471b72005-12-07 13:56:29 +010012096 err = new_analog_input(spec, cfg->input_pins[i],
12097 auto_pin_cfg_labels[i], idx, 0x15);
Kailang Yangdf694da2005-12-05 19:42:22 +010012098 if (err < 0)
12099 return err;
12100
Takashi Iwai4a471b72005-12-07 13:56:29 +010012101 imux->items[imux->num_items].label = auto_pin_cfg_labels[i];
Kailang Yangdf694da2005-12-05 19:42:22 +010012102 imux->items[imux->num_items].index = idx1;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012103 imux->num_items++;
Kailang Yangdf694da2005-12-05 19:42:22 +010012104 }
12105 return 0;
12106}
12107
12108static struct snd_kcontrol_new alc861_capture_mixer[] = {
12109 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
12110 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
12111
12112 {
12113 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
12114 /* The multiple "Capture Source" controls confuse alsamixer
12115 * So call somewhat different..
Kailang Yangdf694da2005-12-05 19:42:22 +010012116 */
12117 /* .name = "Capture Source", */
12118 .name = "Input Source",
12119 .count = 1,
12120 .info = alc_mux_enum_info,
12121 .get = alc_mux_enum_get,
12122 .put = alc_mux_enum_put,
12123 },
12124 { } /* end */
12125};
12126
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012127static void alc861_auto_set_output_and_unmute(struct hda_codec *codec,
12128 hda_nid_t nid,
Kailang Yangdf694da2005-12-05 19:42:22 +010012129 int pin_type, int dac_idx)
12130{
Jacek Luczak564c5be2008-05-03 18:41:23 +020012131 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
12132 pin_type);
12133 snd_hda_codec_write(codec, dac_idx, 0, AC_VERB_SET_AMP_GAIN_MUTE,
12134 AMP_OUT_UNMUTE);
Kailang Yangdf694da2005-12-05 19:42:22 +010012135}
12136
12137static void alc861_auto_init_multi_out(struct hda_codec *codec)
12138{
12139 struct alc_spec *spec = codec->spec;
12140 int i;
12141
Kailang Yangbc9f98a2007-04-12 13:06:07 +020012142 alc_subsystem_id(codec, 0x0e, 0x0f, 0x0b);
Kailang Yangdf694da2005-12-05 19:42:22 +010012143 for (i = 0; i < spec->autocfg.line_outs; i++) {
12144 hda_nid_t nid = spec->autocfg.line_out_pins[i];
Takashi Iwaibaba8ee2007-04-23 17:17:48 +020012145 int pin_type = get_pin_type(spec->autocfg.line_out_type);
Kailang Yangdf694da2005-12-05 19:42:22 +010012146 if (nid)
Takashi Iwaibaba8ee2007-04-23 17:17:48 +020012147 alc861_auto_set_output_and_unmute(codec, nid, pin_type,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012148 spec->multiout.dac_nids[i]);
Kailang Yangdf694da2005-12-05 19:42:22 +010012149 }
12150}
12151
12152static void alc861_auto_init_hp_out(struct hda_codec *codec)
12153{
12154 struct alc_spec *spec = codec->spec;
12155 hda_nid_t pin;
12156
Takashi Iwaieb06ed82006-09-20 17:10:27 +020012157 pin = spec->autocfg.hp_pins[0];
Kailang Yangdf694da2005-12-05 19:42:22 +010012158 if (pin) /* connect to front */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012159 alc861_auto_set_output_and_unmute(codec, pin, PIN_HP,
12160 spec->multiout.dac_nids[0]);
Takashi Iwaif6c7e542008-02-12 18:32:23 +010012161 pin = spec->autocfg.speaker_pins[0];
12162 if (pin)
12163 alc861_auto_set_output_and_unmute(codec, pin, PIN_OUT, 0);
Kailang Yangdf694da2005-12-05 19:42:22 +010012164}
12165
12166static void alc861_auto_init_analog_input(struct hda_codec *codec)
12167{
12168 struct alc_spec *spec = codec->spec;
12169 int i;
12170
12171 for (i = 0; i < AUTO_PIN_LAST; i++) {
12172 hda_nid_t nid = spec->autocfg.input_pins[i];
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012173 if (nid >= 0x0c && nid <= 0x11) {
12174 snd_hda_codec_write(codec, nid, 0,
12175 AC_VERB_SET_PIN_WIDGET_CONTROL,
12176 i <= AUTO_PIN_FRONT_MIC ?
12177 PIN_VREF80 : PIN_IN);
Kailang Yangdf694da2005-12-05 19:42:22 +010012178 }
12179 }
12180}
12181
12182/* parse the BIOS configuration and set up the alc_spec */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012183/* return 1 if successful, 0 if the proper config is not found,
12184 * or a negative error code
12185 */
Kailang Yangdf694da2005-12-05 19:42:22 +010012186static int alc861_parse_auto_config(struct hda_codec *codec)
12187{
12188 struct alc_spec *spec = codec->spec;
12189 int err;
12190 static hda_nid_t alc861_ignore[] = { 0x1d, 0 };
12191
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012192 err = snd_hda_parse_pin_def_config(codec, &spec->autocfg,
12193 alc861_ignore);
12194 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +010012195 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012196 if (!spec->autocfg.line_outs)
Kailang Yangdf694da2005-12-05 19:42:22 +010012197 return 0; /* can't find valid BIOS pin config */
12198
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012199 err = alc861_auto_fill_dac_nids(spec, &spec->autocfg);
12200 if (err < 0)
12201 return err;
12202 err = alc861_auto_create_multi_out_ctls(spec, &spec->autocfg);
12203 if (err < 0)
12204 return err;
12205 err = alc861_auto_create_hp_ctls(spec, spec->autocfg.hp_pins[0]);
12206 if (err < 0)
12207 return err;
12208 err = alc861_auto_create_analog_input_ctls(spec, &spec->autocfg);
12209 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +010012210 return err;
12211
12212 spec->multiout.max_channels = spec->multiout.num_dacs * 2;
12213
12214 if (spec->autocfg.dig_out_pin)
12215 spec->multiout.dig_out_nid = ALC861_DIGOUT_NID;
12216
12217 if (spec->kctl_alloc)
12218 spec->mixers[spec->num_mixers++] = spec->kctl_alloc;
12219
12220 spec->init_verbs[spec->num_init_verbs++] = alc861_auto_init_verbs;
12221
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +020012222 spec->num_mux_defs = 1;
Kailang Yangdf694da2005-12-05 19:42:22 +010012223 spec->input_mux = &spec->private_imux;
12224
12225 spec->adc_nids = alc861_adc_nids;
12226 spec->num_adc_nids = ARRAY_SIZE(alc861_adc_nids);
12227 spec->mixers[spec->num_mixers] = alc861_capture_mixer;
12228 spec->num_mixers++;
12229
12230 return 1;
12231}
12232
Takashi Iwaiae6b8132006-03-03 16:47:17 +010012233/* additional initialization for auto-configuration model */
12234static void alc861_auto_init(struct hda_codec *codec)
Kailang Yangdf694da2005-12-05 19:42:22 +010012235{
Takashi Iwaif6c7e542008-02-12 18:32:23 +010012236 struct alc_spec *spec = codec->spec;
Kailang Yangdf694da2005-12-05 19:42:22 +010012237 alc861_auto_init_multi_out(codec);
12238 alc861_auto_init_hp_out(codec);
12239 alc861_auto_init_analog_input(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +010012240 if (spec->unsol_event)
12241 alc_sku_automute(codec);
Kailang Yangdf694da2005-12-05 19:42:22 +010012242}
12243
Takashi Iwaicb53c622007-08-10 17:21:45 +020012244#ifdef CONFIG_SND_HDA_POWER_SAVE
12245static struct hda_amp_list alc861_loopbacks[] = {
12246 { 0x15, HDA_INPUT, 0 },
12247 { 0x15, HDA_INPUT, 1 },
12248 { 0x15, HDA_INPUT, 2 },
12249 { 0x15, HDA_INPUT, 3 },
12250 { } /* end */
12251};
12252#endif
12253
Kailang Yangdf694da2005-12-05 19:42:22 +010012254
12255/*
12256 * configuration and preset
12257 */
Takashi Iwaif5fcc132006-11-24 17:07:44 +010012258static const char *alc861_models[ALC861_MODEL_LAST] = {
12259 [ALC861_3ST] = "3stack",
12260 [ALC660_3ST] = "3stack-660",
12261 [ALC861_3ST_DIG] = "3stack-dig",
12262 [ALC861_6ST_DIG] = "6stack-dig",
12263 [ALC861_UNIWILL_M31] = "uniwill-m31",
12264 [ALC861_TOSHIBA] = "toshiba",
12265 [ALC861_ASUS] = "asus",
12266 [ALC861_ASUS_LAPTOP] = "asus-laptop",
12267 [ALC861_AUTO] = "auto",
12268};
12269
12270static struct snd_pci_quirk alc861_cfg_tbl[] = {
Takashi Iwai687a47b2007-01-10 11:25:58 +010012271 SND_PCI_QUIRK(0x1043, 0x1205, "ASUS W7J", ALC861_3ST),
Takashi Iwaif5fcc132006-11-24 17:07:44 +010012272 SND_PCI_QUIRK(0x1043, 0x1335, "ASUS F2/3", ALC861_ASUS_LAPTOP),
12273 SND_PCI_QUIRK(0x1043, 0x1338, "ASUS F2/3", ALC861_ASUS_LAPTOP),
12274 SND_PCI_QUIRK(0x1043, 0x1393, "ASUS", ALC861_ASUS),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010012275 SND_PCI_QUIRK(0x1043, 0x13d7, "ASUS A9rp", ALC861_ASUS_LAPTOP),
Kailang Yang83c34212007-07-05 11:43:05 +020012276 SND_PCI_QUIRK(0x1043, 0x81cb, "ASUS P1-AH2", ALC861_3ST_DIG),
Tobin Davisad5e7732007-01-08 10:57:32 +010012277 SND_PCI_QUIRK(0x1179, 0xff00, "Toshiba", ALC861_TOSHIBA),
Takashi Iwai341d4eb2007-07-09 17:53:18 +020012278 /* FIXME: the entry below breaks Toshiba A100 (model=auto works!)
12279 * Any other models that need this preset?
12280 */
12281 /* SND_PCI_QUIRK(0x1179, 0xff10, "Toshiba", ALC861_TOSHIBA), */
Claudio Matsuokaef64adb2007-07-14 00:26:16 +020012282 SND_PCI_QUIRK(0x1462, 0x7254, "HP dx2200 (MSI MS-7254)", ALC861_3ST),
12283 SND_PCI_QUIRK(0x1462, 0x7297, "HP dx2250 (MSI MS-7297)", ALC861_3ST),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010012284 SND_PCI_QUIRK(0x1584, 0x2b01, "Uniwill X40AIx", ALC861_UNIWILL_M31),
12285 SND_PCI_QUIRK(0x1584, 0x9072, "Uniwill m31", ALC861_UNIWILL_M31),
12286 SND_PCI_QUIRK(0x1584, 0x9075, "Airis Praxis N1212", ALC861_ASUS_LAPTOP),
12287 /* FIXME: the below seems conflict */
12288 /* SND_PCI_QUIRK(0x1584, 0x9075, "Uniwill", ALC861_UNIWILL_M31), */
12289 SND_PCI_QUIRK(0x1849, 0x0660, "Asrock 939SLI32", ALC660_3ST),
12290 SND_PCI_QUIRK(0x8086, 0xd600, "Intel", ALC861_3ST),
Kailang Yangdf694da2005-12-05 19:42:22 +010012291 {}
12292};
12293
12294static struct alc_config_preset alc861_presets[] = {
12295 [ALC861_3ST] = {
12296 .mixers = { alc861_3ST_mixer },
12297 .init_verbs = { alc861_threestack_init_verbs },
12298 .num_dacs = ARRAY_SIZE(alc861_dac_nids),
12299 .dac_nids = alc861_dac_nids,
12300 .num_channel_mode = ARRAY_SIZE(alc861_threestack_modes),
12301 .channel_mode = alc861_threestack_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +020012302 .need_dac_fix = 1,
Kailang Yangdf694da2005-12-05 19:42:22 +010012303 .num_adc_nids = ARRAY_SIZE(alc861_adc_nids),
12304 .adc_nids = alc861_adc_nids,
12305 .input_mux = &alc861_capture_source,
12306 },
12307 [ALC861_3ST_DIG] = {
12308 .mixers = { alc861_base_mixer },
12309 .init_verbs = { alc861_threestack_init_verbs },
12310 .num_dacs = ARRAY_SIZE(alc861_dac_nids),
12311 .dac_nids = alc861_dac_nids,
12312 .dig_out_nid = ALC861_DIGOUT_NID,
12313 .num_channel_mode = ARRAY_SIZE(alc861_threestack_modes),
12314 .channel_mode = alc861_threestack_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +020012315 .need_dac_fix = 1,
Kailang Yangdf694da2005-12-05 19:42:22 +010012316 .num_adc_nids = ARRAY_SIZE(alc861_adc_nids),
12317 .adc_nids = alc861_adc_nids,
12318 .input_mux = &alc861_capture_source,
12319 },
12320 [ALC861_6ST_DIG] = {
12321 .mixers = { alc861_base_mixer },
12322 .init_verbs = { alc861_base_init_verbs },
12323 .num_dacs = ARRAY_SIZE(alc861_dac_nids),
12324 .dac_nids = alc861_dac_nids,
12325 .dig_out_nid = ALC861_DIGOUT_NID,
12326 .num_channel_mode = ARRAY_SIZE(alc861_8ch_modes),
12327 .channel_mode = alc861_8ch_modes,
12328 .num_adc_nids = ARRAY_SIZE(alc861_adc_nids),
12329 .adc_nids = alc861_adc_nids,
12330 .input_mux = &alc861_capture_source,
12331 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +020012332 [ALC660_3ST] = {
12333 .mixers = { alc861_3ST_mixer },
12334 .init_verbs = { alc861_threestack_init_verbs },
12335 .num_dacs = ARRAY_SIZE(alc660_dac_nids),
12336 .dac_nids = alc660_dac_nids,
12337 .num_channel_mode = ARRAY_SIZE(alc861_threestack_modes),
12338 .channel_mode = alc861_threestack_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +020012339 .need_dac_fix = 1,
Takashi Iwai9c7f8522006-06-28 15:08:22 +020012340 .num_adc_nids = ARRAY_SIZE(alc861_adc_nids),
12341 .adc_nids = alc861_adc_nids,
12342 .input_mux = &alc861_capture_source,
12343 },
Takashi Iwai22309c32006-08-09 16:57:28 +020012344 [ALC861_UNIWILL_M31] = {
12345 .mixers = { alc861_uniwill_m31_mixer },
12346 .init_verbs = { alc861_uniwill_m31_init_verbs },
12347 .num_dacs = ARRAY_SIZE(alc861_dac_nids),
12348 .dac_nids = alc861_dac_nids,
12349 .dig_out_nid = ALC861_DIGOUT_NID,
12350 .num_channel_mode = ARRAY_SIZE(alc861_uniwill_m31_modes),
12351 .channel_mode = alc861_uniwill_m31_modes,
12352 .need_dac_fix = 1,
12353 .num_adc_nids = ARRAY_SIZE(alc861_adc_nids),
12354 .adc_nids = alc861_adc_nids,
12355 .input_mux = &alc861_capture_source,
12356 },
Tobin Davisa53d1ae2006-10-17 12:00:28 +020012357 [ALC861_TOSHIBA] = {
12358 .mixers = { alc861_toshiba_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012359 .init_verbs = { alc861_base_init_verbs,
12360 alc861_toshiba_init_verbs },
Tobin Davisa53d1ae2006-10-17 12:00:28 +020012361 .num_dacs = ARRAY_SIZE(alc861_dac_nids),
12362 .dac_nids = alc861_dac_nids,
12363 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
12364 .channel_mode = alc883_3ST_2ch_modes,
12365 .num_adc_nids = ARRAY_SIZE(alc861_adc_nids),
12366 .adc_nids = alc861_adc_nids,
12367 .input_mux = &alc861_capture_source,
12368 .unsol_event = alc861_toshiba_unsol_event,
12369 .init_hook = alc861_toshiba_automute,
12370 },
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020012371 [ALC861_ASUS] = {
12372 .mixers = { alc861_asus_mixer },
12373 .init_verbs = { alc861_asus_init_verbs },
12374 .num_dacs = ARRAY_SIZE(alc861_dac_nids),
12375 .dac_nids = alc861_dac_nids,
12376 .dig_out_nid = ALC861_DIGOUT_NID,
12377 .num_channel_mode = ARRAY_SIZE(alc861_asus_modes),
12378 .channel_mode = alc861_asus_modes,
12379 .need_dac_fix = 1,
12380 .hp_nid = 0x06,
12381 .num_adc_nids = ARRAY_SIZE(alc861_adc_nids),
12382 .adc_nids = alc861_adc_nids,
12383 .input_mux = &alc861_capture_source,
12384 },
Takashi Iwai56bb0ca2006-11-22 11:52:52 +010012385 [ALC861_ASUS_LAPTOP] = {
12386 .mixers = { alc861_toshiba_mixer, alc861_asus_laptop_mixer },
12387 .init_verbs = { alc861_asus_init_verbs,
12388 alc861_asus_laptop_init_verbs },
12389 .num_dacs = ARRAY_SIZE(alc861_dac_nids),
12390 .dac_nids = alc861_dac_nids,
12391 .dig_out_nid = ALC861_DIGOUT_NID,
12392 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
12393 .channel_mode = alc883_3ST_2ch_modes,
12394 .need_dac_fix = 1,
12395 .num_adc_nids = ARRAY_SIZE(alc861_adc_nids),
12396 .adc_nids = alc861_adc_nids,
12397 .input_mux = &alc861_capture_source,
12398 },
12399};
Kailang Yangdf694da2005-12-05 19:42:22 +010012400
12401
12402static int patch_alc861(struct hda_codec *codec)
12403{
12404 struct alc_spec *spec;
12405 int board_config;
12406 int err;
12407
Robert P. J. Daydc041e02006-12-19 14:44:15 +010012408 spec = kzalloc(sizeof(*spec), GFP_KERNEL);
Kailang Yangdf694da2005-12-05 19:42:22 +010012409 if (spec == NULL)
12410 return -ENOMEM;
12411
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012412 codec->spec = spec;
Kailang Yangdf694da2005-12-05 19:42:22 +010012413
Takashi Iwaif5fcc132006-11-24 17:07:44 +010012414 board_config = snd_hda_check_board_config(codec, ALC861_MODEL_LAST,
12415 alc861_models,
12416 alc861_cfg_tbl);
Takashi Iwai9c7f8522006-06-28 15:08:22 +020012417
Takashi Iwaif5fcc132006-11-24 17:07:44 +010012418 if (board_config < 0) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +020012419 printk(KERN_INFO "hda_codec: Unknown model for ALC861, "
12420 "trying auto-probe from BIOS...\n");
Kailang Yangdf694da2005-12-05 19:42:22 +010012421 board_config = ALC861_AUTO;
12422 }
12423
12424 if (board_config == ALC861_AUTO) {
12425 /* automatic parse from the BIOS config */
12426 err = alc861_parse_auto_config(codec);
12427 if (err < 0) {
12428 alc_free(codec);
12429 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020012430 } else if (!err) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +020012431 printk(KERN_INFO
12432 "hda_codec: Cannot set up configuration "
12433 "from BIOS. Using base mode...\n");
Kailang Yangdf694da2005-12-05 19:42:22 +010012434 board_config = ALC861_3ST_DIG;
12435 }
12436 }
12437
12438 if (board_config != ALC861_AUTO)
12439 setup_preset(spec, &alc861_presets[board_config]);
12440
12441 spec->stream_name_analog = "ALC861 Analog";
12442 spec->stream_analog_playback = &alc861_pcm_analog_playback;
12443 spec->stream_analog_capture = &alc861_pcm_analog_capture;
12444
12445 spec->stream_name_digital = "ALC861 Digital";
12446 spec->stream_digital_playback = &alc861_pcm_digital_playback;
12447 spec->stream_digital_capture = &alc861_pcm_digital_capture;
12448
Takashi Iwai2134ea42008-01-10 16:53:55 +010012449 spec->vmaster_nid = 0x03;
12450
Kailang Yangdf694da2005-12-05 19:42:22 +010012451 codec->patch_ops = alc_patch_ops;
12452 if (board_config == ALC861_AUTO)
Takashi Iwaiae6b8132006-03-03 16:47:17 +010012453 spec->init_hook = alc861_auto_init;
Takashi Iwaicb53c622007-08-10 17:21:45 +020012454#ifdef CONFIG_SND_HDA_POWER_SAVE
12455 if (!spec->loopback.amplist)
12456 spec->loopback.amplist = alc861_loopbacks;
12457#endif
Kailang Yangdf694da2005-12-05 19:42:22 +010012458
12459 return 0;
12460}
12461
12462/*
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010012463 * ALC861-VD support
12464 *
12465 * Based on ALC882
12466 *
12467 * In addition, an independent DAC
12468 */
12469#define ALC861VD_DIGOUT_NID 0x06
12470
12471static hda_nid_t alc861vd_dac_nids[4] = {
12472 /* front, surr, clfe, side surr */
12473 0x02, 0x03, 0x04, 0x05
12474};
12475
12476/* dac_nids for ALC660vd are in a different order - according to
12477 * Realtek's driver.
12478 * This should probably tesult in a different mixer for 6stack models
12479 * of ALC660vd codecs, but for now there is only 3stack mixer
12480 * - and it is the same as in 861vd.
12481 * adc_nids in ALC660vd are (is) the same as in 861vd
12482 */
12483static hda_nid_t alc660vd_dac_nids[3] = {
12484 /* front, rear, clfe, rear_surr */
12485 0x02, 0x04, 0x03
12486};
12487
12488static hda_nid_t alc861vd_adc_nids[1] = {
12489 /* ADC0 */
12490 0x09,
12491};
12492
Takashi Iwaie1406342008-02-11 18:32:32 +010012493static hda_nid_t alc861vd_capsrc_nids[1] = { 0x22 };
12494
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010012495/* input MUX */
12496/* FIXME: should be a matrix-type input source selection */
12497static struct hda_input_mux alc861vd_capture_source = {
12498 .num_items = 4,
12499 .items = {
12500 { "Mic", 0x0 },
12501 { "Front Mic", 0x1 },
12502 { "Line", 0x2 },
12503 { "CD", 0x4 },
12504 },
12505};
12506
Kailang Yang272a5272007-05-14 11:00:38 +020012507static struct hda_input_mux alc861vd_dallas_capture_source = {
Tobin Davisb419f342008-03-07 11:57:51 +010012508 .num_items = 2,
Kailang Yang272a5272007-05-14 11:00:38 +020012509 .items = {
Tobin Davisb419f342008-03-07 11:57:51 +010012510 { "Ext Mic", 0x0 },
12511 { "Int Mic", 0x1 },
Kailang Yang272a5272007-05-14 11:00:38 +020012512 },
12513};
12514
Kailang Yangd1a991a2007-08-15 16:21:59 +020012515static struct hda_input_mux alc861vd_hp_capture_source = {
12516 .num_items = 2,
12517 .items = {
12518 { "Front Mic", 0x0 },
12519 { "ATAPI Mic", 0x1 },
12520 },
12521};
12522
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010012523#define alc861vd_mux_enum_info alc_mux_enum_info
12524#define alc861vd_mux_enum_get alc_mux_enum_get
Takashi Iwaie1406342008-02-11 18:32:32 +010012525/* ALC861VD has the ALC882-type input selection (but has only one ADC) */
12526#define alc861vd_mux_enum_put alc882_mux_enum_put
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010012527
12528/*
12529 * 2ch mode
12530 */
12531static struct hda_channel_mode alc861vd_3stack_2ch_modes[1] = {
12532 { 2, NULL }
12533};
12534
12535/*
12536 * 6ch mode
12537 */
12538static struct hda_verb alc861vd_6stack_ch6_init[] = {
12539 { 0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
12540 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
12541 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
12542 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
12543 { } /* end */
12544};
12545
12546/*
12547 * 8ch mode
12548 */
12549static struct hda_verb alc861vd_6stack_ch8_init[] = {
12550 { 0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
12551 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
12552 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
12553 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
12554 { } /* end */
12555};
12556
12557static struct hda_channel_mode alc861vd_6stack_modes[2] = {
12558 { 6, alc861vd_6stack_ch6_init },
12559 { 8, alc861vd_6stack_ch8_init },
12560};
12561
12562static struct snd_kcontrol_new alc861vd_chmode_mixer[] = {
12563 {
12564 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
12565 .name = "Channel Mode",
12566 .info = alc_ch_mode_info,
12567 .get = alc_ch_mode_get,
12568 .put = alc_ch_mode_put,
12569 },
12570 { } /* end */
12571};
12572
12573static struct snd_kcontrol_new alc861vd_capture_mixer[] = {
12574 HDA_CODEC_VOLUME("Capture Volume", 0x09, 0x0, HDA_INPUT),
12575 HDA_CODEC_MUTE("Capture Switch", 0x09, 0x0, HDA_INPUT),
12576
12577 {
12578 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
12579 /* The multiple "Capture Source" controls confuse alsamixer
12580 * So call somewhat different..
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010012581 */
12582 /* .name = "Capture Source", */
12583 .name = "Input Source",
12584 .count = 1,
12585 .info = alc861vd_mux_enum_info,
12586 .get = alc861vd_mux_enum_get,
12587 .put = alc861vd_mux_enum_put,
12588 },
12589 { } /* end */
12590};
12591
12592/* Pin assignment: Front=0x14, Rear=0x15, CLFE=0x16, Side=0x17
12593 * Mic=0x18, Front Mic=0x19, Line-In=0x1a, HP=0x1b
12594 */
12595static struct snd_kcontrol_new alc861vd_6st_mixer[] = {
12596 HDA_CODEC_VOLUME("Front Playback Volume", 0x02, 0x0, HDA_OUTPUT),
12597 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
12598
12599 HDA_CODEC_VOLUME("Surround Playback Volume", 0x03, 0x0, HDA_OUTPUT),
12600 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
12601
12602 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x04, 1, 0x0,
12603 HDA_OUTPUT),
12604 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x04, 2, 0x0,
12605 HDA_OUTPUT),
12606 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
12607 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
12608
12609 HDA_CODEC_VOLUME("Side Playback Volume", 0x05, 0x0, HDA_OUTPUT),
12610 HDA_BIND_MUTE("Side Playback Switch", 0x0f, 2, HDA_INPUT),
12611
12612 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
12613
12614 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
12615 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
12616 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
12617
12618 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
12619 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
12620 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
12621
12622 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
12623 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
12624
12625 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
12626 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
12627
12628 HDA_CODEC_VOLUME("PC Speaker Playback Volume", 0x0b, 0x05, HDA_INPUT),
12629 HDA_CODEC_MUTE("PC Speaker Playback Switch", 0x0b, 0x05, HDA_INPUT),
12630
12631 { } /* end */
12632};
12633
12634static struct snd_kcontrol_new alc861vd_3st_mixer[] = {
12635 HDA_CODEC_VOLUME("Front Playback Volume", 0x02, 0x0, HDA_OUTPUT),
12636 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
12637
12638 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
12639
12640 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
12641 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
12642 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
12643
12644 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
12645 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
12646 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
12647
12648 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
12649 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
12650
12651 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
12652 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
12653
12654 HDA_CODEC_VOLUME("PC Speaker Playback Volume", 0x0b, 0x05, HDA_INPUT),
12655 HDA_CODEC_MUTE("PC Speaker Playback Switch", 0x0b, 0x05, HDA_INPUT),
12656
12657 { } /* end */
12658};
12659
Kailang Yangbdd148a2007-05-08 15:19:08 +020012660static struct snd_kcontrol_new alc861vd_lenovo_mixer[] = {
12661 HDA_CODEC_VOLUME("Front Playback Volume", 0x02, 0x0, HDA_OUTPUT),
12662 /*HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),*/
12663 HDA_CODEC_MUTE("Front Playback Switch", 0x14, 0x0, HDA_OUTPUT),
12664
12665 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
12666
12667 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
12668 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
12669 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
12670
12671 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
12672 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
12673 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
12674
12675 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
12676 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
12677
12678 { } /* end */
12679};
12680
Tobin Davisb419f342008-03-07 11:57:51 +010012681/* Pin assignment: Speaker=0x14, HP = 0x15,
12682 * Ext Mic=0x18, Int Mic = 0x19, CD = 0x1c, PC Beep = 0x1d
Kailang Yang272a5272007-05-14 11:00:38 +020012683 */
12684static struct snd_kcontrol_new alc861vd_dallas_mixer[] = {
Tobin Davisb419f342008-03-07 11:57:51 +010012685 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x02, 0x0, HDA_OUTPUT),
12686 HDA_BIND_MUTE("Speaker Playback Switch", 0x0c, 2, HDA_INPUT),
Kailang Yang272a5272007-05-14 11:00:38 +020012687 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x03, 0x0, HDA_OUTPUT),
12688 HDA_BIND_MUTE("Headphone Playback Switch", 0x0d, 2, HDA_INPUT),
Tobin Davisb419f342008-03-07 11:57:51 +010012689 HDA_CODEC_VOLUME("Ext Mic Boost", 0x18, 0, HDA_INPUT),
12690 HDA_CODEC_VOLUME("Ext Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
12691 HDA_CODEC_MUTE("Ext Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
12692 HDA_CODEC_VOLUME("Int Mic Boost", 0x19, 0, HDA_INPUT),
12693 HDA_CODEC_VOLUME("Int Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
12694 HDA_CODEC_MUTE("Int Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
12695 HDA_CODEC_VOLUME("PC Beep Volume", 0x0b, 0x05, HDA_INPUT),
12696 HDA_CODEC_MUTE("PC Beep Switch", 0x0b, 0x05, HDA_INPUT),
Kailang Yang272a5272007-05-14 11:00:38 +020012697 { } /* end */
12698};
12699
Kailang Yangd1a991a2007-08-15 16:21:59 +020012700/* Pin assignment: Speaker=0x14, Line-out = 0x15,
12701 * Front Mic=0x18, ATAPI Mic = 0x19,
12702 */
12703static struct snd_kcontrol_new alc861vd_hp_mixer[] = {
12704 HDA_CODEC_VOLUME("Front Playback Volume", 0x02, 0x0, HDA_OUTPUT),
12705 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
12706 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x03, 0x0, HDA_OUTPUT),
12707 HDA_BIND_MUTE("Headphone Playback Switch", 0x0d, 2, HDA_INPUT),
12708 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
12709 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
12710 HDA_CODEC_VOLUME("ATAPI Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
12711 HDA_CODEC_MUTE("ATAPI Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
12712
12713 { } /* end */
12714};
12715
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010012716/*
12717 * generic initialization of ADC, input mixers and output mixers
12718 */
12719static struct hda_verb alc861vd_volume_init_verbs[] = {
12720 /*
12721 * Unmute ADC0 and set the default input to mic-in
12722 */
12723 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
12724 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
12725
12726 /* Unmute input amps (CD, Line In, Mic 1 & Mic 2) of
12727 * the analog-loopback mixer widget
12728 */
12729 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +020012730 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
12731 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
12732 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
12733 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
12734 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010012735
12736 /* Capture mixer: unmute Mic, F-Mic, Line, CD inputs */
Kailang Yangbdd148a2007-05-08 15:19:08 +020012737 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
12738 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
12739 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010012740 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(4)},
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010012741
12742 /*
12743 * Set up output mixers (0x02 - 0x05)
12744 */
12745 /* set vol=0 to output mixers */
12746 {0x02, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
12747 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
12748 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
12749 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
12750
12751 /* set up input amps for analog loopback */
12752 /* Amp Indices: DAC = 0, mixer = 1 */
12753 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
12754 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
12755 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
12756 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
12757 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
12758 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
12759 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
12760 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
12761
12762 { }
12763};
12764
12765/*
12766 * 3-stack pin configuration:
12767 * front = 0x14, mic/clfe = 0x18, HP = 0x19, line/surr = 0x1a, f-mic = 0x1b
12768 */
12769static struct hda_verb alc861vd_3stack_init_verbs[] = {
12770 /*
12771 * Set pin mode and muting
12772 */
12773 /* set front pin widgets 0x14 for output */
12774 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
12775 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
12776 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
12777
12778 /* Mic (rear) pin: input vref at 80% */
12779 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
12780 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
12781 /* Front Mic pin: input vref at 80% */
12782 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
12783 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
12784 /* Line In pin: input */
12785 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
12786 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
12787 /* Line-2 In: Headphone output (output 0 - 0x0c) */
12788 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
12789 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
12790 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
12791 /* CD pin widget for input */
12792 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
12793
12794 { }
12795};
12796
12797/*
12798 * 6-stack pin configuration:
12799 */
12800static struct hda_verb alc861vd_6stack_init_verbs[] = {
12801 /*
12802 * Set pin mode and muting
12803 */
12804 /* set front pin widgets 0x14 for output */
12805 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
12806 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
12807 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
12808
12809 /* Rear Pin: output 1 (0x0d) */
12810 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
12811 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
12812 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01},
12813 /* CLFE Pin: output 2 (0x0e) */
12814 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
12815 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
12816 {0x16, AC_VERB_SET_CONNECT_SEL, 0x02},
12817 /* Side Pin: output 3 (0x0f) */
12818 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
12819 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
12820 {0x17, AC_VERB_SET_CONNECT_SEL, 0x03},
12821
12822 /* Mic (rear) pin: input vref at 80% */
12823 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
12824 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
12825 /* Front Mic pin: input vref at 80% */
12826 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
12827 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
12828 /* Line In pin: input */
12829 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
12830 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
12831 /* Line-2 In: Headphone output (output 0 - 0x0c) */
12832 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
12833 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
12834 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
12835 /* CD pin widget for input */
12836 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
12837
12838 { }
12839};
12840
Kailang Yangbdd148a2007-05-08 15:19:08 +020012841static struct hda_verb alc861vd_eapd_verbs[] = {
12842 {0x14, AC_VERB_SET_EAPD_BTLENABLE, 2},
12843 { }
12844};
12845
Kailang Yangf9423e72008-05-27 12:32:25 +020012846static struct hda_verb alc660vd_eapd_verbs[] = {
12847 {0x14, AC_VERB_SET_EAPD_BTLENABLE, 2},
12848 {0x15, AC_VERB_SET_EAPD_BTLENABLE, 2},
12849 { }
12850};
12851
Kailang Yangbdd148a2007-05-08 15:19:08 +020012852static struct hda_verb alc861vd_lenovo_unsol_verbs[] = {
12853 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
12854 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
12855 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(5)},
12856 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
12857 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
12858 {}
12859};
12860
12861/* toggle speaker-output according to the hp-jack state */
12862static void alc861vd_lenovo_hp_automute(struct hda_codec *codec)
12863{
12864 unsigned int present;
12865 unsigned char bits;
12866
12867 present = snd_hda_codec_read(codec, 0x1b, 0,
12868 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +020012869 bits = present ? HDA_AMP_MUTE : 0;
12870 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
12871 HDA_AMP_MUTE, bits);
Kailang Yangbdd148a2007-05-08 15:19:08 +020012872}
12873
12874static void alc861vd_lenovo_mic_automute(struct hda_codec *codec)
12875{
12876 unsigned int present;
12877 unsigned char bits;
12878
12879 present = snd_hda_codec_read(codec, 0x18, 0,
12880 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +020012881 bits = present ? HDA_AMP_MUTE : 0;
12882 snd_hda_codec_amp_stereo(codec, 0x0b, HDA_INPUT, 1,
12883 HDA_AMP_MUTE, bits);
Kailang Yangbdd148a2007-05-08 15:19:08 +020012884}
12885
12886static void alc861vd_lenovo_automute(struct hda_codec *codec)
12887{
12888 alc861vd_lenovo_hp_automute(codec);
12889 alc861vd_lenovo_mic_automute(codec);
12890}
12891
12892static void alc861vd_lenovo_unsol_event(struct hda_codec *codec,
12893 unsigned int res)
12894{
12895 switch (res >> 26) {
12896 case ALC880_HP_EVENT:
12897 alc861vd_lenovo_hp_automute(codec);
12898 break;
12899 case ALC880_MIC_EVENT:
12900 alc861vd_lenovo_mic_automute(codec);
12901 break;
12902 }
12903}
12904
Kailang Yang272a5272007-05-14 11:00:38 +020012905static struct hda_verb alc861vd_dallas_verbs[] = {
12906 {0x02, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
12907 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
12908 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
12909 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
12910
12911 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
12912 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
12913 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
12914 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
12915 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
12916 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
12917 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
12918 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
12919
12920 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
12921 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
12922 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
12923 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
12924 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
12925 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
12926 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
12927 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
12928
12929 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF50},
12930 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
12931 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF50},
12932 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
12933 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
12934 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
12935 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
12936 {0x1d, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
12937
12938 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
12939 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
12940 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
12941 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
12942
12943 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
12944 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
12945 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
12946
12947 { } /* end */
12948};
12949
12950/* toggle speaker-output according to the hp-jack state */
12951static void alc861vd_dallas_automute(struct hda_codec *codec)
12952{
12953 unsigned int present;
12954
12955 present = snd_hda_codec_read(codec, 0x15, 0,
12956 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +020012957 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
12958 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
Kailang Yang272a5272007-05-14 11:00:38 +020012959}
12960
12961static void alc861vd_dallas_unsol_event(struct hda_codec *codec, unsigned int res)
12962{
12963 if ((res >> 26) == ALC880_HP_EVENT)
12964 alc861vd_dallas_automute(codec);
12965}
12966
Takashi Iwaicb53c622007-08-10 17:21:45 +020012967#ifdef CONFIG_SND_HDA_POWER_SAVE
12968#define alc861vd_loopbacks alc880_loopbacks
12969#endif
12970
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010012971/* pcm configuration: identiacal with ALC880 */
12972#define alc861vd_pcm_analog_playback alc880_pcm_analog_playback
12973#define alc861vd_pcm_analog_capture alc880_pcm_analog_capture
12974#define alc861vd_pcm_digital_playback alc880_pcm_digital_playback
12975#define alc861vd_pcm_digital_capture alc880_pcm_digital_capture
12976
12977/*
12978 * configuration and preset
12979 */
12980static const char *alc861vd_models[ALC861VD_MODEL_LAST] = {
12981 [ALC660VD_3ST] = "3stack-660",
Takashi Iwai983f8ae2007-08-15 16:44:04 +020012982 [ALC660VD_3ST_DIG] = "3stack-660-digout",
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010012983 [ALC861VD_3ST] = "3stack",
12984 [ALC861VD_3ST_DIG] = "3stack-digout",
12985 [ALC861VD_6ST_DIG] = "6stack-digout",
Kailang Yangbdd148a2007-05-08 15:19:08 +020012986 [ALC861VD_LENOVO] = "lenovo",
Kailang Yang272a5272007-05-14 11:00:38 +020012987 [ALC861VD_DALLAS] = "dallas",
Takashi Iwai983f8ae2007-08-15 16:44:04 +020012988 [ALC861VD_HP] = "hp",
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010012989 [ALC861VD_AUTO] = "auto",
12990};
12991
12992static struct snd_pci_quirk alc861vd_cfg_tbl[] = {
Takashi Iwaiac3e3742007-12-17 17:14:18 +010012993 SND_PCI_QUIRK(0x1019, 0xa88d, "Realtek ALC660 demo", ALC660VD_3ST),
12994 SND_PCI_QUIRK(0x103c, 0x30bf, "HP TX1000", ALC861VD_HP),
Takashi Iwai07e038b2007-02-15 18:23:41 +010012995 SND_PCI_QUIRK(0x1043, 0x12e2, "Asus z35m", ALC660VD_3ST),
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010012996 SND_PCI_QUIRK(0x1043, 0x1339, "Asus G1", ALC660VD_3ST),
Mike Crash6963f842007-06-25 12:12:51 +020012997 SND_PCI_QUIRK(0x1043, 0x81e7, "ASUS", ALC660VD_3ST_DIG),
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010012998 SND_PCI_QUIRK(0x10de, 0x03f0, "Realtek ALC660 demo", ALC660VD_3ST),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010012999 SND_PCI_QUIRK(0x1179, 0xff00, "Toshiba A135", ALC861VD_LENOVO),
Takashi Iwai38baf5a2007-08-16 17:52:43 +020013000 /*SND_PCI_QUIRK(0x1179, 0xff00, "DALLAS", ALC861VD_DALLAS),*/ /*lenovo*/
Kailang Yang272a5272007-05-14 11:00:38 +020013001 SND_PCI_QUIRK(0x1179, 0xff01, "DALLAS", ALC861VD_DALLAS),
Takashi Iwai542d7c62007-08-16 18:57:30 +020013002 SND_PCI_QUIRK(0x1179, 0xff03, "Toshiba P205", ALC861VD_LENOVO),
Tobin Davisb419f342008-03-07 11:57:51 +010013003 SND_PCI_QUIRK(0x1179, 0xff31, "Toshiba L30-149", ALC861VD_DALLAS),
Takashi Iwai39c5d412007-08-15 16:24:17 +020013004 SND_PCI_QUIRK(0x1565, 0x820d, "Biostar NF61S SE", ALC861VD_6ST_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010013005 SND_PCI_QUIRK(0x17aa, 0x2066, "Lenovo", ALC861VD_LENOVO),
13006 SND_PCI_QUIRK(0x17aa, 0x3802, "Lenovo 3000 C200", ALC861VD_LENOVO),
Takashi Iwaibe321a82008-07-07 16:04:04 +020013007 SND_PCI_QUIRK(0x17aa, 0x384e, "Lenovo 3000 N200", ALC861VD_LENOVO),
Tobin Davis625dc0b2007-07-30 21:42:10 +020013008 SND_PCI_QUIRK(0x1849, 0x0862, "ASRock K8NF6G-VSTA", ALC861VD_6ST_DIG),
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013009 {}
13010};
13011
13012static struct alc_config_preset alc861vd_presets[] = {
13013 [ALC660VD_3ST] = {
13014 .mixers = { alc861vd_3st_mixer },
13015 .init_verbs = { alc861vd_volume_init_verbs,
13016 alc861vd_3stack_init_verbs },
13017 .num_dacs = ARRAY_SIZE(alc660vd_dac_nids),
13018 .dac_nids = alc660vd_dac_nids,
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013019 .num_channel_mode = ARRAY_SIZE(alc861vd_3stack_2ch_modes),
13020 .channel_mode = alc861vd_3stack_2ch_modes,
13021 .input_mux = &alc861vd_capture_source,
13022 },
Mike Crash6963f842007-06-25 12:12:51 +020013023 [ALC660VD_3ST_DIG] = {
13024 .mixers = { alc861vd_3st_mixer },
13025 .init_verbs = { alc861vd_volume_init_verbs,
13026 alc861vd_3stack_init_verbs },
13027 .num_dacs = ARRAY_SIZE(alc660vd_dac_nids),
13028 .dac_nids = alc660vd_dac_nids,
13029 .dig_out_nid = ALC861VD_DIGOUT_NID,
Mike Crash6963f842007-06-25 12:12:51 +020013030 .num_channel_mode = ARRAY_SIZE(alc861vd_3stack_2ch_modes),
13031 .channel_mode = alc861vd_3stack_2ch_modes,
13032 .input_mux = &alc861vd_capture_source,
13033 },
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013034 [ALC861VD_3ST] = {
13035 .mixers = { alc861vd_3st_mixer },
13036 .init_verbs = { alc861vd_volume_init_verbs,
13037 alc861vd_3stack_init_verbs },
13038 .num_dacs = ARRAY_SIZE(alc861vd_dac_nids),
13039 .dac_nids = alc861vd_dac_nids,
13040 .num_channel_mode = ARRAY_SIZE(alc861vd_3stack_2ch_modes),
13041 .channel_mode = alc861vd_3stack_2ch_modes,
13042 .input_mux = &alc861vd_capture_source,
13043 },
13044 [ALC861VD_3ST_DIG] = {
13045 .mixers = { alc861vd_3st_mixer },
13046 .init_verbs = { alc861vd_volume_init_verbs,
13047 alc861vd_3stack_init_verbs },
13048 .num_dacs = ARRAY_SIZE(alc861vd_dac_nids),
13049 .dac_nids = alc861vd_dac_nids,
13050 .dig_out_nid = ALC861VD_DIGOUT_NID,
13051 .num_channel_mode = ARRAY_SIZE(alc861vd_3stack_2ch_modes),
13052 .channel_mode = alc861vd_3stack_2ch_modes,
13053 .input_mux = &alc861vd_capture_source,
13054 },
13055 [ALC861VD_6ST_DIG] = {
13056 .mixers = { alc861vd_6st_mixer, alc861vd_chmode_mixer },
13057 .init_verbs = { alc861vd_volume_init_verbs,
13058 alc861vd_6stack_init_verbs },
13059 .num_dacs = ARRAY_SIZE(alc861vd_dac_nids),
13060 .dac_nids = alc861vd_dac_nids,
13061 .dig_out_nid = ALC861VD_DIGOUT_NID,
13062 .num_channel_mode = ARRAY_SIZE(alc861vd_6stack_modes),
13063 .channel_mode = alc861vd_6stack_modes,
13064 .input_mux = &alc861vd_capture_source,
13065 },
Kailang Yangbdd148a2007-05-08 15:19:08 +020013066 [ALC861VD_LENOVO] = {
13067 .mixers = { alc861vd_lenovo_mixer },
13068 .init_verbs = { alc861vd_volume_init_verbs,
13069 alc861vd_3stack_init_verbs,
13070 alc861vd_eapd_verbs,
13071 alc861vd_lenovo_unsol_verbs },
13072 .num_dacs = ARRAY_SIZE(alc660vd_dac_nids),
13073 .dac_nids = alc660vd_dac_nids,
Kailang Yangbdd148a2007-05-08 15:19:08 +020013074 .num_channel_mode = ARRAY_SIZE(alc861vd_3stack_2ch_modes),
13075 .channel_mode = alc861vd_3stack_2ch_modes,
13076 .input_mux = &alc861vd_capture_source,
13077 .unsol_event = alc861vd_lenovo_unsol_event,
13078 .init_hook = alc861vd_lenovo_automute,
13079 },
Kailang Yang272a5272007-05-14 11:00:38 +020013080 [ALC861VD_DALLAS] = {
13081 .mixers = { alc861vd_dallas_mixer },
13082 .init_verbs = { alc861vd_dallas_verbs },
13083 .num_dacs = ARRAY_SIZE(alc861vd_dac_nids),
13084 .dac_nids = alc861vd_dac_nids,
Kailang Yang272a5272007-05-14 11:00:38 +020013085 .num_channel_mode = ARRAY_SIZE(alc861vd_3stack_2ch_modes),
13086 .channel_mode = alc861vd_3stack_2ch_modes,
13087 .input_mux = &alc861vd_dallas_capture_source,
13088 .unsol_event = alc861vd_dallas_unsol_event,
13089 .init_hook = alc861vd_dallas_automute,
Kailang Yangd1a991a2007-08-15 16:21:59 +020013090 },
13091 [ALC861VD_HP] = {
13092 .mixers = { alc861vd_hp_mixer },
13093 .init_verbs = { alc861vd_dallas_verbs, alc861vd_eapd_verbs },
13094 .num_dacs = ARRAY_SIZE(alc861vd_dac_nids),
13095 .dac_nids = alc861vd_dac_nids,
Kailang Yangd1a991a2007-08-15 16:21:59 +020013096 .dig_out_nid = ALC861VD_DIGOUT_NID,
Kailang Yangd1a991a2007-08-15 16:21:59 +020013097 .num_channel_mode = ARRAY_SIZE(alc861vd_3stack_2ch_modes),
13098 .channel_mode = alc861vd_3stack_2ch_modes,
13099 .input_mux = &alc861vd_hp_capture_source,
13100 .unsol_event = alc861vd_dallas_unsol_event,
13101 .init_hook = alc861vd_dallas_automute,
13102 },
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013103};
13104
13105/*
13106 * BIOS auto configuration
13107 */
13108static void alc861vd_auto_set_output_and_unmute(struct hda_codec *codec,
13109 hda_nid_t nid, int pin_type, int dac_idx)
13110{
Takashi Iwaif6c7e542008-02-12 18:32:23 +010013111 alc_set_pin_output(codec, nid, pin_type);
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013112}
13113
13114static void alc861vd_auto_init_multi_out(struct hda_codec *codec)
13115{
13116 struct alc_spec *spec = codec->spec;
13117 int i;
13118
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013119 alc_subsystem_id(codec, 0x15, 0x1b, 0x14);
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013120 for (i = 0; i <= HDA_SIDE; i++) {
13121 hda_nid_t nid = spec->autocfg.line_out_pins[i];
Takashi Iwaibaba8ee2007-04-23 17:17:48 +020013122 int pin_type = get_pin_type(spec->autocfg.line_out_type);
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013123 if (nid)
13124 alc861vd_auto_set_output_and_unmute(codec, nid,
Takashi Iwaibaba8ee2007-04-23 17:17:48 +020013125 pin_type, i);
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013126 }
13127}
13128
13129
13130static void alc861vd_auto_init_hp_out(struct hda_codec *codec)
13131{
13132 struct alc_spec *spec = codec->spec;
13133 hda_nid_t pin;
13134
13135 pin = spec->autocfg.hp_pins[0];
13136 if (pin) /* connect to front and use dac 0 */
13137 alc861vd_auto_set_output_and_unmute(codec, pin, PIN_HP, 0);
Takashi Iwaif6c7e542008-02-12 18:32:23 +010013138 pin = spec->autocfg.speaker_pins[0];
13139 if (pin)
13140 alc861vd_auto_set_output_and_unmute(codec, pin, PIN_OUT, 0);
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013141}
13142
13143#define alc861vd_is_input_pin(nid) alc880_is_input_pin(nid)
13144#define ALC861VD_PIN_CD_NID ALC880_PIN_CD_NID
13145
13146static void alc861vd_auto_init_analog_input(struct hda_codec *codec)
13147{
13148 struct alc_spec *spec = codec->spec;
13149 int i;
13150
13151 for (i = 0; i < AUTO_PIN_LAST; i++) {
13152 hda_nid_t nid = spec->autocfg.input_pins[i];
13153 if (alc861vd_is_input_pin(nid)) {
13154 snd_hda_codec_write(codec, nid, 0,
13155 AC_VERB_SET_PIN_WIDGET_CONTROL,
13156 i <= AUTO_PIN_FRONT_MIC ?
13157 PIN_VREF80 : PIN_IN);
13158 if (nid != ALC861VD_PIN_CD_NID)
13159 snd_hda_codec_write(codec, nid, 0,
13160 AC_VERB_SET_AMP_GAIN_MUTE,
13161 AMP_OUT_MUTE);
13162 }
13163 }
13164}
13165
13166#define alc861vd_idx_to_mixer_vol(nid) ((nid) + 0x02)
13167#define alc861vd_idx_to_mixer_switch(nid) ((nid) + 0x0c)
13168
13169/* add playback controls from the parsed DAC table */
13170/* Based on ALC880 version. But ALC861VD has separate,
13171 * different NIDs for mute/unmute switch and volume control */
13172static int alc861vd_auto_create_multi_out_ctls(struct alc_spec *spec,
13173 const struct auto_pin_cfg *cfg)
13174{
13175 char name[32];
13176 static const char *chname[4] = {"Front", "Surround", "CLFE", "Side"};
13177 hda_nid_t nid_v, nid_s;
13178 int i, err;
13179
13180 for (i = 0; i < cfg->line_outs; i++) {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013181 if (!spec->multiout.dac_nids[i])
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013182 continue;
13183 nid_v = alc861vd_idx_to_mixer_vol(
13184 alc880_dac_to_idx(
13185 spec->multiout.dac_nids[i]));
13186 nid_s = alc861vd_idx_to_mixer_switch(
13187 alc880_dac_to_idx(
13188 spec->multiout.dac_nids[i]));
13189
13190 if (i == 2) {
13191 /* Center/LFE */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013192 err = add_control(spec, ALC_CTL_WIDGET_VOL,
13193 "Center Playback Volume",
13194 HDA_COMPOSE_AMP_VAL(nid_v, 1, 0,
13195 HDA_OUTPUT));
13196 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013197 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013198 err = add_control(spec, ALC_CTL_WIDGET_VOL,
13199 "LFE Playback Volume",
13200 HDA_COMPOSE_AMP_VAL(nid_v, 2, 0,
13201 HDA_OUTPUT));
13202 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013203 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013204 err = add_control(spec, ALC_CTL_BIND_MUTE,
13205 "Center Playback Switch",
13206 HDA_COMPOSE_AMP_VAL(nid_s, 1, 2,
13207 HDA_INPUT));
13208 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013209 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013210 err = add_control(spec, ALC_CTL_BIND_MUTE,
13211 "LFE Playback Switch",
13212 HDA_COMPOSE_AMP_VAL(nid_s, 2, 2,
13213 HDA_INPUT));
13214 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013215 return err;
13216 } else {
13217 sprintf(name, "%s Playback Volume", chname[i]);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013218 err = add_control(spec, ALC_CTL_WIDGET_VOL, name,
13219 HDA_COMPOSE_AMP_VAL(nid_v, 3, 0,
13220 HDA_OUTPUT));
13221 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013222 return err;
13223 sprintf(name, "%s Playback Switch", chname[i]);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013224 err = add_control(spec, ALC_CTL_BIND_MUTE, name,
Kailang Yangbdd148a2007-05-08 15:19:08 +020013225 HDA_COMPOSE_AMP_VAL(nid_s, 3, 2,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013226 HDA_INPUT));
13227 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013228 return err;
13229 }
13230 }
13231 return 0;
13232}
13233
13234/* add playback controls for speaker and HP outputs */
13235/* Based on ALC880 version. But ALC861VD has separate,
13236 * different NIDs for mute/unmute switch and volume control */
13237static int alc861vd_auto_create_extra_out(struct alc_spec *spec,
13238 hda_nid_t pin, const char *pfx)
13239{
13240 hda_nid_t nid_v, nid_s;
13241 int err;
13242 char name[32];
13243
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013244 if (!pin)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013245 return 0;
13246
13247 if (alc880_is_fixed_pin(pin)) {
13248 nid_v = alc880_idx_to_dac(alc880_fixed_pin_idx(pin));
13249 /* specify the DAC as the extra output */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013250 if (!spec->multiout.hp_nid)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013251 spec->multiout.hp_nid = nid_v;
13252 else
13253 spec->multiout.extra_out_nid[0] = nid_v;
13254 /* control HP volume/switch on the output mixer amp */
13255 nid_v = alc861vd_idx_to_mixer_vol(
13256 alc880_fixed_pin_idx(pin));
13257 nid_s = alc861vd_idx_to_mixer_switch(
13258 alc880_fixed_pin_idx(pin));
13259
13260 sprintf(name, "%s Playback Volume", pfx);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013261 err = add_control(spec, ALC_CTL_WIDGET_VOL, name,
13262 HDA_COMPOSE_AMP_VAL(nid_v, 3, 0, HDA_OUTPUT));
13263 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013264 return err;
13265 sprintf(name, "%s Playback Switch", pfx);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013266 err = add_control(spec, ALC_CTL_BIND_MUTE, name,
13267 HDA_COMPOSE_AMP_VAL(nid_s, 3, 2, HDA_INPUT));
13268 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013269 return err;
13270 } else if (alc880_is_multi_pin(pin)) {
13271 /* set manual connection */
13272 /* we have only a switch on HP-out PIN */
13273 sprintf(name, "%s Playback Switch", pfx);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013274 err = add_control(spec, ALC_CTL_WIDGET_MUTE, name,
13275 HDA_COMPOSE_AMP_VAL(pin, 3, 0, HDA_OUTPUT));
13276 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013277 return err;
13278 }
13279 return 0;
13280}
13281
13282/* parse the BIOS configuration and set up the alc_spec
13283 * return 1 if successful, 0 if the proper config is not found,
13284 * or a negative error code
13285 * Based on ALC880 version - had to change it to override
13286 * alc880_auto_create_extra_out and alc880_auto_create_multi_out_ctls */
13287static int alc861vd_parse_auto_config(struct hda_codec *codec)
13288{
13289 struct alc_spec *spec = codec->spec;
13290 int err;
13291 static hda_nid_t alc861vd_ignore[] = { 0x1d, 0 };
13292
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013293 err = snd_hda_parse_pin_def_config(codec, &spec->autocfg,
13294 alc861vd_ignore);
13295 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013296 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013297 if (!spec->autocfg.line_outs)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013298 return 0; /* can't find valid BIOS pin config */
13299
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013300 err = alc880_auto_fill_dac_nids(spec, &spec->autocfg);
13301 if (err < 0)
13302 return err;
13303 err = alc861vd_auto_create_multi_out_ctls(spec, &spec->autocfg);
13304 if (err < 0)
13305 return err;
13306 err = alc861vd_auto_create_extra_out(spec,
13307 spec->autocfg.speaker_pins[0],
13308 "Speaker");
13309 if (err < 0)
13310 return err;
13311 err = alc861vd_auto_create_extra_out(spec,
13312 spec->autocfg.hp_pins[0],
13313 "Headphone");
13314 if (err < 0)
13315 return err;
13316 err = alc880_auto_create_analog_input_ctls(spec, &spec->autocfg);
13317 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013318 return err;
13319
13320 spec->multiout.max_channels = spec->multiout.num_dacs * 2;
13321
13322 if (spec->autocfg.dig_out_pin)
13323 spec->multiout.dig_out_nid = ALC861VD_DIGOUT_NID;
13324
13325 if (spec->kctl_alloc)
13326 spec->mixers[spec->num_mixers++] = spec->kctl_alloc;
13327
13328 spec->init_verbs[spec->num_init_verbs++]
13329 = alc861vd_volume_init_verbs;
13330
13331 spec->num_mux_defs = 1;
13332 spec->input_mux = &spec->private_imux;
13333
Takashi Iwai776e1842007-08-29 15:07:11 +020013334 err = alc_auto_add_mic_boost(codec);
13335 if (err < 0)
13336 return err;
13337
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013338 return 1;
13339}
13340
13341/* additional initialization for auto-configuration model */
13342static void alc861vd_auto_init(struct hda_codec *codec)
13343{
Takashi Iwaif6c7e542008-02-12 18:32:23 +010013344 struct alc_spec *spec = codec->spec;
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013345 alc861vd_auto_init_multi_out(codec);
13346 alc861vd_auto_init_hp_out(codec);
13347 alc861vd_auto_init_analog_input(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +010013348 if (spec->unsol_event)
13349 alc_sku_automute(codec);
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013350}
13351
13352static int patch_alc861vd(struct hda_codec *codec)
13353{
13354 struct alc_spec *spec;
13355 int err, board_config;
13356
13357 spec = kzalloc(sizeof(*spec), GFP_KERNEL);
13358 if (spec == NULL)
13359 return -ENOMEM;
13360
13361 codec->spec = spec;
13362
13363 board_config = snd_hda_check_board_config(codec, ALC861VD_MODEL_LAST,
13364 alc861vd_models,
13365 alc861vd_cfg_tbl);
13366
13367 if (board_config < 0 || board_config >= ALC861VD_MODEL_LAST) {
13368 printk(KERN_INFO "hda_codec: Unknown model for ALC660VD/"
13369 "ALC861VD, trying auto-probe from BIOS...\n");
13370 board_config = ALC861VD_AUTO;
13371 }
13372
13373 if (board_config == ALC861VD_AUTO) {
13374 /* automatic parse from the BIOS config */
13375 err = alc861vd_parse_auto_config(codec);
13376 if (err < 0) {
13377 alc_free(codec);
13378 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013379 } else if (!err) {
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013380 printk(KERN_INFO
13381 "hda_codec: Cannot set up configuration "
13382 "from BIOS. Using base mode...\n");
13383 board_config = ALC861VD_3ST;
13384 }
13385 }
13386
13387 if (board_config != ALC861VD_AUTO)
13388 setup_preset(spec, &alc861vd_presets[board_config]);
13389
Kailang Yang2f893282008-05-27 12:14:47 +020013390 if (codec->vendor_id == 0x10ec0660) {
13391 spec->stream_name_analog = "ALC660-VD Analog";
13392 spec->stream_name_digital = "ALC660-VD Digital";
Kailang Yangf9423e72008-05-27 12:32:25 +020013393 /* always turn on EAPD */
13394 spec->init_verbs[spec->num_init_verbs++] = alc660vd_eapd_verbs;
Kailang Yang2f893282008-05-27 12:14:47 +020013395 } else {
13396 spec->stream_name_analog = "ALC861VD Analog";
13397 spec->stream_name_digital = "ALC861VD Digital";
13398 }
13399
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013400 spec->stream_analog_playback = &alc861vd_pcm_analog_playback;
13401 spec->stream_analog_capture = &alc861vd_pcm_analog_capture;
13402
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013403 spec->stream_digital_playback = &alc861vd_pcm_digital_playback;
13404 spec->stream_digital_capture = &alc861vd_pcm_digital_capture;
13405
13406 spec->adc_nids = alc861vd_adc_nids;
13407 spec->num_adc_nids = ARRAY_SIZE(alc861vd_adc_nids);
Takashi Iwaie1406342008-02-11 18:32:32 +010013408 spec->capsrc_nids = alc861vd_capsrc_nids;
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013409
13410 spec->mixers[spec->num_mixers] = alc861vd_capture_mixer;
13411 spec->num_mixers++;
13412
Takashi Iwai2134ea42008-01-10 16:53:55 +010013413 spec->vmaster_nid = 0x02;
13414
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013415 codec->patch_ops = alc_patch_ops;
13416
13417 if (board_config == ALC861VD_AUTO)
13418 spec->init_hook = alc861vd_auto_init;
Takashi Iwaicb53c622007-08-10 17:21:45 +020013419#ifdef CONFIG_SND_HDA_POWER_SAVE
13420 if (!spec->loopback.amplist)
13421 spec->loopback.amplist = alc861vd_loopbacks;
13422#endif
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010013423
13424 return 0;
13425}
13426
13427/*
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013428 * ALC662 support
13429 *
13430 * ALC662 is almost identical with ALC880 but has cleaner and more flexible
13431 * configuration. Each pin widget can choose any input DACs and a mixer.
13432 * Each ADC is connected from a mixer of all inputs. This makes possible
13433 * 6-channel independent captures.
13434 *
13435 * In addition, an independent DAC for the multi-playback (not used in this
13436 * driver yet).
13437 */
13438#define ALC662_DIGOUT_NID 0x06
13439#define ALC662_DIGIN_NID 0x0a
13440
13441static hda_nid_t alc662_dac_nids[4] = {
13442 /* front, rear, clfe, rear_surr */
13443 0x02, 0x03, 0x04
13444};
13445
13446static hda_nid_t alc662_adc_nids[1] = {
13447 /* ADC1-2 */
13448 0x09,
13449};
Takashi Iwaie1406342008-02-11 18:32:32 +010013450
Kailang Yang77a261b2008-02-19 11:38:05 +010013451static hda_nid_t alc662_capsrc_nids[1] = { 0x22 };
Takashi Iwaie1406342008-02-11 18:32:32 +010013452
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013453/* input MUX */
13454/* FIXME: should be a matrix-type input source selection */
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013455static struct hda_input_mux alc662_capture_source = {
13456 .num_items = 4,
13457 .items = {
13458 { "Mic", 0x0 },
13459 { "Front Mic", 0x1 },
13460 { "Line", 0x2 },
13461 { "CD", 0x4 },
13462 },
13463};
13464
13465static struct hda_input_mux alc662_lenovo_101e_capture_source = {
13466 .num_items = 2,
13467 .items = {
13468 { "Mic", 0x1 },
13469 { "Line", 0x2 },
13470 },
13471};
Kailang Yang291702f2007-10-16 14:28:03 +020013472
13473static struct hda_input_mux alc662_eeepc_capture_source = {
13474 .num_items = 2,
13475 .items = {
13476 { "i-Mic", 0x1 },
13477 { "e-Mic", 0x0 },
13478 },
13479};
13480
Kailang Yang6dda9f42008-05-27 12:05:31 +020013481static struct hda_input_mux alc663_capture_source = {
13482 .num_items = 3,
13483 .items = {
13484 { "Mic", 0x0 },
13485 { "Front Mic", 0x1 },
13486 { "Line", 0x2 },
13487 },
13488};
13489
13490static struct hda_input_mux alc663_m51va_capture_source = {
13491 .num_items = 2,
13492 .items = {
13493 { "Ext-Mic", 0x0 },
13494 { "D-Mic", 0x9 },
13495 },
13496};
13497
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013498#define alc662_mux_enum_info alc_mux_enum_info
13499#define alc662_mux_enum_get alc_mux_enum_get
Takashi Iwaie1406342008-02-11 18:32:32 +010013500#define alc662_mux_enum_put alc882_mux_enum_put
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013501
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013502/*
13503 * 2ch mode
13504 */
13505static struct hda_channel_mode alc662_3ST_2ch_modes[1] = {
13506 { 2, NULL }
13507};
13508
13509/*
13510 * 2ch mode
13511 */
13512static struct hda_verb alc662_3ST_ch2_init[] = {
13513 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
13514 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
13515 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
13516 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
13517 { } /* end */
13518};
13519
13520/*
13521 * 6ch mode
13522 */
13523static struct hda_verb alc662_3ST_ch6_init[] = {
13524 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
13525 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
13526 { 0x18, AC_VERB_SET_CONNECT_SEL, 0x02 },
13527 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
13528 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
13529 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x01 },
13530 { } /* end */
13531};
13532
13533static struct hda_channel_mode alc662_3ST_6ch_modes[2] = {
13534 { 2, alc662_3ST_ch2_init },
13535 { 6, alc662_3ST_ch6_init },
13536};
13537
13538/*
13539 * 2ch mode
13540 */
13541static struct hda_verb alc662_sixstack_ch6_init[] = {
13542 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
13543 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
13544 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
13545 { } /* end */
13546};
13547
13548/*
13549 * 6ch mode
13550 */
13551static struct hda_verb alc662_sixstack_ch8_init[] = {
13552 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
13553 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
13554 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
13555 { } /* end */
13556};
13557
13558static struct hda_channel_mode alc662_5stack_modes[2] = {
13559 { 2, alc662_sixstack_ch6_init },
13560 { 6, alc662_sixstack_ch8_init },
13561};
13562
13563/* Pin assignment: Front=0x14, Rear=0x15, CLFE=0x16, Side=0x17
13564 * Mic=0x18, Front Mic=0x19, Line-In=0x1a, HP=0x1b
13565 */
13566
13567static struct snd_kcontrol_new alc662_base_mixer[] = {
13568 /* output mixer control */
13569 HDA_CODEC_VOLUME("Front Playback Volume", 0x2, 0x0, HDA_OUTPUT),
Herton Ronaldo Krzesinski7055ad82008-03-14 12:52:20 +010013570 HDA_CODEC_MUTE("Front Playback Switch", 0x0c, 0x0, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013571 HDA_CODEC_VOLUME("Surround Playback Volume", 0x3, 0x0, HDA_OUTPUT),
Herton Ronaldo Krzesinski7055ad82008-03-14 12:52:20 +010013572 HDA_CODEC_MUTE("Surround Playback Switch", 0x0d, 0x0, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013573 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x04, 1, 0x0, HDA_OUTPUT),
13574 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x04, 2, 0x0, HDA_OUTPUT),
Herton Ronaldo Krzesinski7055ad82008-03-14 12:52:20 +010013575 HDA_CODEC_MUTE_MONO("Center Playback Switch", 0x0e, 1, 0x0, HDA_INPUT),
13576 HDA_CODEC_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 0x0, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013577 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
13578
13579 /*Input mixer control */
13580 HDA_CODEC_VOLUME("CD Playback Volume", 0xb, 0x4, HDA_INPUT),
13581 HDA_CODEC_MUTE("CD Playback Switch", 0xb, 0x4, HDA_INPUT),
13582 HDA_CODEC_VOLUME("Line Playback Volume", 0xb, 0x02, HDA_INPUT),
13583 HDA_CODEC_MUTE("Line Playback Switch", 0xb, 0x02, HDA_INPUT),
13584 HDA_CODEC_VOLUME("Mic Playback Volume", 0xb, 0x0, HDA_INPUT),
13585 HDA_CODEC_MUTE("Mic Playback Switch", 0xb, 0x0, HDA_INPUT),
13586 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0xb, 0x01, HDA_INPUT),
13587 HDA_CODEC_MUTE("Front Mic Playback Switch", 0xb, 0x01, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013588 { } /* end */
13589};
13590
13591static struct snd_kcontrol_new alc662_3ST_2ch_mixer[] = {
13592 HDA_CODEC_VOLUME("Front Playback Volume", 0x02, 0x0, HDA_OUTPUT),
Herton Ronaldo Krzesinski7055ad82008-03-14 12:52:20 +010013593 HDA_CODEC_MUTE("Front Playback Switch", 0x0c, 0x0, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013594 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
13595 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
13596 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
13597 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
13598 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
13599 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
13600 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
13601 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
13602 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
13603 HDA_CODEC_VOLUME("PC Speaker Playback Volume", 0x0b, 0x05, HDA_INPUT),
13604 HDA_CODEC_MUTE("PC Speaker Playback Switch", 0x0b, 0x05, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013605 { } /* end */
13606};
13607
13608static struct snd_kcontrol_new alc662_3ST_6ch_mixer[] = {
13609 HDA_CODEC_VOLUME("Front Playback Volume", 0x02, 0x0, HDA_OUTPUT),
Herton Ronaldo Krzesinski7055ad82008-03-14 12:52:20 +010013610 HDA_CODEC_MUTE("Front Playback Switch", 0x0c, 0x0, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013611 HDA_CODEC_VOLUME("Surround Playback Volume", 0x03, 0x0, HDA_OUTPUT),
Herton Ronaldo Krzesinski7055ad82008-03-14 12:52:20 +010013612 HDA_CODEC_MUTE("Surround Playback Switch", 0x0d, 0x0, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013613 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x04, 1, 0x0, HDA_OUTPUT),
13614 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x04, 2, 0x0, HDA_OUTPUT),
Herton Ronaldo Krzesinski7055ad82008-03-14 12:52:20 +010013615 HDA_CODEC_MUTE_MONO("Center Playback Switch", 0x0e, 1, 0x0, HDA_INPUT),
13616 HDA_CODEC_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 0x0, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013617 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
13618 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
13619 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
13620 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
13621 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
13622 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
13623 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
13624 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
13625 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
13626 HDA_CODEC_VOLUME("PC Speaker Playback Volume", 0x0b, 0x05, HDA_INPUT),
13627 HDA_CODEC_MUTE("PC Speaker Playback Switch", 0x0b, 0x05, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013628 { } /* end */
13629};
13630
13631static struct snd_kcontrol_new alc662_lenovo_101e_mixer[] = {
13632 HDA_CODEC_VOLUME("Front Playback Volume", 0x02, 0x0, HDA_OUTPUT),
13633 HDA_BIND_MUTE("Front Playback Switch", 0x02, 2, HDA_INPUT),
Takashi Iwai86cd9292008-01-28 18:09:56 +010013634 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x03, 0x0, HDA_OUTPUT),
13635 HDA_BIND_MUTE("Speaker Playback Switch", 0x03, 2, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013636 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
13637 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
13638 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
13639 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
13640 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013641 { } /* end */
13642};
13643
Kailang Yang291702f2007-10-16 14:28:03 +020013644static struct snd_kcontrol_new alc662_eeepc_p701_mixer[] = {
Takashi Iwai86cd9292008-01-28 18:09:56 +010013645 HDA_CODEC_MUTE("Speaker Playback Switch", 0x14, 0x0, HDA_OUTPUT),
Kailang Yang291702f2007-10-16 14:28:03 +020013646
Herton Ronaldo Krzesinskib4818492008-02-23 11:34:12 +010013647 HDA_CODEC_VOLUME("Line-Out Playback Volume", 0x02, 0x0, HDA_OUTPUT),
13648 HDA_CODEC_MUTE("Line-Out Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
Kailang Yang291702f2007-10-16 14:28:03 +020013649
13650 HDA_CODEC_VOLUME("e-Mic Boost", 0x18, 0, HDA_INPUT),
13651 HDA_CODEC_VOLUME("e-Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
13652 HDA_CODEC_MUTE("e-Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
13653
13654 HDA_CODEC_VOLUME("i-Mic Boost", 0x19, 0, HDA_INPUT),
13655 HDA_CODEC_VOLUME("i-Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
13656 HDA_CODEC_MUTE("i-Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
13657 { } /* end */
13658};
13659
Kailang Yang8c427222008-01-10 13:03:59 +010013660static struct snd_kcontrol_new alc662_eeepc_ep20_mixer[] = {
Takashi Iwai31bffaa2008-02-27 16:10:44 +010013661 HDA_CODEC_VOLUME("Line-Out Playback Volume", 0x02, 0x0, HDA_OUTPUT),
13662 HDA_CODEC_MUTE("Line-Out Playback Switch", 0x14, 0x0, HDA_OUTPUT),
Kailang Yang8c427222008-01-10 13:03:59 +010013663 HDA_CODEC_VOLUME("Surround Playback Volume", 0x03, 0x0, HDA_OUTPUT),
13664 HDA_BIND_MUTE("Surround Playback Switch", 0x03, 2, HDA_INPUT),
13665 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x04, 1, 0x0, HDA_OUTPUT),
13666 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x04, 2, 0x0, HDA_OUTPUT),
13667 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x04, 1, 2, HDA_INPUT),
13668 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x04, 2, 2, HDA_INPUT),
Takashi Iwai86cd9292008-01-28 18:09:56 +010013669 HDA_CODEC_MUTE("Speaker Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
Kailang Yang8c427222008-01-10 13:03:59 +010013670 HDA_BIND_MUTE("MuteCtrl Playback Switch", 0x0c, 2, HDA_INPUT),
13671 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
13672 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
13673 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
13674 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
13675 { } /* end */
13676};
13677
Kailang Yang6dda9f42008-05-27 12:05:31 +020013678static struct snd_kcontrol_new alc663_m51va_mixer[] = {
13679 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x02, 0x0, HDA_OUTPUT),
13680 HDA_CODEC_MUTE("Speaker Playback Switch", 0x14, 0x0, HDA_OUTPUT),
13681 HDA_CODEC_MUTE("Headphone Playback Switch", 0x21, 0x0, HDA_OUTPUT),
13682 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
13683 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
13684 HDA_CODEC_MUTE("DMic Playback Switch", 0x23, 0x9, HDA_INPUT),
13685 { } /* end */
13686};
13687
13688static struct snd_kcontrol_new alc663_g71v_mixer[] = {
13689 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x02, 0x0, HDA_OUTPUT),
13690 HDA_CODEC_MUTE("Speaker Playback Switch", 0x14, 0x0, HDA_OUTPUT),
13691 HDA_CODEC_VOLUME("Front Playback Volume", 0x03, 0x0, HDA_OUTPUT),
13692 HDA_CODEC_MUTE("Front Playback Switch", 0x15, 0x0, HDA_OUTPUT),
13693 HDA_CODEC_MUTE("Headphone Playback Switch", 0x21, 0x0, HDA_OUTPUT),
13694
13695 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
13696 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
13697 HDA_CODEC_VOLUME("i-Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
13698 HDA_CODEC_MUTE("i-Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
13699 { } /* end */
13700};
13701
13702static struct snd_kcontrol_new alc663_g50v_mixer[] = {
13703 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x02, 0x0, HDA_OUTPUT),
13704 HDA_CODEC_MUTE("Speaker Playback Switch", 0x14, 0x0, HDA_OUTPUT),
13705 HDA_CODEC_MUTE("Headphone Playback Switch", 0x21, 0x0, HDA_OUTPUT),
13706
13707 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
13708 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
13709 HDA_CODEC_VOLUME("i-Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
13710 HDA_CODEC_MUTE("i-Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
13711 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
13712 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
13713 { } /* end */
13714};
13715
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013716static struct snd_kcontrol_new alc662_chmode_mixer[] = {
13717 {
13718 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
13719 .name = "Channel Mode",
13720 .info = alc_ch_mode_info,
13721 .get = alc_ch_mode_get,
13722 .put = alc_ch_mode_put,
13723 },
13724 { } /* end */
13725};
13726
13727static struct hda_verb alc662_init_verbs[] = {
13728 /* ADC: mute amp left and right */
13729 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
13730 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
13731 /* Front mixer: unmute input/output amp left and right (volume = 0) */
13732
Takashi Iwaicb53c622007-08-10 17:21:45 +020013733 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
13734 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
13735 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
13736 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
13737 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013738
Kailang Yangb60dd392007-09-20 12:50:29 +020013739 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13740 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13741 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13742 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13743 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13744 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013745
13746 /* Front Pin: output 0 (0x0c) */
13747 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
13748 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13749
13750 /* Rear Pin: output 1 (0x0d) */
13751 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
13752 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13753
13754 /* CLFE Pin: output 2 (0x0e) */
13755 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
13756 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13757
13758 /* Mic (rear) pin: input vref at 80% */
13759 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
13760 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
13761 /* Front Mic pin: input vref at 80% */
13762 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
13763 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
13764 /* Line In pin: input */
13765 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
13766 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
13767 /* Line-2 In: Headphone output (output 0 - 0x0c) */
13768 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
13769 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13770 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
13771 /* CD pin widget for input */
13772 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
13773
13774 /* FIXME: use matrix-type input source selection */
13775 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
13776 /* Input mixer */
13777 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13778 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13779 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
13780 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(4)},
Kailang Yang291702f2007-10-16 14:28:03 +020013781
13782 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13783 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13784 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
13785 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(4)},
Kailang Yang6dda9f42008-05-27 12:05:31 +020013786
13787 /* always trun on EAPD */
13788 {0x14, AC_VERB_SET_EAPD_BTLENABLE, 2},
13789 {0x15, AC_VERB_SET_EAPD_BTLENABLE, 2},
13790
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013791 { }
13792};
13793
13794static struct hda_verb alc662_sue_init_verbs[] = {
13795 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|ALC880_FRONT_EVENT},
13796 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|ALC880_HP_EVENT},
Kailang Yang291702f2007-10-16 14:28:03 +020013797 {}
13798};
13799
13800static struct hda_verb alc662_eeepc_sue_init_verbs[] = {
13801 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
13802 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
13803 {}
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013804};
13805
Kailang Yang8c427222008-01-10 13:03:59 +010013806/* Set Unsolicited Event*/
13807static struct hda_verb alc662_eeepc_ep20_sue_init_verbs[] = {
13808 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
13809 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
13810 {}
13811};
13812
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013813/*
13814 * generic initialization of ADC, input mixers and output mixers
13815 */
13816static struct hda_verb alc662_auto_init_verbs[] = {
13817 /*
13818 * Unmute ADC and set the default input to mic-in
13819 */
13820 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
13821 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13822
13823 /* Unmute input amps (CD, Line In, Mic 1 & Mic 2) of the analog-loopback
13824 * mixer widget
13825 * Note: PASD motherboards uses the Line In 2 as the input for front
13826 * panel mic (mic 2)
13827 */
13828 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +020013829 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
13830 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
13831 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
13832 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
13833 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013834
13835 /*
13836 * Set up output mixers (0x0c - 0x0f)
13837 */
13838 /* set vol=0 to output mixers */
13839 {0x02, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
13840 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
13841 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
13842
13843 /* set up input amps for analog loopback */
13844 /* Amp Indices: DAC = 0, mixer = 1 */
Kailang Yangb60dd392007-09-20 12:50:29 +020013845 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13846 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13847 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13848 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13849 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13850 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013851
13852
13853 /* FIXME: use matrix-type input source selection */
13854 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
13855 /* Input mixer */
13856 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Kailang Yangd1a991a2007-08-15 16:21:59 +020013857 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013858 { }
13859};
13860
Kailang Yang6dda9f42008-05-27 12:05:31 +020013861static struct hda_verb alc663_m51va_init_verbs[] = {
13862 {0x21, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
13863 {0x21, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13864 {0x21, AC_VERB_SET_CONNECT_SEL, 0x00}, /* Headphone */
13865
13866 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(9)},
13867
13868 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
13869 {0x21, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
13870 {}
13871};
13872
13873static struct hda_verb alc663_g71v_init_verbs[] = {
13874 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
13875 /* {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE}, */
13876 /* {0x15, AC_VERB_SET_CONNECT_SEL, 0x01}, */ /* Headphone */
13877
13878 {0x21, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
13879 {0x21, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13880 {0x21, AC_VERB_SET_CONNECT_SEL, 0x00}, /* Headphone */
13881
13882 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|ALC880_FRONT_EVENT},
13883 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|ALC880_MIC_EVENT},
13884 {0x21, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|ALC880_HP_EVENT},
13885 {}
13886};
13887
13888static struct hda_verb alc663_g50v_init_verbs[] = {
13889 {0x21, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
13890 {0x21, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13891 {0x21, AC_VERB_SET_CONNECT_SEL, 0x00}, /* Headphone */
13892
13893 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
13894 {0x21, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
13895 {}
13896};
13897
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013898/* capture mixer elements */
13899static struct snd_kcontrol_new alc662_capture_mixer[] = {
13900 HDA_CODEC_VOLUME("Capture Volume", 0x09, 0x0, HDA_INPUT),
13901 HDA_CODEC_MUTE("Capture Switch", 0x09, 0x0, HDA_INPUT),
13902 {
13903 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
13904 /* The multiple "Capture Source" controls confuse alsamixer
13905 * So call somewhat different..
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013906 */
13907 /* .name = "Capture Source", */
13908 .name = "Input Source",
13909 .count = 1,
Herton Ronaldo Krzesinski6e7939b2007-12-19 17:49:02 +010013910 .info = alc662_mux_enum_info,
13911 .get = alc662_mux_enum_get,
13912 .put = alc662_mux_enum_put,
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013913 },
13914 { } /* end */
13915};
13916
13917static void alc662_lenovo_101e_ispeaker_automute(struct hda_codec *codec)
13918{
13919 unsigned int present;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013920 unsigned char bits;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013921
13922 present = snd_hda_codec_read(codec, 0x14, 0,
13923 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +020013924 bits = present ? HDA_AMP_MUTE : 0;
13925 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
13926 HDA_AMP_MUTE, bits);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013927}
13928
13929static void alc662_lenovo_101e_all_automute(struct hda_codec *codec)
13930{
13931 unsigned int present;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013932 unsigned char bits;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013933
13934 present = snd_hda_codec_read(codec, 0x1b, 0,
13935 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
Takashi Iwai47fd8302007-08-10 17:11:07 +020013936 bits = present ? HDA_AMP_MUTE : 0;
13937 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
13938 HDA_AMP_MUTE, bits);
13939 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
13940 HDA_AMP_MUTE, bits);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020013941}
13942
13943static void alc662_lenovo_101e_unsol_event(struct hda_codec *codec,
13944 unsigned int res)
13945{
13946 if ((res >> 26) == ALC880_HP_EVENT)
13947 alc662_lenovo_101e_all_automute(codec);
13948 if ((res >> 26) == ALC880_FRONT_EVENT)
13949 alc662_lenovo_101e_ispeaker_automute(codec);
13950}
13951
Kailang Yang291702f2007-10-16 14:28:03 +020013952static void alc662_eeepc_mic_automute(struct hda_codec *codec)
13953{
13954 unsigned int present;
13955
13956 present = snd_hda_codec_read(codec, 0x18, 0,
13957 AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
13958 snd_hda_codec_write(codec, 0x22, 0, AC_VERB_SET_AMP_GAIN_MUTE,
13959 0x7000 | (0x00 << 8) | (present ? 0 : 0x80));
13960 snd_hda_codec_write(codec, 0x23, 0, AC_VERB_SET_AMP_GAIN_MUTE,
13961 0x7000 | (0x00 << 8) | (present ? 0 : 0x80));
13962 snd_hda_codec_write(codec, 0x22, 0, AC_VERB_SET_AMP_GAIN_MUTE,
13963 0x7000 | (0x01 << 8) | (present ? 0x80 : 0));
13964 snd_hda_codec_write(codec, 0x23, 0, AC_VERB_SET_AMP_GAIN_MUTE,
13965 0x7000 | (0x01 << 8) | (present ? 0x80 : 0));
13966}
13967
13968/* unsolicited event for HP jack sensing */
13969static void alc662_eeepc_unsol_event(struct hda_codec *codec,
13970 unsigned int res)
13971{
13972 if ((res >> 26) == ALC880_HP_EVENT)
13973 alc262_hippo1_automute( codec );
13974
13975 if ((res >> 26) == ALC880_MIC_EVENT)
13976 alc662_eeepc_mic_automute(codec);
13977}
13978
13979static void alc662_eeepc_inithook(struct hda_codec *codec)
13980{
13981 alc262_hippo1_automute( codec );
13982 alc662_eeepc_mic_automute(codec);
13983}
13984
Kailang Yang8c427222008-01-10 13:03:59 +010013985static void alc662_eeepc_ep20_automute(struct hda_codec *codec)
13986{
13987 unsigned int mute;
13988 unsigned int present;
13989
13990 snd_hda_codec_read(codec, 0x14, 0, AC_VERB_SET_PIN_SENSE, 0);
13991 present = snd_hda_codec_read(codec, 0x14, 0,
13992 AC_VERB_GET_PIN_SENSE, 0);
13993 present = (present & 0x80000000) != 0;
13994 if (present) {
13995 /* mute internal speaker */
13996 snd_hda_codec_amp_stereo(codec, 0x1b, HDA_OUTPUT, 0,
13997 HDA_AMP_MUTE, HDA_AMP_MUTE);
13998 } else {
13999 /* unmute internal speaker if necessary */
14000 mute = snd_hda_codec_amp_read(codec, 0x14, 0, HDA_OUTPUT, 0);
14001 snd_hda_codec_amp_stereo(codec, 0x1b, HDA_OUTPUT, 0,
14002 HDA_AMP_MUTE, mute);
14003 }
14004}
14005
14006/* unsolicited event for HP jack sensing */
14007static void alc662_eeepc_ep20_unsol_event(struct hda_codec *codec,
14008 unsigned int res)
14009{
14010 if ((res >> 26) == ALC880_HP_EVENT)
14011 alc662_eeepc_ep20_automute(codec);
14012}
14013
14014static void alc662_eeepc_ep20_inithook(struct hda_codec *codec)
14015{
14016 alc662_eeepc_ep20_automute(codec);
14017}
14018
Kailang Yang6dda9f42008-05-27 12:05:31 +020014019static void alc663_m51va_speaker_automute(struct hda_codec *codec)
14020{
14021 unsigned int present;
14022 unsigned char bits;
14023
14024 present = snd_hda_codec_read(codec, 0x21, 0,
14025 AC_VERB_GET_PIN_SENSE, 0)
14026 & AC_PINSENSE_PRESENCE;
14027 bits = present ? HDA_AMP_MUTE : 0;
14028 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
14029 HDA_AMP_MUTE, bits);
14030}
14031
14032static void alc663_m51va_mic_automute(struct hda_codec *codec)
14033{
14034 unsigned int present;
14035
14036 present = snd_hda_codec_read(codec, 0x18, 0,
14037 AC_VERB_GET_PIN_SENSE, 0)
14038 & AC_PINSENSE_PRESENCE;
14039 snd_hda_codec_write_cache(codec, 0x22, 0, AC_VERB_SET_AMP_GAIN_MUTE,
14040 0x7000 | (0x00 << 8) | (present ? 0 : 0x80));
14041 snd_hda_codec_write_cache(codec, 0x23, 0, AC_VERB_SET_AMP_GAIN_MUTE,
14042 0x7000 | (0x00 << 8) | (present ? 0 : 0x80));
14043 snd_hda_codec_write_cache(codec, 0x22, 0, AC_VERB_SET_AMP_GAIN_MUTE,
14044 0x7000 | (0x09 << 8) | (present ? 0x80 : 0));
14045 snd_hda_codec_write_cache(codec, 0x23, 0, AC_VERB_SET_AMP_GAIN_MUTE,
14046 0x7000 | (0x09 << 8) | (present ? 0x80 : 0));
14047}
14048
14049static void alc663_m51va_unsol_event(struct hda_codec *codec,
14050 unsigned int res)
14051{
14052 switch (res >> 26) {
14053 case ALC880_HP_EVENT:
14054 alc663_m51va_speaker_automute(codec);
14055 break;
14056 case ALC880_MIC_EVENT:
14057 alc663_m51va_mic_automute(codec);
14058 break;
14059 }
14060}
14061
14062static void alc663_m51va_inithook(struct hda_codec *codec)
14063{
14064 alc663_m51va_speaker_automute(codec);
14065 alc663_m51va_mic_automute(codec);
14066}
14067
14068static void alc663_g71v_hp_automute(struct hda_codec *codec)
14069{
14070 unsigned int present;
14071 unsigned char bits;
14072
14073 present = snd_hda_codec_read(codec, 0x21, 0,
14074 AC_VERB_GET_PIN_SENSE, 0)
14075 & AC_PINSENSE_PRESENCE;
14076 bits = present ? HDA_AMP_MUTE : 0;
14077 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
14078 HDA_AMP_MUTE, bits);
14079 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
14080 HDA_AMP_MUTE, bits);
14081}
14082
14083static void alc663_g71v_front_automute(struct hda_codec *codec)
14084{
14085 unsigned int present;
14086 unsigned char bits;
14087
14088 present = snd_hda_codec_read(codec, 0x15, 0,
14089 AC_VERB_GET_PIN_SENSE, 0)
14090 & AC_PINSENSE_PRESENCE;
14091 bits = present ? HDA_AMP_MUTE : 0;
14092 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
14093 HDA_AMP_MUTE, bits);
14094}
14095
14096static void alc663_g71v_unsol_event(struct hda_codec *codec,
14097 unsigned int res)
14098{
14099 switch (res >> 26) {
14100 case ALC880_HP_EVENT:
14101 alc663_g71v_hp_automute(codec);
14102 break;
14103 case ALC880_FRONT_EVENT:
14104 alc663_g71v_front_automute(codec);
14105 break;
14106 case ALC880_MIC_EVENT:
14107 alc662_eeepc_mic_automute(codec);
14108 break;
14109 }
14110}
14111
14112static void alc663_g71v_inithook(struct hda_codec *codec)
14113{
14114 alc663_g71v_front_automute(codec);
14115 alc663_g71v_hp_automute(codec);
14116 alc662_eeepc_mic_automute(codec);
14117}
14118
14119static void alc663_g50v_unsol_event(struct hda_codec *codec,
14120 unsigned int res)
14121{
14122 switch (res >> 26) {
14123 case ALC880_HP_EVENT:
14124 alc663_m51va_speaker_automute(codec);
14125 break;
14126 case ALC880_MIC_EVENT:
14127 alc662_eeepc_mic_automute(codec);
14128 break;
14129 }
14130}
14131
14132static void alc663_g50v_inithook(struct hda_codec *codec)
14133{
14134 alc663_m51va_speaker_automute(codec);
14135 alc662_eeepc_mic_automute(codec);
14136}
14137
Takashi Iwaicb53c622007-08-10 17:21:45 +020014138#ifdef CONFIG_SND_HDA_POWER_SAVE
14139#define alc662_loopbacks alc880_loopbacks
14140#endif
14141
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014142
14143/* pcm configuration: identiacal with ALC880 */
14144#define alc662_pcm_analog_playback alc880_pcm_analog_playback
14145#define alc662_pcm_analog_capture alc880_pcm_analog_capture
14146#define alc662_pcm_digital_playback alc880_pcm_digital_playback
14147#define alc662_pcm_digital_capture alc880_pcm_digital_capture
14148
14149/*
14150 * configuration and preset
14151 */
14152static const char *alc662_models[ALC662_MODEL_LAST] = {
14153 [ALC662_3ST_2ch_DIG] = "3stack-dig",
14154 [ALC662_3ST_6ch_DIG] = "3stack-6ch-dig",
14155 [ALC662_3ST_6ch] = "3stack-6ch",
14156 [ALC662_5ST_DIG] = "6stack-dig",
14157 [ALC662_LENOVO_101E] = "lenovo-101e",
Takashi Iwaib995d762007-10-17 10:41:06 +020014158 [ALC662_ASUS_EEEPC_P701] = "eeepc-p701",
Kailang Yang8c427222008-01-10 13:03:59 +010014159 [ALC662_ASUS_EEEPC_EP20] = "eeepc-ep20",
Kailang Yang6dda9f42008-05-27 12:05:31 +020014160 [ALC663_ASUS_M51VA] = "m51va",
14161 [ALC663_ASUS_G71V] = "g71v",
14162 [ALC663_ASUS_H13] = "h13",
14163 [ALC663_ASUS_G50V] = "g50v",
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014164 [ALC662_AUTO] = "auto",
14165};
14166
14167static struct snd_pci_quirk alc662_cfg_tbl[] = {
Kailang Yang6dda9f42008-05-27 12:05:31 +020014168 SND_PCI_QUIRK(0x1043, 0x11c3, "ASUS G71V", ALC663_ASUS_G71V),
14169 SND_PCI_QUIRK(0x1043, 0x1878, "ASUS M51VA", ALC663_ASUS_M51VA),
14170 SND_PCI_QUIRK(0x1043, 0x19a3, "ASUS M51VA", ALC663_ASUS_G50V),
Herton Ronaldo Krzesinski3da23ca2008-03-14 12:52:59 +010014171 SND_PCI_QUIRK(0x1043, 0x8290, "ASUS P5GC-MX", ALC662_3ST_6ch_DIG),
Kailang Yang291702f2007-10-16 14:28:03 +020014172 SND_PCI_QUIRK(0x1043, 0x82a1, "ASUS Eeepc", ALC662_ASUS_EEEPC_P701),
Kailang Yang8c427222008-01-10 13:03:59 +010014173 SND_PCI_QUIRK(0x1043, 0x82d1, "ASUS Eeepc EP20", ALC662_ASUS_EEEPC_EP20),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010014174 SND_PCI_QUIRK(0x17aa, 0x101e, "Lenovo", ALC662_LENOVO_101E),
Kailang Yang6dda9f42008-05-27 12:05:31 +020014175 SND_PCI_QUIRK(0x1854, 0x2000, "ASUS H13-2000", ALC663_ASUS_H13),
14176 SND_PCI_QUIRK(0x1854, 0x2001, "ASUS H13-2001", ALC663_ASUS_H13),
14177 SND_PCI_QUIRK(0x1854, 0x2002, "ASUS H13-2002", ALC663_ASUS_H13),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014178 {}
14179};
14180
14181static struct alc_config_preset alc662_presets[] = {
14182 [ALC662_3ST_2ch_DIG] = {
Kailang Yang291702f2007-10-16 14:28:03 +020014183 .mixers = { alc662_3ST_2ch_mixer, alc662_capture_mixer },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014184 .init_verbs = { alc662_init_verbs },
14185 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
14186 .dac_nids = alc662_dac_nids,
14187 .dig_out_nid = ALC662_DIGOUT_NID,
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014188 .dig_in_nid = ALC662_DIGIN_NID,
14189 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
14190 .channel_mode = alc662_3ST_2ch_modes,
14191 .input_mux = &alc662_capture_source,
14192 },
14193 [ALC662_3ST_6ch_DIG] = {
Kailang Yang291702f2007-10-16 14:28:03 +020014194 .mixers = { alc662_3ST_6ch_mixer, alc662_chmode_mixer,
14195 alc662_capture_mixer },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014196 .init_verbs = { alc662_init_verbs },
14197 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
14198 .dac_nids = alc662_dac_nids,
14199 .dig_out_nid = ALC662_DIGOUT_NID,
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014200 .dig_in_nid = ALC662_DIGIN_NID,
14201 .num_channel_mode = ARRAY_SIZE(alc662_3ST_6ch_modes),
14202 .channel_mode = alc662_3ST_6ch_modes,
14203 .need_dac_fix = 1,
14204 .input_mux = &alc662_capture_source,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014205 },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014206 [ALC662_3ST_6ch] = {
Kailang Yang291702f2007-10-16 14:28:03 +020014207 .mixers = { alc662_3ST_6ch_mixer, alc662_chmode_mixer,
14208 alc662_capture_mixer },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014209 .init_verbs = { alc662_init_verbs },
14210 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
14211 .dac_nids = alc662_dac_nids,
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014212 .num_channel_mode = ARRAY_SIZE(alc662_3ST_6ch_modes),
14213 .channel_mode = alc662_3ST_6ch_modes,
14214 .need_dac_fix = 1,
14215 .input_mux = &alc662_capture_source,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014216 },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014217 [ALC662_5ST_DIG] = {
Kailang Yang291702f2007-10-16 14:28:03 +020014218 .mixers = { alc662_base_mixer, alc662_chmode_mixer,
14219 alc662_capture_mixer },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014220 .init_verbs = { alc662_init_verbs },
14221 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
14222 .dac_nids = alc662_dac_nids,
14223 .dig_out_nid = ALC662_DIGOUT_NID,
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014224 .dig_in_nid = ALC662_DIGIN_NID,
14225 .num_channel_mode = ARRAY_SIZE(alc662_5stack_modes),
14226 .channel_mode = alc662_5stack_modes,
14227 .input_mux = &alc662_capture_source,
14228 },
14229 [ALC662_LENOVO_101E] = {
Kailang Yang291702f2007-10-16 14:28:03 +020014230 .mixers = { alc662_lenovo_101e_mixer, alc662_capture_mixer },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014231 .init_verbs = { alc662_init_verbs, alc662_sue_init_verbs },
14232 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
14233 .dac_nids = alc662_dac_nids,
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014234 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
14235 .channel_mode = alc662_3ST_2ch_modes,
14236 .input_mux = &alc662_lenovo_101e_capture_source,
14237 .unsol_event = alc662_lenovo_101e_unsol_event,
14238 .init_hook = alc662_lenovo_101e_all_automute,
14239 },
Kailang Yang291702f2007-10-16 14:28:03 +020014240 [ALC662_ASUS_EEEPC_P701] = {
14241 .mixers = { alc662_eeepc_p701_mixer, alc662_capture_mixer },
14242 .init_verbs = { alc662_init_verbs,
14243 alc662_eeepc_sue_init_verbs },
14244 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
14245 .dac_nids = alc662_dac_nids,
Kailang Yang291702f2007-10-16 14:28:03 +020014246 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
14247 .channel_mode = alc662_3ST_2ch_modes,
14248 .input_mux = &alc662_eeepc_capture_source,
14249 .unsol_event = alc662_eeepc_unsol_event,
14250 .init_hook = alc662_eeepc_inithook,
14251 },
Kailang Yang8c427222008-01-10 13:03:59 +010014252 [ALC662_ASUS_EEEPC_EP20] = {
14253 .mixers = { alc662_eeepc_ep20_mixer, alc662_capture_mixer,
14254 alc662_chmode_mixer },
14255 .init_verbs = { alc662_init_verbs,
14256 alc662_eeepc_ep20_sue_init_verbs },
14257 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
14258 .dac_nids = alc662_dac_nids,
Kailang Yang8c427222008-01-10 13:03:59 +010014259 .num_channel_mode = ARRAY_SIZE(alc662_3ST_6ch_modes),
14260 .channel_mode = alc662_3ST_6ch_modes,
14261 .input_mux = &alc662_lenovo_101e_capture_source,
14262 .unsol_event = alc662_eeepc_ep20_unsol_event,
14263 .init_hook = alc662_eeepc_ep20_inithook,
14264 },
Kailang Yang6dda9f42008-05-27 12:05:31 +020014265 [ALC663_ASUS_M51VA] = {
14266 .mixers = { alc663_m51va_mixer, alc662_capture_mixer},
14267 .init_verbs = { alc662_init_verbs, alc663_m51va_init_verbs },
14268 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
14269 .dac_nids = alc662_dac_nids,
14270 .dig_out_nid = ALC662_DIGOUT_NID,
14271 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
14272 .channel_mode = alc662_3ST_2ch_modes,
14273 .input_mux = &alc663_m51va_capture_source,
14274 .unsol_event = alc663_m51va_unsol_event,
14275 .init_hook = alc663_m51va_inithook,
14276 },
14277 [ALC663_ASUS_G71V] = {
14278 .mixers = { alc663_g71v_mixer, alc662_capture_mixer},
14279 .init_verbs = { alc662_init_verbs, alc663_g71v_init_verbs },
14280 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
14281 .dac_nids = alc662_dac_nids,
14282 .dig_out_nid = ALC662_DIGOUT_NID,
14283 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
14284 .channel_mode = alc662_3ST_2ch_modes,
14285 .input_mux = &alc662_eeepc_capture_source,
14286 .unsol_event = alc663_g71v_unsol_event,
14287 .init_hook = alc663_g71v_inithook,
14288 },
14289 [ALC663_ASUS_H13] = {
14290 .mixers = { alc663_m51va_mixer, alc662_capture_mixer},
14291 .init_verbs = { alc662_init_verbs, alc663_m51va_init_verbs },
14292 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
14293 .dac_nids = alc662_dac_nids,
14294 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
14295 .channel_mode = alc662_3ST_2ch_modes,
14296 .input_mux = &alc663_m51va_capture_source,
14297 .unsol_event = alc663_m51va_unsol_event,
14298 .init_hook = alc663_m51va_inithook,
14299 },
14300 [ALC663_ASUS_G50V] = {
14301 .mixers = { alc663_g50v_mixer, alc662_capture_mixer},
14302 .init_verbs = { alc662_init_verbs, alc663_g50v_init_verbs },
14303 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
14304 .dac_nids = alc662_dac_nids,
14305 .dig_out_nid = ALC662_DIGOUT_NID,
14306 .num_channel_mode = ARRAY_SIZE(alc662_3ST_6ch_modes),
14307 .channel_mode = alc662_3ST_6ch_modes,
14308 .input_mux = &alc663_capture_source,
14309 .unsol_event = alc663_g50v_unsol_event,
14310 .init_hook = alc663_g50v_inithook,
14311 },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014312};
14313
14314
14315/*
14316 * BIOS auto configuration
14317 */
14318
14319/* add playback controls from the parsed DAC table */
14320static int alc662_auto_create_multi_out_ctls(struct alc_spec *spec,
14321 const struct auto_pin_cfg *cfg)
14322{
14323 char name[32];
14324 static const char *chname[4] = {
14325 "Front", "Surround", NULL /*CLFE*/, "Side"
14326 };
14327 hda_nid_t nid;
14328 int i, err;
14329
14330 for (i = 0; i < cfg->line_outs; i++) {
14331 if (!spec->multiout.dac_nids[i])
14332 continue;
Kailang Yangb60dd392007-09-20 12:50:29 +020014333 nid = alc880_idx_to_dac(i);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014334 if (i == 2) {
14335 /* Center/LFE */
14336 err = add_control(spec, ALC_CTL_WIDGET_VOL,
14337 "Center Playback Volume",
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014338 HDA_COMPOSE_AMP_VAL(nid, 1, 0,
14339 HDA_OUTPUT));
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014340 if (err < 0)
14341 return err;
14342 err = add_control(spec, ALC_CTL_WIDGET_VOL,
14343 "LFE Playback Volume",
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014344 HDA_COMPOSE_AMP_VAL(nid, 2, 0,
14345 HDA_OUTPUT));
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014346 if (err < 0)
14347 return err;
14348 err = add_control(spec, ALC_CTL_BIND_MUTE,
14349 "Center Playback Switch",
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014350 HDA_COMPOSE_AMP_VAL(nid, 1, 2,
14351 HDA_INPUT));
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014352 if (err < 0)
14353 return err;
14354 err = add_control(spec, ALC_CTL_BIND_MUTE,
14355 "LFE Playback Switch",
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014356 HDA_COMPOSE_AMP_VAL(nid, 2, 2,
14357 HDA_INPUT));
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014358 if (err < 0)
14359 return err;
14360 } else {
14361 sprintf(name, "%s Playback Volume", chname[i]);
14362 err = add_control(spec, ALC_CTL_WIDGET_VOL, name,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014363 HDA_COMPOSE_AMP_VAL(nid, 3, 0,
14364 HDA_OUTPUT));
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014365 if (err < 0)
14366 return err;
14367 sprintf(name, "%s Playback Switch", chname[i]);
14368 err = add_control(spec, ALC_CTL_BIND_MUTE, name,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014369 HDA_COMPOSE_AMP_VAL(nid, 3, 2,
14370 HDA_INPUT));
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014371 if (err < 0)
14372 return err;
14373 }
14374 }
14375 return 0;
14376}
14377
14378/* add playback controls for speaker and HP outputs */
14379static int alc662_auto_create_extra_out(struct alc_spec *spec, hda_nid_t pin,
14380 const char *pfx)
14381{
14382 hda_nid_t nid;
14383 int err;
14384 char name[32];
14385
14386 if (!pin)
14387 return 0;
14388
14389 if (alc880_is_fixed_pin(pin)) {
14390 nid = alc880_idx_to_dac(alc880_fixed_pin_idx(pin));
14391 /* printk("DAC nid=%x\n",nid); */
14392 /* specify the DAC as the extra output */
14393 if (!spec->multiout.hp_nid)
14394 spec->multiout.hp_nid = nid;
14395 else
14396 spec->multiout.extra_out_nid[0] = nid;
14397 /* control HP volume/switch on the output mixer amp */
14398 nid = alc880_idx_to_dac(alc880_fixed_pin_idx(pin));
14399 sprintf(name, "%s Playback Volume", pfx);
14400 err = add_control(spec, ALC_CTL_WIDGET_VOL, name,
14401 HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_OUTPUT));
14402 if (err < 0)
14403 return err;
14404 sprintf(name, "%s Playback Switch", pfx);
14405 err = add_control(spec, ALC_CTL_BIND_MUTE, name,
14406 HDA_COMPOSE_AMP_VAL(nid, 3, 2, HDA_INPUT));
14407 if (err < 0)
14408 return err;
14409 } else if (alc880_is_multi_pin(pin)) {
14410 /* set manual connection */
14411 /* we have only a switch on HP-out PIN */
14412 sprintf(name, "%s Playback Switch", pfx);
14413 err = add_control(spec, ALC_CTL_WIDGET_MUTE, name,
14414 HDA_COMPOSE_AMP_VAL(pin, 3, 0, HDA_OUTPUT));
14415 if (err < 0)
14416 return err;
14417 }
14418 return 0;
14419}
14420
14421/* create playback/capture controls for input pins */
14422static int alc662_auto_create_analog_input_ctls(struct alc_spec *spec,
14423 const struct auto_pin_cfg *cfg)
14424{
14425 struct hda_input_mux *imux = &spec->private_imux;
14426 int i, err, idx;
14427
14428 for (i = 0; i < AUTO_PIN_LAST; i++) {
14429 if (alc880_is_input_pin(cfg->input_pins[i])) {
14430 idx = alc880_input_pin_idx(cfg->input_pins[i]);
14431 err = new_analog_input(spec, cfg->input_pins[i],
14432 auto_pin_cfg_labels[i],
14433 idx, 0x0b);
14434 if (err < 0)
14435 return err;
14436 imux->items[imux->num_items].label =
14437 auto_pin_cfg_labels[i];
14438 imux->items[imux->num_items].index =
14439 alc880_input_pin_idx(cfg->input_pins[i]);
14440 imux->num_items++;
14441 }
14442 }
14443 return 0;
14444}
14445
14446static void alc662_auto_set_output_and_unmute(struct hda_codec *codec,
14447 hda_nid_t nid, int pin_type,
14448 int dac_idx)
14449{
Takashi Iwaif6c7e542008-02-12 18:32:23 +010014450 alc_set_pin_output(codec, nid, pin_type);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014451 /* need the manual connection? */
14452 if (alc880_is_multi_pin(nid)) {
14453 struct alc_spec *spec = codec->spec;
14454 int idx = alc880_multi_pin_idx(nid);
14455 snd_hda_codec_write(codec, alc880_idx_to_selector(idx), 0,
14456 AC_VERB_SET_CONNECT_SEL,
14457 alc880_dac_to_idx(spec->multiout.dac_nids[dac_idx]));
14458 }
14459}
14460
14461static void alc662_auto_init_multi_out(struct hda_codec *codec)
14462{
14463 struct alc_spec *spec = codec->spec;
14464 int i;
14465
Kailang Yang8c427222008-01-10 13:03:59 +010014466 alc_subsystem_id(codec, 0x15, 0x1b, 0x14);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014467 for (i = 0; i <= HDA_SIDE; i++) {
14468 hda_nid_t nid = spec->autocfg.line_out_pins[i];
Takashi Iwaibaba8ee2007-04-23 17:17:48 +020014469 int pin_type = get_pin_type(spec->autocfg.line_out_type);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014470 if (nid)
Takashi Iwaibaba8ee2007-04-23 17:17:48 +020014471 alc662_auto_set_output_and_unmute(codec, nid, pin_type,
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014472 i);
14473 }
14474}
14475
14476static void alc662_auto_init_hp_out(struct hda_codec *codec)
14477{
14478 struct alc_spec *spec = codec->spec;
14479 hda_nid_t pin;
14480
14481 pin = spec->autocfg.hp_pins[0];
14482 if (pin) /* connect to front */
14483 /* use dac 0 */
14484 alc662_auto_set_output_and_unmute(codec, pin, PIN_HP, 0);
Takashi Iwaif6c7e542008-02-12 18:32:23 +010014485 pin = spec->autocfg.speaker_pins[0];
14486 if (pin)
14487 alc662_auto_set_output_and_unmute(codec, pin, PIN_OUT, 0);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014488}
14489
14490#define alc662_is_input_pin(nid) alc880_is_input_pin(nid)
14491#define ALC662_PIN_CD_NID ALC880_PIN_CD_NID
14492
14493static void alc662_auto_init_analog_input(struct hda_codec *codec)
14494{
14495 struct alc_spec *spec = codec->spec;
14496 int i;
14497
14498 for (i = 0; i < AUTO_PIN_LAST; i++) {
14499 hda_nid_t nid = spec->autocfg.input_pins[i];
14500 if (alc662_is_input_pin(nid)) {
14501 snd_hda_codec_write(codec, nid, 0,
14502 AC_VERB_SET_PIN_WIDGET_CONTROL,
14503 (i <= AUTO_PIN_FRONT_MIC ?
14504 PIN_VREF80 : PIN_IN));
14505 if (nid != ALC662_PIN_CD_NID)
14506 snd_hda_codec_write(codec, nid, 0,
14507 AC_VERB_SET_AMP_GAIN_MUTE,
14508 AMP_OUT_MUTE);
14509 }
14510 }
14511}
14512
14513static int alc662_parse_auto_config(struct hda_codec *codec)
14514{
14515 struct alc_spec *spec = codec->spec;
14516 int err;
14517 static hda_nid_t alc662_ignore[] = { 0x1d, 0 };
14518
14519 err = snd_hda_parse_pin_def_config(codec, &spec->autocfg,
14520 alc662_ignore);
14521 if (err < 0)
14522 return err;
14523 if (!spec->autocfg.line_outs)
14524 return 0; /* can't find valid BIOS pin config */
14525
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014526 err = alc880_auto_fill_dac_nids(spec, &spec->autocfg);
14527 if (err < 0)
14528 return err;
14529 err = alc662_auto_create_multi_out_ctls(spec, &spec->autocfg);
14530 if (err < 0)
14531 return err;
14532 err = alc662_auto_create_extra_out(spec,
14533 spec->autocfg.speaker_pins[0],
14534 "Speaker");
14535 if (err < 0)
14536 return err;
14537 err = alc662_auto_create_extra_out(spec, spec->autocfg.hp_pins[0],
14538 "Headphone");
14539 if (err < 0)
14540 return err;
14541 err = alc662_auto_create_analog_input_ctls(spec, &spec->autocfg);
14542 if (err < 0)
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014543 return err;
14544
14545 spec->multiout.max_channels = spec->multiout.num_dacs * 2;
14546
14547 if (spec->autocfg.dig_out_pin)
14548 spec->multiout.dig_out_nid = ALC880_DIGOUT_NID;
14549
14550 if (spec->kctl_alloc)
14551 spec->mixers[spec->num_mixers++] = spec->kctl_alloc;
14552
14553 spec->num_mux_defs = 1;
14554 spec->input_mux = &spec->private_imux;
14555
Takashi Iwai8c872862007-06-19 12:11:16 +020014556 spec->init_verbs[spec->num_init_verbs++] = alc662_auto_init_verbs;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014557 spec->mixers[spec->num_mixers] = alc662_capture_mixer;
14558 spec->num_mixers++;
Takashi Iwai8c872862007-06-19 12:11:16 +020014559 return 1;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014560}
14561
14562/* additional initialization for auto-configuration model */
14563static void alc662_auto_init(struct hda_codec *codec)
14564{
Takashi Iwaif6c7e542008-02-12 18:32:23 +010014565 struct alc_spec *spec = codec->spec;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014566 alc662_auto_init_multi_out(codec);
14567 alc662_auto_init_hp_out(codec);
14568 alc662_auto_init_analog_input(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +010014569 if (spec->unsol_event)
14570 alc_sku_automute(codec);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014571}
14572
14573static int patch_alc662(struct hda_codec *codec)
14574{
14575 struct alc_spec *spec;
14576 int err, board_config;
14577
14578 spec = kzalloc(sizeof(*spec), GFP_KERNEL);
14579 if (!spec)
14580 return -ENOMEM;
14581
14582 codec->spec = spec;
14583
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +020014584 alc_fix_pll_init(codec, 0x20, 0x04, 15);
14585
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014586 board_config = snd_hda_check_board_config(codec, ALC662_MODEL_LAST,
14587 alc662_models,
14588 alc662_cfg_tbl);
14589 if (board_config < 0) {
14590 printk(KERN_INFO "hda_codec: Unknown model for ALC662, "
14591 "trying auto-probe from BIOS...\n");
14592 board_config = ALC662_AUTO;
14593 }
14594
14595 if (board_config == ALC662_AUTO) {
14596 /* automatic parse from the BIOS config */
14597 err = alc662_parse_auto_config(codec);
14598 if (err < 0) {
14599 alc_free(codec);
14600 return err;
Takashi Iwai8c872862007-06-19 12:11:16 +020014601 } else if (!err) {
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014602 printk(KERN_INFO
14603 "hda_codec: Cannot set up configuration "
14604 "from BIOS. Using base mode...\n");
14605 board_config = ALC662_3ST_2ch_DIG;
14606 }
14607 }
14608
14609 if (board_config != ALC662_AUTO)
14610 setup_preset(spec, &alc662_presets[board_config]);
14611
Kailang Yang6dda9f42008-05-27 12:05:31 +020014612 if (codec->vendor_id == 0x10ec0663) {
14613 spec->stream_name_analog = "ALC663 Analog";
14614 spec->stream_name_digital = "ALC663 Digital";
14615 } else {
14616 spec->stream_name_analog = "ALC662 Analog";
14617 spec->stream_name_digital = "ALC662 Digital";
14618 }
14619
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014620 spec->stream_analog_playback = &alc662_pcm_analog_playback;
14621 spec->stream_analog_capture = &alc662_pcm_analog_capture;
14622
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014623 spec->stream_digital_playback = &alc662_pcm_digital_playback;
14624 spec->stream_digital_capture = &alc662_pcm_digital_capture;
14625
Takashi Iwaie1406342008-02-11 18:32:32 +010014626 spec->adc_nids = alc662_adc_nids;
14627 spec->num_adc_nids = ARRAY_SIZE(alc662_adc_nids);
14628 spec->capsrc_nids = alc662_capsrc_nids;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014629
Takashi Iwai2134ea42008-01-10 16:53:55 +010014630 spec->vmaster_nid = 0x02;
14631
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014632 codec->patch_ops = alc_patch_ops;
14633 if (board_config == ALC662_AUTO)
14634 spec->init_hook = alc662_auto_init;
Takashi Iwaicb53c622007-08-10 17:21:45 +020014635#ifdef CONFIG_SND_HDA_POWER_SAVE
14636 if (!spec->loopback.amplist)
14637 spec->loopback.amplist = alc662_loopbacks;
14638#endif
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014639
14640 return 0;
14641}
14642
14643/*
Linus Torvalds1da177e2005-04-16 15:20:36 -070014644 * patch entries
14645 */
14646struct hda_codec_preset snd_hda_preset_realtek[] = {
14647 { .id = 0x10ec0260, .name = "ALC260", .patch = patch_alc260 },
Kailang Yangdf694da2005-12-05 19:42:22 +010014648 { .id = 0x10ec0262, .name = "ALC262", .patch = patch_alc262 },
Kailang Yangf6a92242007-12-13 16:52:54 +010014649 { .id = 0x10ec0267, .name = "ALC267", .patch = patch_alc268 },
Kailang Yanga361d842007-06-05 12:30:55 +020014650 { .id = 0x10ec0268, .name = "ALC268", .patch = patch_alc268 },
Kailang Yangf6a92242007-12-13 16:52:54 +010014651 { .id = 0x10ec0269, .name = "ALC269", .patch = patch_alc269 },
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010014652 { .id = 0x10ec0861, .rev = 0x100340, .name = "ALC660",
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014653 .patch = patch_alc861 },
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010014654 { .id = 0x10ec0660, .name = "ALC660-VD", .patch = patch_alc861vd },
14655 { .id = 0x10ec0861, .name = "ALC861", .patch = patch_alc861 },
14656 { .id = 0x10ec0862, .name = "ALC861-VD", .patch = patch_alc861vd },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020014657 { .id = 0x10ec0662, .rev = 0x100002, .name = "ALC662 rev2",
14658 .patch = patch_alc883 },
14659 { .id = 0x10ec0662, .rev = 0x100101, .name = "ALC662 rev1",
14660 .patch = patch_alc662 },
Kailang Yang6dda9f42008-05-27 12:05:31 +020014661 { .id = 0x10ec0663, .name = "ALC663", .patch = patch_alc662 },
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010014662 { .id = 0x10ec0880, .name = "ALC880", .patch = patch_alc880 },
Linus Torvalds1da177e2005-04-16 15:20:36 -070014663 { .id = 0x10ec0882, .name = "ALC882", .patch = patch_alc882 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +020014664 { .id = 0x10ec0883, .name = "ALC883", .patch = patch_alc883 },
Takashi Iwaicb308f92008-04-16 14:13:29 +020014665 { .id = 0x10ec0885, .rev = 0x100103, .name = "ALC889A",
Takashi Iwai7943a8a2008-04-16 17:29:09 +020014666 .patch = patch_alc882 }, /* should be patch_alc883() in future */
Kailang Yangdf694da2005-12-05 19:42:22 +010014667 { .id = 0x10ec0885, .name = "ALC885", .patch = patch_alc882 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +020014668 { .id = 0x10ec0888, .name = "ALC888", .patch = patch_alc883 },
Kailang Yangf6a92242007-12-13 16:52:54 +010014669 { .id = 0x10ec0889, .name = "ALC889", .patch = patch_alc883 },
Linus Torvalds1da177e2005-04-16 15:20:36 -070014670 {} /* terminator */
14671};