blob: d29fa18232ad692ec5e1a6051e9272b916406bbb [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"
Kusanagi Kouichi680cd532009-02-05 00:00:58 +090033#include "hda_beep.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 Yang3f878302008-08-26 13:02:23 +020075 ALC260_HP_DC7600,
Kailang Yangdf694da2005-12-05 19:42:22 +010076 ALC260_HP_3013,
77 ALC260_FUJITSU_S702X,
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +010078 ALC260_ACER,
Kailang Yangbc9f98a2007-04-12 13:06:07 +020079 ALC260_WILL,
80 ALC260_REPLACER_672V,
Michael Schwingencc959482009-02-22 18:58:45 +010081 ALC260_FAVORIT100,
Jonathan Woithe7cf51e42006-02-09 12:01:26 +010082#ifdef CONFIG_SND_DEBUG
83 ALC260_TEST,
84#endif
Kailang Yangdf694da2005-12-05 19:42:22 +010085 ALC260_AUTO,
Takashi Iwai16ded522005-06-10 19:58:24 +020086 ALC260_MODEL_LAST /* last tag */
Linus Torvalds1da177e2005-04-16 15:20:36 -070087};
88
Kailang Yangdf694da2005-12-05 19:42:22 +010089/* ALC262 models */
90enum {
91 ALC262_BASIC,
Kailang Yangccc656c2006-10-17 12:32:26 +020092 ALC262_HIPPO,
93 ALC262_HIPPO_1,
Takashi Iwai834be882006-03-01 14:16:17 +010094 ALC262_FUJITSU,
Takashi Iwai9c7f8522006-06-28 15:08:22 +020095 ALC262_HP_BPC,
Kailang Yangcd7509a2007-01-26 18:33:17 +010096 ALC262_HP_BPC_D7000_WL,
97 ALC262_HP_BPC_D7000_WF,
Kailang Yang66d2a9d2007-11-14 12:06:21 +010098 ALC262_HP_TC_T5735,
Kailang Yang8c427222008-01-10 13:03:59 +010099 ALC262_HP_RP5700,
Takashi Iwai304dcaa2006-07-25 14:51:16 +0200100 ALC262_BENQ_ED8,
Kailang Yang272a5272007-05-14 11:00:38 +0200101 ALC262_SONY_ASSAMD,
Kailang Yang83c34212007-07-05 11:43:05 +0200102 ALC262_BENQ_T31,
Tobin Davisf651b502007-10-26 12:40:47 +0200103 ALC262_ULTRA,
Jiang zhe0e31daf2008-03-20 12:12:39 +0100104 ALC262_LENOVO_3000,
Pascal Terjane8f9ae22008-08-04 14:36:05 +0200105 ALC262_NEC,
Kailang Yang4e555fe2008-08-26 13:05:55 +0200106 ALC262_TOSHIBA_S06,
Hiroshi Miura9f99a632008-08-28 16:09:06 +0200107 ALC262_TOSHIBA_RX1,
Tony Vroonba340e82009-02-02 19:01:30 +0000108 ALC262_TYAN,
Kailang Yangdf694da2005-12-05 19:42:22 +0100109 ALC262_AUTO,
110 ALC262_MODEL_LAST /* last tag */
111};
112
Kailang Yanga361d842007-06-05 12:30:55 +0200113/* ALC268 models */
114enum {
Herton Ronaldo Krzesinskieb5a6622008-04-14 13:46:28 +0200115 ALC267_QUANTA_IL1,
Kailang Yanga361d842007-06-05 12:30:55 +0200116 ALC268_3ST,
Kailang Yangd1a991a2007-08-15 16:21:59 +0200117 ALC268_TOSHIBA,
Takashi Iwaid2738092007-08-16 14:59:45 +0200118 ALC268_ACER,
Takashi Iwaic238b4f2008-11-05 14:57:20 +0100119 ALC268_ACER_DMIC,
Kailang Yang8ef355d2008-08-26 13:10:22 +0200120 ALC268_ACER_ASPIRE_ONE,
Takashi Iwai3866f0b2008-01-15 12:37:42 +0100121 ALC268_DELL,
Mirco Tischlerf12462c2008-02-04 12:33:59 +0100122 ALC268_ZEPTO,
Jonathan Woithe86c53bd2008-01-08 12:33:19 +0100123#ifdef CONFIG_SND_DEBUG
124 ALC268_TEST,
125#endif
Kailang Yanga361d842007-06-05 12:30:55 +0200126 ALC268_AUTO,
127 ALC268_MODEL_LAST /* last tag */
128};
129
Kailang Yangf6a92242007-12-13 16:52:54 +0100130/* ALC269 models */
131enum {
132 ALC269_BASIC,
Kailang Yang60db6b52008-08-26 13:13:00 +0200133 ALC269_QUANTA_FL1,
Kailang Yangf53281e2008-07-18 12:36:43 +0200134 ALC269_ASUS_EEEPC_P703,
135 ALC269_ASUS_EEEPC_P901,
Takashi Iwai26f5df22008-11-03 17:39:46 +0100136 ALC269_FUJITSU,
Tony Vroon64154832008-11-06 15:08:49 +0000137 ALC269_LIFEBOOK,
Kailang Yangf6a92242007-12-13 16:52:54 +0100138 ALC269_AUTO,
139 ALC269_MODEL_LAST /* last tag */
140};
141
Kailang Yangdf694da2005-12-05 19:42:22 +0100142/* ALC861 models */
143enum {
144 ALC861_3ST,
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200145 ALC660_3ST,
Kailang Yangdf694da2005-12-05 19:42:22 +0100146 ALC861_3ST_DIG,
147 ALC861_6ST_DIG,
Takashi Iwai22309c32006-08-09 16:57:28 +0200148 ALC861_UNIWILL_M31,
Tobin Davisa53d1ae2006-10-17 12:00:28 +0200149 ALC861_TOSHIBA,
Mariusz Domanski7cdbff92006-10-23 13:42:56 +0200150 ALC861_ASUS,
Takashi Iwai56bb0ca2006-11-22 11:52:52 +0100151 ALC861_ASUS_LAPTOP,
Kailang Yangdf694da2005-12-05 19:42:22 +0100152 ALC861_AUTO,
153 ALC861_MODEL_LAST,
154};
155
Jakub Schmidtkef32610e2007-02-02 18:17:27 +0100156/* ALC861-VD models */
157enum {
158 ALC660VD_3ST,
Mike Crash6963f842007-06-25 12:12:51 +0200159 ALC660VD_3ST_DIG,
Takashi Iwai13c94742008-11-05 08:06:08 +0100160 ALC660VD_ASUS_V1S,
Jakub Schmidtkef32610e2007-02-02 18:17:27 +0100161 ALC861VD_3ST,
162 ALC861VD_3ST_DIG,
163 ALC861VD_6ST_DIG,
Kailang Yangbdd148a2007-05-08 15:19:08 +0200164 ALC861VD_LENOVO,
Kailang Yang272a5272007-05-14 11:00:38 +0200165 ALC861VD_DALLAS,
Kailang Yangd1a991a2007-08-15 16:21:59 +0200166 ALC861VD_HP,
Jakub Schmidtkef32610e2007-02-02 18:17:27 +0100167 ALC861VD_AUTO,
168 ALC861VD_MODEL_LAST,
169};
170
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200171/* ALC662 models */
172enum {
173 ALC662_3ST_2ch_DIG,
174 ALC662_3ST_6ch_DIG,
175 ALC662_3ST_6ch,
176 ALC662_5ST_DIG,
177 ALC662_LENOVO_101E,
Kailang Yang291702f2007-10-16 14:28:03 +0200178 ALC662_ASUS_EEEPC_P701,
Kailang Yang8c427222008-01-10 13:03:59 +0100179 ALC662_ASUS_EEEPC_EP20,
Kailang Yang6dda9f42008-05-27 12:05:31 +0200180 ALC663_ASUS_M51VA,
181 ALC663_ASUS_G71V,
182 ALC663_ASUS_H13,
183 ALC663_ASUS_G50V,
Kailang Yangf1d4e282008-08-26 14:03:29 +0200184 ALC662_ECS,
185 ALC663_ASUS_MODE1,
186 ALC662_ASUS_MODE2,
187 ALC663_ASUS_MODE3,
188 ALC663_ASUS_MODE4,
189 ALC663_ASUS_MODE5,
190 ALC663_ASUS_MODE6,
Kailang Yang622e84c2009-04-21 07:39:04 +0200191 ALC272_DELL,
192 ALC272_DELL_ZM1,
Chris Pockelé9541ba12009-05-12 08:08:53 +0200193 ALC272_SAMSUNG_NC10,
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200194 ALC662_AUTO,
195 ALC662_MODEL_LAST,
196};
197
Kailang Yangdf694da2005-12-05 19:42:22 +0100198/* ALC882 models */
199enum {
200 ALC882_3ST_DIG,
201 ALC882_6ST_DIG,
Takashi Iwai4b146cb2006-07-28 14:42:36 +0200202 ALC882_ARIMA,
Kailang Yangbdd148a2007-05-08 15:19:08 +0200203 ALC882_W2JC,
Kailang Yang272a5272007-05-14 11:00:38 +0200204 ALC882_TARGA,
205 ALC882_ASUS_A7J,
Takashi Iwai914759b2007-09-06 14:52:04 +0200206 ALC882_ASUS_A7M,
Tobin Davis9102cd12006-12-15 10:02:12 +0100207 ALC885_MACPRO,
Takashi Iwai87350ad2007-08-16 18:19:38 +0200208 ALC885_MBP3,
Kacper Szczesniak41d55452009-05-07 12:47:43 +0200209 ALC885_MB5,
Nicola Fagnanic54728d2007-07-19 23:28:52 +0200210 ALC885_IMAC24,
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200211 ALC883_3ST_2ch_DIG,
212 ALC883_3ST_6ch_DIG,
213 ALC883_3ST_6ch,
214 ALC883_6ST_DIG,
Kailang Yangccc656c2006-10-17 12:32:26 +0200215 ALC883_TARGA_DIG,
216 ALC883_TARGA_2ch_DIG,
David Heidelberger64a8be72009-06-08 16:15:18 +0200217 ALC883_TARGA_8ch_DIG,
Vladimir Avdoninbab282b2006-08-22 13:31:58 +0200218 ALC883_ACER,
Tobin Davis2880a862007-08-07 11:50:26 +0200219 ALC883_ACER_ASPIRE,
Vincent Petry5b2d1ec2008-11-18 22:21:57 +0800220 ALC888_ACER_ASPIRE_4930G,
Tony Vroond2fd4b02009-06-21 00:40:10 +0100221 ALC888_ACER_ASPIRE_6530G,
Hector Martin3b315d72009-06-02 10:54:19 +0200222 ALC888_ACER_ASPIRE_8930G,
Denis Kuplyakovfc86f952009-08-25 18:15:59 +0200223 ALC888_ACER_ASPIRE_7730G,
Tobin Davisc07584c2006-10-13 12:32:16 +0200224 ALC883_MEDION,
Kailang Yangea1fb292008-08-26 12:58:38 +0200225 ALC883_MEDION_MD2,
Andrew L. Neporadab373bde2006-11-07 11:37:08 +0100226 ALC883_LAPTOP_EAPD,
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200227 ALC883_LENOVO_101E_2ch,
Kailang Yang272a5272007-05-14 11:00:38 +0200228 ALC883_LENOVO_NB0763,
Kailang Yang189609a2007-08-20 11:31:23 +0200229 ALC888_LENOVO_MS7195_DIG,
Kailang Yange2757d52008-08-26 13:17:46 +0200230 ALC888_LENOVO_SKY,
Kailang Yangea1fb292008-08-26 12:58:38 +0200231 ALC883_HAIER_W66,
Claudio Matsuoka4723c022007-07-13 14:36:19 +0200232 ALC888_3ST_HP,
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +0100233 ALC888_6ST_DELL,
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +0100234 ALC883_MITAC,
ddiaz@cenditel.gob.vea65cc602009-09-05 16:28:06 -0430235 ALC883_CLEVO_M540R,
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +0100236 ALC883_CLEVO_M720,
Jiang zhefb97dc62008-03-06 11:07:11 +0100237 ALC883_FUJITSU_PI2515,
Vincent Petryef8ef5f2008-11-23 11:31:41 +0800238 ALC888_FUJITSU_XA3530,
Jiang zhe17bba1b2008-06-04 12:11:07 +0200239 ALC883_3ST_6ch_INTEL,
Jaroslav Kysela87a8c372009-07-23 10:58:29 +0200240 ALC889A_INTEL,
241 ALC889_INTEL,
Kailang Yange2757d52008-08-26 13:17:46 +0200242 ALC888_ASUS_M90V,
243 ALC888_ASUS_EEE1601,
Torben Schulzeb4c41d2009-05-18 15:02:35 +0200244 ALC889A_MB31,
Wu Fengguang3ab90932008-11-17 09:51:09 +0100245 ALC1200_ASUS_P5Q,
Guido Günther3e1647c2009-06-05 00:47:26 +0200246 ALC883_SONY_VAIO_TT,
Takashi Iwai49535502009-06-30 15:28:30 +0200247 ALC882_AUTO,
248 ALC882_MODEL_LAST,
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200249};
250
Kailang Yangdf694da2005-12-05 19:42:22 +0100251/* for GPIO Poll */
252#define GPIO_MASK 0x03
253
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200254/* extra amp-initialization sequence types */
255enum {
256 ALC_INIT_NONE,
257 ALC_INIT_DEFAULT,
258 ALC_INIT_GPIO1,
259 ALC_INIT_GPIO2,
260 ALC_INIT_GPIO3,
261};
262
Takashi Iwai6c819492009-08-10 18:47:44 +0200263struct alc_mic_route {
264 hda_nid_t pin;
265 unsigned char mux_idx;
266 unsigned char amix_idx;
267};
268
269#define MUX_IDX_UNDEF ((unsigned char)-1)
270
Linus Torvalds1da177e2005-04-16 15:20:36 -0700271struct alc_spec {
272 /* codec parameterization */
Kailang Yangdf694da2005-12-05 19:42:22 +0100273 struct snd_kcontrol_new *mixers[5]; /* mixer arrays */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700274 unsigned int num_mixers;
Takashi Iwaif9e336f2008-10-31 16:37:07 +0100275 struct snd_kcontrol_new *cap_mixer; /* capture mixer */
Takashi Iwai45bdd1c2009-02-06 16:11:25 +0100276 unsigned int beep_amp; /* beep amp value, set via set_beep_amp() */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700277
Takashi Iwai2d9c6482009-10-13 08:06:55 +0200278 const struct hda_verb *init_verbs[10]; /* initialization verbs
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200279 * don't forget NULL
280 * termination!
Takashi Iwaie9edcee2005-06-13 14:16:38 +0200281 */
282 unsigned int num_init_verbs;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700283
Takashi Iwaiaa563af2009-07-31 10:05:11 +0200284 char stream_name_analog[32]; /* analog PCM stream */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700285 struct hda_pcm_stream *stream_analog_playback;
286 struct hda_pcm_stream *stream_analog_capture;
Takashi Iwai63300792008-01-24 15:31:36 +0100287 struct hda_pcm_stream *stream_analog_alt_playback;
288 struct hda_pcm_stream *stream_analog_alt_capture;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700289
Takashi Iwaiaa563af2009-07-31 10:05:11 +0200290 char stream_name_digital[32]; /* digital PCM stream */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700291 struct hda_pcm_stream *stream_digital_playback;
292 struct hda_pcm_stream *stream_digital_capture;
293
294 /* playback */
Takashi Iwai16ded522005-06-10 19:58:24 +0200295 struct hda_multi_out multiout; /* playback set-up
296 * max_channels, dacs must be set
297 * dig_out_nid and hp_nid are optional
298 */
Takashi Iwai63300792008-01-24 15:31:36 +0100299 hda_nid_t alt_dac_nid;
Takashi Iwai6a05ac42009-02-13 11:19:09 +0100300 hda_nid_t slave_dig_outs[3]; /* optional - for auto-parsing */
Takashi Iwai8c441982009-01-20 18:30:20 +0100301 int dig_out_type;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700302
303 /* capture */
304 unsigned int num_adc_nids;
305 hda_nid_t *adc_nids;
Takashi Iwaie1406342008-02-11 18:32:32 +0100306 hda_nid_t *capsrc_nids;
Takashi Iwai16ded522005-06-10 19:58:24 +0200307 hda_nid_t dig_in_nid; /* digital-in NID; optional */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700308
309 /* capture source */
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200310 unsigned int num_mux_defs;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700311 const struct hda_input_mux *input_mux;
312 unsigned int cur_mux[3];
Takashi Iwai6c819492009-08-10 18:47:44 +0200313 struct alc_mic_route ext_mic;
314 struct alc_mic_route int_mic;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700315
316 /* channel model */
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +0100317 const struct hda_channel_mode *channel_mode;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700318 int num_channel_mode;
Takashi Iwai4e195a72006-07-28 14:47:34 +0200319 int need_dac_fix;
Hector Martin3b315d72009-06-02 10:54:19 +0200320 int const_channel_count;
321 int ext_channel_count;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700322
323 /* PCM information */
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100324 struct hda_pcm pcm_rec[3]; /* used in alc_build_pcms() */
Takashi Iwai41e41f12005-06-08 14:48:49 +0200325
Takashi Iwaie9edcee2005-06-13 14:16:38 +0200326 /* dynamic controls, init_verbs and input_mux */
327 struct auto_pin_cfg autocfg;
Takashi Iwai603c4012008-07-30 15:01:44 +0200328 struct snd_array kctls;
Herton Ronaldo Krzesinski61b9b9b2009-01-28 09:16:33 -0200329 struct hda_input_mux private_imux[3];
Takashi Iwai41923e42007-10-22 17:20:10 +0200330 hda_nid_t private_dac_nids[AUTO_CFG_MAX_OUTS];
Takashi Iwai49535502009-06-30 15:28:30 +0200331 hda_nid_t private_adc_nids[AUTO_CFG_MAX_OUTS];
332 hda_nid_t private_capsrc_nids[AUTO_CFG_MAX_OUTS];
Takashi Iwai834be882006-03-01 14:16:17 +0100333
Takashi Iwaiae6b8132006-03-03 16:47:17 +0100334 /* hooks */
335 void (*init_hook)(struct hda_codec *codec);
336 void (*unsol_event)(struct hda_codec *codec, unsigned int res);
337
Takashi Iwai834be882006-03-01 14:16:17 +0100338 /* for pin sensing */
339 unsigned int sense_updated: 1;
340 unsigned int jack_present: 1;
Takashi Iwaibec15c32008-01-28 18:16:30 +0100341 unsigned int master_sw: 1;
Takashi Iwai6c819492009-08-10 18:47:44 +0200342 unsigned int auto_mic:1;
Takashi Iwaicb53c622007-08-10 17:21:45 +0200343
Takashi Iwaie64f14f2009-01-20 18:32:55 +0100344 /* other flags */
345 unsigned int no_analog :1; /* digital I/O only */
Takashi Iwai4a79ba32009-04-22 16:31:35 +0200346 int init_amp;
Takashi Iwaie64f14f2009-01-20 18:32:55 +0100347
Takashi Iwai2134ea42008-01-10 16:53:55 +0100348 /* for virtual master */
349 hda_nid_t vmaster_nid;
Takashi Iwaicb53c622007-08-10 17:21:45 +0200350#ifdef CONFIG_SND_HDA_POWER_SAVE
351 struct hda_loopback_check loopback;
352#endif
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +0200353
354 /* for PLL fix */
355 hda_nid_t pll_nid;
356 unsigned int pll_coef_idx, pll_coef_bit;
Kailang Yangdf694da2005-12-05 19:42:22 +0100357};
358
359/*
360 * configuration template - to be copied to the spec instance
361 */
362struct alc_config_preset {
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200363 struct snd_kcontrol_new *mixers[5]; /* should be identical size
364 * with spec
365 */
Takashi Iwaif9e336f2008-10-31 16:37:07 +0100366 struct snd_kcontrol_new *cap_mixer; /* capture mixer */
Kailang Yangdf694da2005-12-05 19:42:22 +0100367 const struct hda_verb *init_verbs[5];
368 unsigned int num_dacs;
369 hda_nid_t *dac_nids;
370 hda_nid_t dig_out_nid; /* optional */
371 hda_nid_t hp_nid; /* optional */
Wu Fengguangb25c9da2009-02-06 15:02:27 +0800372 hda_nid_t *slave_dig_outs;
Kailang Yangdf694da2005-12-05 19:42:22 +0100373 unsigned int num_adc_nids;
374 hda_nid_t *adc_nids;
Takashi Iwaie1406342008-02-11 18:32:32 +0100375 hda_nid_t *capsrc_nids;
Kailang Yangdf694da2005-12-05 19:42:22 +0100376 hda_nid_t dig_in_nid;
377 unsigned int num_channel_mode;
378 const struct hda_channel_mode *channel_mode;
Takashi Iwai4e195a72006-07-28 14:47:34 +0200379 int need_dac_fix;
Hector Martin3b315d72009-06-02 10:54:19 +0200380 int const_channel_count;
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200381 unsigned int num_mux_defs;
Kailang Yangdf694da2005-12-05 19:42:22 +0100382 const struct hda_input_mux *input_mux;
Takashi Iwaiae6b8132006-03-03 16:47:17 +0100383 void (*unsol_event)(struct hda_codec *, unsigned int);
Takashi Iwaie9c364c2009-08-11 17:16:13 +0200384 void (*setup)(struct hda_codec *);
Takashi Iwaiae6b8132006-03-03 16:47:17 +0100385 void (*init_hook)(struct hda_codec *);
Takashi Iwaicb53c622007-08-10 17:21:45 +0200386#ifdef CONFIG_SND_HDA_POWER_SAVE
387 struct hda_amp_list *loopbacks;
388#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700389};
390
Linus Torvalds1da177e2005-04-16 15:20:36 -0700391
392/*
393 * input MUX handling
394 */
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200395static int alc_mux_enum_info(struct snd_kcontrol *kcontrol,
396 struct snd_ctl_elem_info *uinfo)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700397{
398 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
399 struct alc_spec *spec = codec->spec;
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200400 unsigned int mux_idx = snd_ctl_get_ioffidx(kcontrol, &uinfo->id);
401 if (mux_idx >= spec->num_mux_defs)
402 mux_idx = 0;
403 return snd_hda_input_mux_info(&spec->input_mux[mux_idx], uinfo);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700404}
405
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200406static int alc_mux_enum_get(struct snd_kcontrol *kcontrol,
407 struct snd_ctl_elem_value *ucontrol)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700408{
409 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
410 struct alc_spec *spec = codec->spec;
411 unsigned int adc_idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
412
413 ucontrol->value.enumerated.item[0] = spec->cur_mux[adc_idx];
414 return 0;
415}
416
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200417static int alc_mux_enum_put(struct snd_kcontrol *kcontrol,
418 struct snd_ctl_elem_value *ucontrol)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700419{
420 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
421 struct alc_spec *spec = codec->spec;
Takashi Iwaicd896c32008-11-18 12:36:33 +0100422 const struct hda_input_mux *imux;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700423 unsigned int adc_idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
Takashi Iwaicd896c32008-11-18 12:36:33 +0100424 unsigned int mux_idx;
Takashi Iwaie1406342008-02-11 18:32:32 +0100425 hda_nid_t nid = spec->capsrc_nids ?
426 spec->capsrc_nids[adc_idx] : spec->adc_nids[adc_idx];
Takashi Iwai0169b6b2009-06-22 10:50:19 +0200427 unsigned int type;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700428
Takashi Iwaicd896c32008-11-18 12:36:33 +0100429 mux_idx = adc_idx >= spec->num_mux_defs ? 0 : adc_idx;
430 imux = &spec->input_mux[mux_idx];
431
Takashi Iwaia22d5432009-07-27 12:54:26 +0200432 type = get_wcaps_type(get_wcaps(codec, nid));
Takashi Iwai0169b6b2009-06-22 10:50:19 +0200433 if (type == AC_WID_AUD_MIX) {
Takashi Iwai54cbc9a2008-10-31 15:24:04 +0100434 /* Matrix-mixer style (e.g. ALC882) */
435 unsigned int *cur_val = &spec->cur_mux[adc_idx];
436 unsigned int i, idx;
437
438 idx = ucontrol->value.enumerated.item[0];
439 if (idx >= imux->num_items)
440 idx = imux->num_items - 1;
441 if (*cur_val == idx)
442 return 0;
443 for (i = 0; i < imux->num_items; i++) {
444 unsigned int v = (i == idx) ? 0 : HDA_AMP_MUTE;
445 snd_hda_codec_amp_stereo(codec, nid, HDA_INPUT,
446 imux->items[i].index,
447 HDA_AMP_MUTE, v);
448 }
449 *cur_val = idx;
450 return 1;
451 } else {
452 /* MUX style (e.g. ALC880) */
Takashi Iwaicd896c32008-11-18 12:36:33 +0100453 return snd_hda_input_mux_put(codec, imux, ucontrol, nid,
Takashi Iwai54cbc9a2008-10-31 15:24:04 +0100454 &spec->cur_mux[adc_idx]);
455 }
456}
Takashi Iwaie9edcee2005-06-13 14:16:38 +0200457
Linus Torvalds1da177e2005-04-16 15:20:36 -0700458/*
459 * channel mode setting
460 */
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200461static int alc_ch_mode_info(struct snd_kcontrol *kcontrol,
462 struct snd_ctl_elem_info *uinfo)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700463{
464 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
465 struct alc_spec *spec = codec->spec;
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +0100466 return snd_hda_ch_mode_info(codec, uinfo, spec->channel_mode,
467 spec->num_channel_mode);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700468}
469
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200470static int alc_ch_mode_get(struct snd_kcontrol *kcontrol,
471 struct snd_ctl_elem_value *ucontrol)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700472{
473 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
474 struct alc_spec *spec = codec->spec;
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +0100475 return snd_hda_ch_mode_get(codec, ucontrol, spec->channel_mode,
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200476 spec->num_channel_mode,
Hector Martin3b315d72009-06-02 10:54:19 +0200477 spec->ext_channel_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700478}
479
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200480static int alc_ch_mode_put(struct snd_kcontrol *kcontrol,
481 struct snd_ctl_elem_value *ucontrol)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700482{
483 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
484 struct alc_spec *spec = codec->spec;
Takashi Iwai4e195a72006-07-28 14:47:34 +0200485 int err = snd_hda_ch_mode_put(codec, ucontrol, spec->channel_mode,
486 spec->num_channel_mode,
Hector Martin3b315d72009-06-02 10:54:19 +0200487 &spec->ext_channel_count);
488 if (err >= 0 && !spec->const_channel_count) {
489 spec->multiout.max_channels = spec->ext_channel_count;
490 if (spec->need_dac_fix)
491 spec->multiout.num_dacs = spec->multiout.max_channels / 2;
492 }
Takashi Iwai4e195a72006-07-28 14:47:34 +0200493 return err;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700494}
495
Linus Torvalds1da177e2005-04-16 15:20:36 -0700496/*
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100497 * Control the mode of pin widget settings via the mixer. "pc" is used
Kailang Yangea1fb292008-08-26 12:58:38 +0200498 * instead of "%" to avoid consequences of accidently treating the % as
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100499 * being part of a format specifier. Maximum allowed length of a value is
500 * 63 characters plus NULL terminator.
Jonathan Woithe7cf51e42006-02-09 12:01:26 +0100501 *
502 * Note: some retasking pin complexes seem to ignore requests for input
503 * states other than HiZ (eg: PIN_VREFxx) and revert to HiZ if any of these
504 * are requested. Therefore order this list so that this behaviour will not
505 * cause problems when mixer clients move through the enum sequentially.
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200506 * NIDs 0x0f and 0x10 have been observed to have this behaviour as of
507 * March 2006.
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200508 */
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100509static char *alc_pin_mode_names[] = {
Jonathan Woithe7cf51e42006-02-09 12:01:26 +0100510 "Mic 50pc bias", "Mic 80pc bias",
511 "Line in", "Line out", "Headphone out",
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100512};
513static unsigned char alc_pin_mode_values[] = {
Jonathan Woithe7cf51e42006-02-09 12:01:26 +0100514 PIN_VREF50, PIN_VREF80, PIN_IN, PIN_OUT, PIN_HP,
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100515};
516/* The control can present all 5 options, or it can limit the options based
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200517 * in the pin being assumed to be exclusively an input or an output pin. In
518 * addition, "input" pins may or may not process the mic bias option
519 * depending on actual widget capability (NIDs 0x0f and 0x10 don't seem to
520 * accept requests for bias as of chip versions up to March 2006) and/or
521 * wiring in the computer.
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100522 */
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200523#define ALC_PIN_DIR_IN 0x00
524#define ALC_PIN_DIR_OUT 0x01
525#define ALC_PIN_DIR_INOUT 0x02
526#define ALC_PIN_DIR_IN_NOMICBIAS 0x03
527#define ALC_PIN_DIR_INOUT_NOMICBIAS 0x04
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100528
Kailang Yangea1fb292008-08-26 12:58:38 +0200529/* Info about the pin modes supported by the different pin direction modes.
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100530 * For each direction the minimum and maximum values are given.
531 */
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200532static signed char alc_pin_mode_dir_info[5][2] = {
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100533 { 0, 2 }, /* ALC_PIN_DIR_IN */
534 { 3, 4 }, /* ALC_PIN_DIR_OUT */
535 { 0, 4 }, /* ALC_PIN_DIR_INOUT */
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200536 { 2, 2 }, /* ALC_PIN_DIR_IN_NOMICBIAS */
537 { 2, 4 }, /* ALC_PIN_DIR_INOUT_NOMICBIAS */
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100538};
539#define alc_pin_mode_min(_dir) (alc_pin_mode_dir_info[_dir][0])
540#define alc_pin_mode_max(_dir) (alc_pin_mode_dir_info[_dir][1])
541#define alc_pin_mode_n_items(_dir) \
542 (alc_pin_mode_max(_dir)-alc_pin_mode_min(_dir)+1)
543
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200544static int alc_pin_mode_info(struct snd_kcontrol *kcontrol,
545 struct snd_ctl_elem_info *uinfo)
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200546{
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100547 unsigned int item_num = uinfo->value.enumerated.item;
548 unsigned char dir = (kcontrol->private_value >> 16) & 0xff;
549
550 uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200551 uinfo->count = 1;
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100552 uinfo->value.enumerated.items = alc_pin_mode_n_items(dir);
553
554 if (item_num<alc_pin_mode_min(dir) || item_num>alc_pin_mode_max(dir))
555 item_num = alc_pin_mode_min(dir);
556 strcpy(uinfo->value.enumerated.name, alc_pin_mode_names[item_num]);
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200557 return 0;
558}
559
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200560static int alc_pin_mode_get(struct snd_kcontrol *kcontrol,
561 struct snd_ctl_elem_value *ucontrol)
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200562{
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100563 unsigned int i;
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200564 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
565 hda_nid_t nid = kcontrol->private_value & 0xffff;
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100566 unsigned char dir = (kcontrol->private_value >> 16) & 0xff;
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200567 long *valp = ucontrol->value.integer.value;
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200568 unsigned int pinctl = snd_hda_codec_read(codec, nid, 0,
569 AC_VERB_GET_PIN_WIDGET_CONTROL,
570 0x00);
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200571
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100572 /* Find enumerated value for current pinctl setting */
573 i = alc_pin_mode_min(dir);
Roel Kluin4b35d2c2009-08-02 13:30:45 +0200574 while (i <= alc_pin_mode_max(dir) && alc_pin_mode_values[i] != pinctl)
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100575 i++;
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200576 *valp = i <= alc_pin_mode_max(dir) ? i: alc_pin_mode_min(dir);
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100577 return 0;
578}
579
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200580static int alc_pin_mode_put(struct snd_kcontrol *kcontrol,
581 struct snd_ctl_elem_value *ucontrol)
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100582{
583 signed int change;
584 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
585 hda_nid_t nid = kcontrol->private_value & 0xffff;
586 unsigned char dir = (kcontrol->private_value >> 16) & 0xff;
587 long val = *ucontrol->value.integer.value;
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200588 unsigned int pinctl = snd_hda_codec_read(codec, nid, 0,
589 AC_VERB_GET_PIN_WIDGET_CONTROL,
590 0x00);
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100591
Takashi Iwaif12ab1e2007-04-12 15:51:47 +0200592 if (val < alc_pin_mode_min(dir) || val > alc_pin_mode_max(dir))
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100593 val = alc_pin_mode_min(dir);
594
595 change = pinctl != alc_pin_mode_values[val];
Jonathan Woithecdcd9262006-02-28 11:36:42 +0100596 if (change) {
597 /* Set pin mode to that requested */
Takashi Iwai82beb8f2007-08-10 17:09:26 +0200598 snd_hda_codec_write_cache(codec, nid, 0,
599 AC_VERB_SET_PIN_WIDGET_CONTROL,
600 alc_pin_mode_values[val]);
Jonathan Woithecdcd9262006-02-28 11:36:42 +0100601
Kailang Yangea1fb292008-08-26 12:58:38 +0200602 /* Also enable the retasking pin's input/output as required
Jonathan Woithecdcd9262006-02-28 11:36:42 +0100603 * for the requested pin mode. Enum values of 2 or less are
604 * input modes.
605 *
606 * Dynamically switching the input/output buffers probably
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200607 * reduces noise slightly (particularly on input) so we'll
608 * do it. However, having both input and output buffers
609 * enabled simultaneously doesn't seem to be problematic if
610 * this turns out to be necessary in the future.
Jonathan Woithecdcd9262006-02-28 11:36:42 +0100611 */
612 if (val <= 2) {
Takashi Iwai47fd8302007-08-10 17:11:07 +0200613 snd_hda_codec_amp_stereo(codec, nid, HDA_OUTPUT, 0,
614 HDA_AMP_MUTE, HDA_AMP_MUTE);
615 snd_hda_codec_amp_stereo(codec, nid, HDA_INPUT, 0,
616 HDA_AMP_MUTE, 0);
Jonathan Woithecdcd9262006-02-28 11:36:42 +0100617 } else {
Takashi Iwai47fd8302007-08-10 17:11:07 +0200618 snd_hda_codec_amp_stereo(codec, nid, HDA_INPUT, 0,
619 HDA_AMP_MUTE, HDA_AMP_MUTE);
620 snd_hda_codec_amp_stereo(codec, nid, HDA_OUTPUT, 0,
621 HDA_AMP_MUTE, 0);
Jonathan Woithecdcd9262006-02-28 11:36:42 +0100622 }
623 }
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200624 return change;
625}
626
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100627#define ALC_PIN_MODE(xname, nid, dir) \
Jonathan Woithea9430dd2005-09-16 19:12:48 +0200628 { .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, .index = 0, \
Jonathan Woithe4c5186e2006-02-09 11:53:48 +0100629 .info = alc_pin_mode_info, \
630 .get = alc_pin_mode_get, \
631 .put = alc_pin_mode_put, \
632 .private_value = nid | (dir<<16) }
Kailang Yangdf694da2005-12-05 19:42:22 +0100633
Jonathan Woithe5c8f8582006-02-28 11:43:27 +0100634/* A switch control for ALC260 GPIO pins. Multiple GPIOs can be ganged
635 * together using a mask with more than one bit set. This control is
636 * currently used only by the ALC260 test model. At this stage they are not
637 * needed for any "production" models.
638 */
639#ifdef CONFIG_SND_DEBUG
Takashi Iwaia5ce8892007-07-23 15:42:26 +0200640#define alc_gpio_data_info snd_ctl_boolean_mono_info
Takashi Iwaif12ab1e2007-04-12 15:51:47 +0200641
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200642static int alc_gpio_data_get(struct snd_kcontrol *kcontrol,
643 struct snd_ctl_elem_value *ucontrol)
Jonathan Woithe5c8f8582006-02-28 11:43:27 +0100644{
645 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
646 hda_nid_t nid = kcontrol->private_value & 0xffff;
647 unsigned char mask = (kcontrol->private_value >> 16) & 0xff;
648 long *valp = ucontrol->value.integer.value;
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200649 unsigned int val = snd_hda_codec_read(codec, nid, 0,
650 AC_VERB_GET_GPIO_DATA, 0x00);
Jonathan Woithe5c8f8582006-02-28 11:43:27 +0100651
652 *valp = (val & mask) != 0;
653 return 0;
654}
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200655static int alc_gpio_data_put(struct snd_kcontrol *kcontrol,
656 struct snd_ctl_elem_value *ucontrol)
Jonathan Woithe5c8f8582006-02-28 11:43:27 +0100657{
658 signed int change;
659 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
660 hda_nid_t nid = kcontrol->private_value & 0xffff;
661 unsigned char mask = (kcontrol->private_value >> 16) & 0xff;
662 long val = *ucontrol->value.integer.value;
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200663 unsigned int gpio_data = snd_hda_codec_read(codec, nid, 0,
664 AC_VERB_GET_GPIO_DATA,
665 0x00);
Jonathan Woithe5c8f8582006-02-28 11:43:27 +0100666
667 /* Set/unset the masked GPIO bit(s) as needed */
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200668 change = (val == 0 ? 0 : mask) != (gpio_data & mask);
669 if (val == 0)
Jonathan Woithe5c8f8582006-02-28 11:43:27 +0100670 gpio_data &= ~mask;
671 else
672 gpio_data |= mask;
Takashi Iwai82beb8f2007-08-10 17:09:26 +0200673 snd_hda_codec_write_cache(codec, nid, 0,
674 AC_VERB_SET_GPIO_DATA, gpio_data);
Jonathan Woithe5c8f8582006-02-28 11:43:27 +0100675
676 return change;
677}
678#define ALC_GPIO_DATA_SWITCH(xname, nid, mask) \
679 { .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, .index = 0, \
680 .info = alc_gpio_data_info, \
681 .get = alc_gpio_data_get, \
682 .put = alc_gpio_data_put, \
683 .private_value = nid | (mask<<16) }
684#endif /* CONFIG_SND_DEBUG */
685
Jonathan Woithe92621f12006-02-28 11:47:47 +0100686/* A switch control to allow the enabling of the digital IO pins on the
687 * ALC260. This is incredibly simplistic; the intention of this control is
688 * to provide something in the test model allowing digital outputs to be
689 * identified if present. If models are found which can utilise these
690 * outputs a more complete mixer control can be devised for those models if
691 * necessary.
692 */
693#ifdef CONFIG_SND_DEBUG
Takashi Iwaia5ce8892007-07-23 15:42:26 +0200694#define alc_spdif_ctrl_info snd_ctl_boolean_mono_info
Takashi Iwaif12ab1e2007-04-12 15:51:47 +0200695
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200696static int alc_spdif_ctrl_get(struct snd_kcontrol *kcontrol,
697 struct snd_ctl_elem_value *ucontrol)
Jonathan Woithe92621f12006-02-28 11:47:47 +0100698{
699 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
700 hda_nid_t nid = kcontrol->private_value & 0xffff;
701 unsigned char mask = (kcontrol->private_value >> 16) & 0xff;
702 long *valp = ucontrol->value.integer.value;
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200703 unsigned int val = snd_hda_codec_read(codec, nid, 0,
Andrew Paprocki3982d172007-12-19 12:13:44 +0100704 AC_VERB_GET_DIGI_CONVERT_1, 0x00);
Jonathan Woithe92621f12006-02-28 11:47:47 +0100705
706 *valp = (val & mask) != 0;
707 return 0;
708}
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200709static int alc_spdif_ctrl_put(struct snd_kcontrol *kcontrol,
710 struct snd_ctl_elem_value *ucontrol)
Jonathan Woithe92621f12006-02-28 11:47:47 +0100711{
712 signed int change;
713 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
714 hda_nid_t nid = kcontrol->private_value & 0xffff;
715 unsigned char mask = (kcontrol->private_value >> 16) & 0xff;
716 long val = *ucontrol->value.integer.value;
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200717 unsigned int ctrl_data = snd_hda_codec_read(codec, nid, 0,
Andrew Paprocki3982d172007-12-19 12:13:44 +0100718 AC_VERB_GET_DIGI_CONVERT_1,
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200719 0x00);
Jonathan Woithe92621f12006-02-28 11:47:47 +0100720
721 /* Set/unset the masked control bit(s) as needed */
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200722 change = (val == 0 ? 0 : mask) != (ctrl_data & mask);
Jonathan Woithe92621f12006-02-28 11:47:47 +0100723 if (val==0)
724 ctrl_data &= ~mask;
725 else
726 ctrl_data |= mask;
Takashi Iwai82beb8f2007-08-10 17:09:26 +0200727 snd_hda_codec_write_cache(codec, nid, 0, AC_VERB_SET_DIGI_CONVERT_1,
728 ctrl_data);
Jonathan Woithe92621f12006-02-28 11:47:47 +0100729
730 return change;
731}
732#define ALC_SPDIF_CTRL_SWITCH(xname, nid, mask) \
733 { .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, .index = 0, \
734 .info = alc_spdif_ctrl_info, \
735 .get = alc_spdif_ctrl_get, \
736 .put = alc_spdif_ctrl_put, \
737 .private_value = nid | (mask<<16) }
738#endif /* CONFIG_SND_DEBUG */
739
Jonathan Woithef8225f62008-01-08 12:16:54 +0100740/* A switch control to allow the enabling EAPD digital outputs on the ALC26x.
741 * Again, this is only used in the ALC26x test models to help identify when
742 * the EAPD line must be asserted for features to work.
743 */
744#ifdef CONFIG_SND_DEBUG
745#define alc_eapd_ctrl_info snd_ctl_boolean_mono_info
746
747static int alc_eapd_ctrl_get(struct snd_kcontrol *kcontrol,
748 struct snd_ctl_elem_value *ucontrol)
749{
750 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
751 hda_nid_t nid = kcontrol->private_value & 0xffff;
752 unsigned char mask = (kcontrol->private_value >> 16) & 0xff;
753 long *valp = ucontrol->value.integer.value;
754 unsigned int val = snd_hda_codec_read(codec, nid, 0,
755 AC_VERB_GET_EAPD_BTLENABLE, 0x00);
756
757 *valp = (val & mask) != 0;
758 return 0;
759}
760
761static int alc_eapd_ctrl_put(struct snd_kcontrol *kcontrol,
762 struct snd_ctl_elem_value *ucontrol)
763{
764 int change;
765 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
766 hda_nid_t nid = kcontrol->private_value & 0xffff;
767 unsigned char mask = (kcontrol->private_value >> 16) & 0xff;
768 long val = *ucontrol->value.integer.value;
769 unsigned int ctrl_data = snd_hda_codec_read(codec, nid, 0,
770 AC_VERB_GET_EAPD_BTLENABLE,
771 0x00);
772
773 /* Set/unset the masked control bit(s) as needed */
774 change = (!val ? 0 : mask) != (ctrl_data & mask);
775 if (!val)
776 ctrl_data &= ~mask;
777 else
778 ctrl_data |= mask;
779 snd_hda_codec_write_cache(codec, nid, 0, AC_VERB_SET_EAPD_BTLENABLE,
780 ctrl_data);
781
782 return change;
783}
784
785#define ALC_EAPD_CTRL_SWITCH(xname, nid, mask) \
786 { .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, .index = 0, \
787 .info = alc_eapd_ctrl_info, \
788 .get = alc_eapd_ctrl_get, \
789 .put = alc_eapd_ctrl_put, \
790 .private_value = nid | (mask<<16) }
791#endif /* CONFIG_SND_DEBUG */
792
Kailang Yangdf694da2005-12-05 19:42:22 +0100793/*
Takashi Iwai23f0c042009-02-26 13:03:58 +0100794 * set up the input pin config (depending on the given auto-pin type)
795 */
796static void alc_set_input_pin(struct hda_codec *codec, hda_nid_t nid,
797 int auto_pin_type)
798{
799 unsigned int val = PIN_IN;
800
801 if (auto_pin_type <= AUTO_PIN_FRONT_MIC) {
802 unsigned int pincap;
Takashi Iwai1327a322009-03-23 13:07:47 +0100803 pincap = snd_hda_query_pin_caps(codec, nid);
Takashi Iwai23f0c042009-02-26 13:03:58 +0100804 pincap = (pincap & AC_PINCAP_VREF) >> AC_PINCAP_VREF_SHIFT;
805 if (pincap & AC_PINCAP_VREF_80)
806 val = PIN_VREF80;
Takashi Iwai461c6c32009-05-25 08:06:02 +0200807 else if (pincap & AC_PINCAP_VREF_50)
808 val = PIN_VREF50;
809 else if (pincap & AC_PINCAP_VREF_100)
810 val = PIN_VREF100;
811 else if (pincap & AC_PINCAP_VREF_GRD)
812 val = PIN_VREFGRD;
Takashi Iwai23f0c042009-02-26 13:03:58 +0100813 }
814 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_PIN_WIDGET_CONTROL, val);
815}
816
817/*
Takashi Iwaid88897e2008-10-31 15:01:37 +0100818 */
819static void add_mixer(struct alc_spec *spec, struct snd_kcontrol_new *mix)
820{
821 if (snd_BUG_ON(spec->num_mixers >= ARRAY_SIZE(spec->mixers)))
822 return;
823 spec->mixers[spec->num_mixers++] = mix;
824}
825
826static void add_verb(struct alc_spec *spec, const struct hda_verb *verb)
827{
828 if (snd_BUG_ON(spec->num_init_verbs >= ARRAY_SIZE(spec->init_verbs)))
829 return;
830 spec->init_verbs[spec->num_init_verbs++] = verb;
831}
832
Takashi Iwaidaead532008-11-28 12:55:36 +0100833#ifdef CONFIG_PROC_FS
834/*
835 * hook for proc
836 */
837static void print_realtek_coef(struct snd_info_buffer *buffer,
838 struct hda_codec *codec, hda_nid_t nid)
839{
840 int coeff;
841
842 if (nid != 0x20)
843 return;
844 coeff = snd_hda_codec_read(codec, nid, 0, AC_VERB_GET_PROC_COEF, 0);
845 snd_iprintf(buffer, " Processing Coefficient: 0x%02x\n", coeff);
846 coeff = snd_hda_codec_read(codec, nid, 0,
847 AC_VERB_GET_COEF_INDEX, 0);
848 snd_iprintf(buffer, " Coefficient Index: 0x%02x\n", coeff);
849}
850#else
851#define print_realtek_coef NULL
852#endif
853
Takashi Iwaid88897e2008-10-31 15:01:37 +0100854/*
Kailang Yangdf694da2005-12-05 19:42:22 +0100855 * set up from the preset table
856 */
Takashi Iwaie9c364c2009-08-11 17:16:13 +0200857static void setup_preset(struct hda_codec *codec,
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200858 const struct alc_config_preset *preset)
Kailang Yangdf694da2005-12-05 19:42:22 +0100859{
Takashi Iwaie9c364c2009-08-11 17:16:13 +0200860 struct alc_spec *spec = codec->spec;
Kailang Yangdf694da2005-12-05 19:42:22 +0100861 int i;
862
863 for (i = 0; i < ARRAY_SIZE(preset->mixers) && preset->mixers[i]; i++)
Takashi Iwaid88897e2008-10-31 15:01:37 +0100864 add_mixer(spec, preset->mixers[i]);
Takashi Iwaif9e336f2008-10-31 16:37:07 +0100865 spec->cap_mixer = preset->cap_mixer;
Takashi Iwai9c7f8522006-06-28 15:08:22 +0200866 for (i = 0; i < ARRAY_SIZE(preset->init_verbs) && preset->init_verbs[i];
867 i++)
Takashi Iwaid88897e2008-10-31 15:01:37 +0100868 add_verb(spec, preset->init_verbs[i]);
Kailang Yangea1fb292008-08-26 12:58:38 +0200869
Kailang Yangdf694da2005-12-05 19:42:22 +0100870 spec->channel_mode = preset->channel_mode;
871 spec->num_channel_mode = preset->num_channel_mode;
Takashi Iwai4e195a72006-07-28 14:47:34 +0200872 spec->need_dac_fix = preset->need_dac_fix;
Hector Martin3b315d72009-06-02 10:54:19 +0200873 spec->const_channel_count = preset->const_channel_count;
Kailang Yangdf694da2005-12-05 19:42:22 +0100874
Hector Martin3b315d72009-06-02 10:54:19 +0200875 if (preset->const_channel_count)
876 spec->multiout.max_channels = preset->const_channel_count;
877 else
878 spec->multiout.max_channels = spec->channel_mode[0].channels;
879 spec->ext_channel_count = spec->channel_mode[0].channels;
Kailang Yangdf694da2005-12-05 19:42:22 +0100880
881 spec->multiout.num_dacs = preset->num_dacs;
882 spec->multiout.dac_nids = preset->dac_nids;
883 spec->multiout.dig_out_nid = preset->dig_out_nid;
Wu Fengguangb25c9da2009-02-06 15:02:27 +0800884 spec->multiout.slave_dig_outs = preset->slave_dig_outs;
Kailang Yangdf694da2005-12-05 19:42:22 +0100885 spec->multiout.hp_nid = preset->hp_nid;
Kailang Yangea1fb292008-08-26 12:58:38 +0200886
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200887 spec->num_mux_defs = preset->num_mux_defs;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +0200888 if (!spec->num_mux_defs)
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +0200889 spec->num_mux_defs = 1;
Kailang Yangdf694da2005-12-05 19:42:22 +0100890 spec->input_mux = preset->input_mux;
891
892 spec->num_adc_nids = preset->num_adc_nids;
893 spec->adc_nids = preset->adc_nids;
Takashi Iwaie1406342008-02-11 18:32:32 +0100894 spec->capsrc_nids = preset->capsrc_nids;
Kailang Yangdf694da2005-12-05 19:42:22 +0100895 spec->dig_in_nid = preset->dig_in_nid;
Takashi Iwaiae6b8132006-03-03 16:47:17 +0100896
897 spec->unsol_event = preset->unsol_event;
898 spec->init_hook = preset->init_hook;
Takashi Iwaicb53c622007-08-10 17:21:45 +0200899#ifdef CONFIG_SND_HDA_POWER_SAVE
900 spec->loopback.amplist = preset->loopbacks;
901#endif
Takashi Iwaie9c364c2009-08-11 17:16:13 +0200902
903 if (preset->setup)
904 preset->setup(codec);
Kailang Yangdf694da2005-12-05 19:42:22 +0100905}
906
Kailang Yangbc9f98a2007-04-12 13:06:07 +0200907/* Enable GPIO mask and set output */
908static struct hda_verb alc_gpio1_init_verbs[] = {
909 {0x01, AC_VERB_SET_GPIO_MASK, 0x01},
910 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x01},
911 {0x01, AC_VERB_SET_GPIO_DATA, 0x01},
912 { }
913};
914
915static struct hda_verb alc_gpio2_init_verbs[] = {
916 {0x01, AC_VERB_SET_GPIO_MASK, 0x02},
917 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x02},
918 {0x01, AC_VERB_SET_GPIO_DATA, 0x02},
919 { }
920};
921
Kailang Yangbdd148a2007-05-08 15:19:08 +0200922static struct hda_verb alc_gpio3_init_verbs[] = {
923 {0x01, AC_VERB_SET_GPIO_MASK, 0x03},
924 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x03},
925 {0x01, AC_VERB_SET_GPIO_DATA, 0x03},
926 { }
927};
928
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +0200929/*
930 * Fix hardware PLL issue
931 * On some codecs, the analog PLL gating control must be off while
932 * the default value is 1.
933 */
934static void alc_fix_pll(struct hda_codec *codec)
935{
936 struct alc_spec *spec = codec->spec;
937 unsigned int val;
938
939 if (!spec->pll_nid)
940 return;
941 snd_hda_codec_write(codec, spec->pll_nid, 0, AC_VERB_SET_COEF_INDEX,
942 spec->pll_coef_idx);
943 val = snd_hda_codec_read(codec, spec->pll_nid, 0,
944 AC_VERB_GET_PROC_COEF, 0);
945 snd_hda_codec_write(codec, spec->pll_nid, 0, AC_VERB_SET_COEF_INDEX,
946 spec->pll_coef_idx);
947 snd_hda_codec_write(codec, spec->pll_nid, 0, AC_VERB_SET_PROC_COEF,
948 val & ~(1 << spec->pll_coef_bit));
949}
950
951static void alc_fix_pll_init(struct hda_codec *codec, hda_nid_t nid,
952 unsigned int coef_idx, unsigned int coef_bit)
953{
954 struct alc_spec *spec = codec->spec;
955 spec->pll_nid = nid;
956 spec->pll_coef_idx = coef_idx;
957 spec->pll_coef_bit = coef_bit;
958 alc_fix_pll(codec);
959}
960
Takashi Iwaia9fd4f32009-05-08 15:57:59 +0200961static void alc_automute_pin(struct hda_codec *codec)
Kailang Yangc9b58002007-10-16 14:30:01 +0200962{
963 struct alc_spec *spec = codec->spec;
Takashi Iwaia9fd4f32009-05-08 15:57:59 +0200964 unsigned int nid = spec->autocfg.hp_pins[0];
965 int i;
Kailang Yangc9b58002007-10-16 14:30:01 +0200966
Takashi Iwaiad87c642009-11-02 14:23:15 +0100967 if (!nid)
968 return;
Wu Fengguang864f92b2009-11-18 12:38:02 +0800969 spec->jack_present = snd_hda_jack_detect(codec, nid);
Takashi Iwaia9fd4f32009-05-08 15:57:59 +0200970 for (i = 0; i < ARRAY_SIZE(spec->autocfg.speaker_pins); i++) {
971 nid = spec->autocfg.speaker_pins[i];
972 if (!nid)
973 break;
974 snd_hda_codec_write(codec, nid, 0,
975 AC_VERB_SET_PIN_WIDGET_CONTROL,
976 spec->jack_present ? 0 : PIN_OUT);
977 }
Kailang Yangc9b58002007-10-16 14:30:01 +0200978}
979
Takashi Iwai6c819492009-08-10 18:47:44 +0200980static int get_connection_index(struct hda_codec *codec, hda_nid_t mux,
981 hda_nid_t nid)
982{
983 hda_nid_t conn[HDA_MAX_NUM_INPUTS];
984 int i, nums;
985
986 nums = snd_hda_get_connections(codec, mux, conn, ARRAY_SIZE(conn));
987 for (i = 0; i < nums; i++)
988 if (conn[i] == nid)
989 return i;
990 return -1;
991}
992
Kailang Yang7fb0d782008-10-15 11:12:35 +0200993static void alc_mic_automute(struct hda_codec *codec)
994{
995 struct alc_spec *spec = codec->spec;
Takashi Iwai6c819492009-08-10 18:47:44 +0200996 struct alc_mic_route *dead, *alive;
997 unsigned int present, type;
998 hda_nid_t cap_nid;
Kailang Yang7fb0d782008-10-15 11:12:35 +0200999
Takashi Iwaib59bdf32009-08-11 09:47:30 +02001000 if (!spec->auto_mic)
1001 return;
Takashi Iwai6c819492009-08-10 18:47:44 +02001002 if (!spec->int_mic.pin || !spec->ext_mic.pin)
1003 return;
1004 if (snd_BUG_ON(!spec->adc_nids))
1005 return;
1006
1007 cap_nid = spec->capsrc_nids ? spec->capsrc_nids[0] : spec->adc_nids[0];
1008
Wu Fengguang864f92b2009-11-18 12:38:02 +08001009 present = snd_hda_jack_detect(codec, spec->ext_mic.pin);
Takashi Iwai6c819492009-08-10 18:47:44 +02001010 if (present) {
1011 alive = &spec->ext_mic;
1012 dead = &spec->int_mic;
1013 } else {
1014 alive = &spec->int_mic;
1015 dead = &spec->ext_mic;
1016 }
1017
Takashi Iwai6c819492009-08-10 18:47:44 +02001018 type = get_wcaps_type(get_wcaps(codec, cap_nid));
1019 if (type == AC_WID_AUD_MIX) {
1020 /* Matrix-mixer style (e.g. ALC882) */
1021 snd_hda_codec_amp_stereo(codec, cap_nid, HDA_INPUT,
1022 alive->mux_idx,
1023 HDA_AMP_MUTE, 0);
1024 snd_hda_codec_amp_stereo(codec, cap_nid, HDA_INPUT,
1025 dead->mux_idx,
1026 HDA_AMP_MUTE, HDA_AMP_MUTE);
1027 } else {
1028 /* MUX style (e.g. ALC880) */
1029 snd_hda_codec_write_cache(codec, cap_nid, 0,
1030 AC_VERB_SET_CONNECT_SEL,
1031 alive->mux_idx);
1032 }
1033
1034 /* FIXME: analog mixer */
Kailang Yang7fb0d782008-10-15 11:12:35 +02001035}
1036
Kailang Yangc9b58002007-10-16 14:30:01 +02001037/* unsolicited event for HP jack sensing */
1038static void alc_sku_unsol_event(struct hda_codec *codec, unsigned int res)
1039{
1040 if (codec->vendor_id == 0x10ec0880)
1041 res >>= 28;
1042 else
1043 res >>= 26;
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02001044 switch (res) {
1045 case ALC880_HP_EVENT:
1046 alc_automute_pin(codec);
1047 break;
1048 case ALC880_MIC_EVENT:
Kailang Yang7fb0d782008-10-15 11:12:35 +02001049 alc_mic_automute(codec);
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02001050 break;
1051 }
Kailang Yang7fb0d782008-10-15 11:12:35 +02001052}
1053
1054static void alc_inithook(struct hda_codec *codec)
1055{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02001056 alc_automute_pin(codec);
Kailang Yang7fb0d782008-10-15 11:12:35 +02001057 alc_mic_automute(codec);
Kailang Yangc9b58002007-10-16 14:30:01 +02001058}
1059
Kailang Yangf9423e72008-05-27 12:32:25 +02001060/* additional initialization for ALC888 variants */
1061static void alc888_coef_init(struct hda_codec *codec)
1062{
1063 unsigned int tmp;
1064
1065 snd_hda_codec_write(codec, 0x20, 0, AC_VERB_SET_COEF_INDEX, 0);
1066 tmp = snd_hda_codec_read(codec, 0x20, 0, AC_VERB_GET_PROC_COEF, 0);
1067 snd_hda_codec_write(codec, 0x20, 0, AC_VERB_SET_COEF_INDEX, 7);
Takashi Iwai37db6232009-03-05 09:40:16 +01001068 if ((tmp & 0xf0) == 0x20)
Kailang Yangf9423e72008-05-27 12:32:25 +02001069 /* alc888S-VC */
1070 snd_hda_codec_read(codec, 0x20, 0,
1071 AC_VERB_SET_PROC_COEF, 0x830);
1072 else
1073 /* alc888-VB */
1074 snd_hda_codec_read(codec, 0x20, 0,
1075 AC_VERB_SET_PROC_COEF, 0x3030);
1076}
1077
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02001078static void alc889_coef_init(struct hda_codec *codec)
1079{
1080 unsigned int tmp;
1081
1082 snd_hda_codec_write(codec, 0x20, 0, AC_VERB_SET_COEF_INDEX, 7);
1083 tmp = snd_hda_codec_read(codec, 0x20, 0, AC_VERB_GET_PROC_COEF, 0);
1084 snd_hda_codec_write(codec, 0x20, 0, AC_VERB_SET_COEF_INDEX, 7);
1085 snd_hda_codec_write(codec, 0x20, 0, AC_VERB_SET_PROC_COEF, tmp|0x2010);
1086}
1087
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001088static void alc_auto_init_amp(struct hda_codec *codec, int type)
Kailang Yangbc9f98a2007-04-12 13:06:07 +02001089{
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001090 unsigned int tmp;
Kailang Yangbc9f98a2007-04-12 13:06:07 +02001091
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001092 switch (type) {
1093 case ALC_INIT_GPIO1:
Kailang Yangbc9f98a2007-04-12 13:06:07 +02001094 snd_hda_sequence_write(codec, alc_gpio1_init_verbs);
1095 break;
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001096 case ALC_INIT_GPIO2:
Kailang Yangbc9f98a2007-04-12 13:06:07 +02001097 snd_hda_sequence_write(codec, alc_gpio2_init_verbs);
1098 break;
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001099 case ALC_INIT_GPIO3:
Kailang Yangbdd148a2007-05-08 15:19:08 +02001100 snd_hda_sequence_write(codec, alc_gpio3_init_verbs);
1101 break;
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001102 case ALC_INIT_DEFAULT:
Kailang Yangbdd148a2007-05-08 15:19:08 +02001103 switch (codec->vendor_id) {
Kailang Yangc9b58002007-10-16 14:30:01 +02001104 case 0x10ec0260:
1105 snd_hda_codec_write(codec, 0x0f, 0,
1106 AC_VERB_SET_EAPD_BTLENABLE, 2);
1107 snd_hda_codec_write(codec, 0x10, 0,
1108 AC_VERB_SET_EAPD_BTLENABLE, 2);
1109 break;
1110 case 0x10ec0262:
Kailang Yangbdd148a2007-05-08 15:19:08 +02001111 case 0x10ec0267:
1112 case 0x10ec0268:
Kailang Yangc9b58002007-10-16 14:30:01 +02001113 case 0x10ec0269:
Takashi Iwaic6e8f2d2009-02-06 12:45:52 +01001114 case 0x10ec0272:
Kailang Yangf9423e72008-05-27 12:32:25 +02001115 case 0x10ec0660:
1116 case 0x10ec0662:
1117 case 0x10ec0663:
Kailang Yangc9b58002007-10-16 14:30:01 +02001118 case 0x10ec0862:
Takashi Iwai20a3a052008-05-23 17:52:53 +02001119 case 0x10ec0889:
Kailang Yangbdd148a2007-05-08 15:19:08 +02001120 snd_hda_codec_write(codec, 0x14, 0,
1121 AC_VERB_SET_EAPD_BTLENABLE, 2);
1122 snd_hda_codec_write(codec, 0x15, 0,
1123 AC_VERB_SET_EAPD_BTLENABLE, 2);
Kailang Yangc9b58002007-10-16 14:30:01 +02001124 break;
Kailang Yangbdd148a2007-05-08 15:19:08 +02001125 }
Kailang Yangc9b58002007-10-16 14:30:01 +02001126 switch (codec->vendor_id) {
1127 case 0x10ec0260:
1128 snd_hda_codec_write(codec, 0x1a, 0,
1129 AC_VERB_SET_COEF_INDEX, 7);
1130 tmp = snd_hda_codec_read(codec, 0x1a, 0,
1131 AC_VERB_GET_PROC_COEF, 0);
1132 snd_hda_codec_write(codec, 0x1a, 0,
1133 AC_VERB_SET_COEF_INDEX, 7);
1134 snd_hda_codec_write(codec, 0x1a, 0,
1135 AC_VERB_SET_PROC_COEF,
1136 tmp | 0x2010);
1137 break;
1138 case 0x10ec0262:
1139 case 0x10ec0880:
1140 case 0x10ec0882:
1141 case 0x10ec0883:
1142 case 0x10ec0885:
Takashi Iwai4a5a4c52009-02-06 12:46:59 +01001143 case 0x10ec0887:
Takashi Iwai20a3a052008-05-23 17:52:53 +02001144 case 0x10ec0889:
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02001145 alc889_coef_init(codec);
Kailang Yangc9b58002007-10-16 14:30:01 +02001146 break;
Kailang Yangf9423e72008-05-27 12:32:25 +02001147 case 0x10ec0888:
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001148 alc888_coef_init(codec);
Kailang Yangf9423e72008-05-27 12:32:25 +02001149 break;
Kailang Yangc9b58002007-10-16 14:30:01 +02001150 case 0x10ec0267:
1151 case 0x10ec0268:
1152 snd_hda_codec_write(codec, 0x20, 0,
1153 AC_VERB_SET_COEF_INDEX, 7);
1154 tmp = snd_hda_codec_read(codec, 0x20, 0,
1155 AC_VERB_GET_PROC_COEF, 0);
1156 snd_hda_codec_write(codec, 0x20, 0,
Kailang Yangea1fb292008-08-26 12:58:38 +02001157 AC_VERB_SET_COEF_INDEX, 7);
Kailang Yangc9b58002007-10-16 14:30:01 +02001158 snd_hda_codec_write(codec, 0x20, 0,
1159 AC_VERB_SET_PROC_COEF,
1160 tmp | 0x3000);
1161 break;
Kailang Yangbc9f98a2007-04-12 13:06:07 +02001162 }
Kailang Yangbc9f98a2007-04-12 13:06:07 +02001163 break;
1164 }
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001165}
Kailang Yangea1fb292008-08-26 12:58:38 +02001166
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001167static void alc_init_auto_hp(struct hda_codec *codec)
1168{
1169 struct alc_spec *spec = codec->spec;
1170
1171 if (!spec->autocfg.hp_pins[0])
Kailang Yangc9b58002007-10-16 14:30:01 +02001172 return;
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001173
Kailang Yangc9b58002007-10-16 14:30:01 +02001174 if (!spec->autocfg.speaker_pins[0]) {
Takashi Iwai2a2ed0d2009-04-28 13:01:26 +02001175 if (spec->autocfg.line_out_pins[0] &&
1176 spec->autocfg.line_out_type == AUTO_PIN_SPEAKER_OUT)
Kailang Yangc9b58002007-10-16 14:30:01 +02001177 spec->autocfg.speaker_pins[0] =
Kailang Yang8c427222008-01-10 13:03:59 +01001178 spec->autocfg.line_out_pins[0];
Kailang Yangc9b58002007-10-16 14:30:01 +02001179 else
1180 return;
1181 }
1182
Takashi Iwai2a2ed0d2009-04-28 13:01:26 +02001183 snd_printdd("realtek: Enable HP auto-muting on NID 0x%x\n",
1184 spec->autocfg.hp_pins[0]);
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001185 snd_hda_codec_write_cache(codec, spec->autocfg.hp_pins[0], 0,
1186 AC_VERB_SET_UNSOLICITED_ENABLE,
1187 AC_USRSP_EN | ALC880_HP_EVENT);
1188 spec->unsol_event = alc_sku_unsol_event;
1189}
1190
Takashi Iwai6c819492009-08-10 18:47:44 +02001191static void alc_init_auto_mic(struct hda_codec *codec)
1192{
1193 struct alc_spec *spec = codec->spec;
1194 struct auto_pin_cfg *cfg = &spec->autocfg;
1195 hda_nid_t fixed, ext;
1196 int i;
1197
1198 /* there must be only two mic inputs exclusively */
1199 for (i = AUTO_PIN_LINE; i < AUTO_PIN_LAST; i++)
1200 if (cfg->input_pins[i])
1201 return;
1202
1203 fixed = ext = 0;
1204 for (i = AUTO_PIN_MIC; i <= AUTO_PIN_FRONT_MIC; i++) {
1205 hda_nid_t nid = cfg->input_pins[i];
1206 unsigned int defcfg;
1207 if (!nid)
1208 return;
1209 defcfg = snd_hda_codec_get_pincfg(codec, nid);
1210 switch (get_defcfg_connect(defcfg)) {
1211 case AC_JACK_PORT_FIXED:
1212 if (fixed)
1213 return; /* already occupied */
1214 fixed = nid;
1215 break;
1216 case AC_JACK_PORT_COMPLEX:
1217 if (ext)
1218 return; /* already occupied */
1219 ext = nid;
1220 break;
1221 default:
1222 return; /* invalid entry */
1223 }
1224 }
1225 if (!(get_wcaps(codec, ext) & AC_WCAP_UNSOL_CAP))
1226 return; /* no unsol support */
1227 snd_printdd("realtek: Enable auto-mic switch on NID 0x%x/0x%x\n",
1228 ext, fixed);
1229 spec->ext_mic.pin = ext;
1230 spec->int_mic.pin = fixed;
1231 spec->ext_mic.mux_idx = MUX_IDX_UNDEF; /* set later */
1232 spec->int_mic.mux_idx = MUX_IDX_UNDEF; /* set later */
1233 spec->auto_mic = 1;
1234 snd_hda_codec_write_cache(codec, spec->ext_mic.pin, 0,
1235 AC_VERB_SET_UNSOLICITED_ENABLE,
1236 AC_USRSP_EN | ALC880_MIC_EVENT);
1237 spec->unsol_event = alc_sku_unsol_event;
1238}
1239
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001240/* check subsystem ID and set up device-specific initialization;
1241 * return 1 if initialized, 0 if invalid SSID
1242 */
1243/* 32-bit subsystem ID for BIOS loading in HD Audio codec.
1244 * 31 ~ 16 : Manufacture ID
1245 * 15 ~ 8 : SKU ID
1246 * 7 ~ 0 : Assembly ID
1247 * port-A --> pin 39/41, port-E --> pin 14/15, port-D --> pin 35/36
1248 */
1249static int alc_subsystem_id(struct hda_codec *codec,
1250 hda_nid_t porta, hda_nid_t porte,
1251 hda_nid_t portd)
1252{
1253 unsigned int ass, tmp, i;
1254 unsigned nid;
1255 struct alc_spec *spec = codec->spec;
1256
1257 ass = codec->subsystem_id & 0xffff;
1258 if ((ass != codec->bus->pci->subsystem_device) && (ass & 1))
1259 goto do_sku;
1260
1261 /* invalid SSID, check the special NID pin defcfg instead */
1262 /*
Sasha Alexandrdef319f2009-06-16 16:00:15 -04001263 * 31~30 : port connectivity
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001264 * 29~21 : reserve
1265 * 20 : PCBEEP input
1266 * 19~16 : Check sum (15:1)
1267 * 15~1 : Custom
1268 * 0 : override
1269 */
1270 nid = 0x1d;
1271 if (codec->vendor_id == 0x10ec0260)
1272 nid = 0x17;
1273 ass = snd_hda_codec_get_pincfg(codec, nid);
1274 snd_printd("realtek: No valid SSID, "
1275 "checking pincfg 0x%08x for NID 0x%x\n",
Takashi Iwaicb6605c2009-04-28 13:03:19 +02001276 ass, nid);
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001277 if (!(ass & 1) && !(ass & 0x100000))
1278 return 0;
1279 if ((ass >> 30) != 1) /* no physical connection */
1280 return 0;
1281
1282 /* check sum */
1283 tmp = 0;
1284 for (i = 1; i < 16; i++) {
1285 if ((ass >> i) & 1)
1286 tmp++;
1287 }
1288 if (((ass >> 16) & 0xf) != tmp)
1289 return 0;
1290do_sku:
1291 snd_printd("realtek: Enabling init ASM_ID=0x%04x CODEC_ID=%08x\n",
1292 ass & 0xffff, codec->vendor_id);
1293 /*
1294 * 0 : override
1295 * 1 : Swap Jack
1296 * 2 : 0 --> Desktop, 1 --> Laptop
1297 * 3~5 : External Amplifier control
1298 * 7~6 : Reserved
1299 */
1300 tmp = (ass & 0x38) >> 3; /* external Amp control */
1301 switch (tmp) {
1302 case 1:
1303 spec->init_amp = ALC_INIT_GPIO1;
1304 break;
1305 case 3:
1306 spec->init_amp = ALC_INIT_GPIO2;
1307 break;
1308 case 7:
1309 spec->init_amp = ALC_INIT_GPIO3;
1310 break;
1311 case 5:
1312 spec->init_amp = ALC_INIT_DEFAULT;
1313 break;
1314 }
1315
1316 /* is laptop or Desktop and enable the function "Mute internal speaker
1317 * when the external headphone out jack is plugged"
1318 */
1319 if (!(ass & 0x8000))
1320 return 1;
1321 /*
1322 * 10~8 : Jack location
1323 * 12~11: Headphone out -> 00: PortA, 01: PortE, 02: PortD, 03: Resvered
1324 * 14~13: Resvered
1325 * 15 : 1 --> enable the function "Mute internal speaker
1326 * when the external headphone out jack is plugged"
1327 */
Kailang Yangc9b58002007-10-16 14:30:01 +02001328 if (!spec->autocfg.hp_pins[0]) {
Takashi Iwai01d48252009-10-06 13:21:54 +02001329 hda_nid_t nid;
Kailang Yangc9b58002007-10-16 14:30:01 +02001330 tmp = (ass >> 11) & 0x3; /* HP to chassis */
1331 if (tmp == 0)
Takashi Iwai01d48252009-10-06 13:21:54 +02001332 nid = porta;
Kailang Yangc9b58002007-10-16 14:30:01 +02001333 else if (tmp == 1)
Takashi Iwai01d48252009-10-06 13:21:54 +02001334 nid = porte;
Kailang Yangc9b58002007-10-16 14:30:01 +02001335 else if (tmp == 2)
Takashi Iwai01d48252009-10-06 13:21:54 +02001336 nid = portd;
Kailang Yangc9b58002007-10-16 14:30:01 +02001337 else
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001338 return 1;
Takashi Iwai01d48252009-10-06 13:21:54 +02001339 for (i = 0; i < spec->autocfg.line_outs; i++)
1340 if (spec->autocfg.line_out_pins[i] == nid)
1341 return 1;
1342 spec->autocfg.hp_pins[0] = nid;
Kailang Yangc9b58002007-10-16 14:30:01 +02001343 }
1344
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001345 alc_init_auto_hp(codec);
Takashi Iwai6c819492009-08-10 18:47:44 +02001346 alc_init_auto_mic(codec);
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001347 return 1;
1348}
Kailang Yangea1fb292008-08-26 12:58:38 +02001349
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001350static void alc_ssid_check(struct hda_codec *codec,
1351 hda_nid_t porta, hda_nid_t porte, hda_nid_t portd)
1352{
1353 if (!alc_subsystem_id(codec, porta, porte, portd)) {
1354 struct alc_spec *spec = codec->spec;
1355 snd_printd("realtek: "
1356 "Enable default setup for auto mode as fallback\n");
1357 spec->init_amp = ALC_INIT_DEFAULT;
1358 alc_init_auto_hp(codec);
Takashi Iwai6c819492009-08-10 18:47:44 +02001359 alc_init_auto_mic(codec);
Takashi Iwai4a79ba32009-04-22 16:31:35 +02001360 }
Kailang Yangbc9f98a2007-04-12 13:06:07 +02001361}
1362
Takashi Iwai41e41f12005-06-08 14:48:49 +02001363/*
Takashi Iwaif8f25ba2009-10-06 08:31:29 +02001364 * Fix-up pin default configurations and add default verbs
Takashi Iwaif95474e2007-07-10 00:47:43 +02001365 */
1366
1367struct alc_pincfg {
1368 hda_nid_t nid;
1369 u32 val;
1370};
1371
Takashi Iwaif8f25ba2009-10-06 08:31:29 +02001372struct alc_fixup {
1373 const struct alc_pincfg *pins;
1374 const struct hda_verb *verbs;
1375};
1376
1377static void alc_pick_fixup(struct hda_codec *codec,
Takashi Iwaif95474e2007-07-10 00:47:43 +02001378 const struct snd_pci_quirk *quirk,
Takashi Iwaif8f25ba2009-10-06 08:31:29 +02001379 const struct alc_fixup *fix)
Takashi Iwaif95474e2007-07-10 00:47:43 +02001380{
1381 const struct alc_pincfg *cfg;
1382
1383 quirk = snd_pci_quirk_lookup(codec->bus->pci, quirk);
1384 if (!quirk)
1385 return;
1386
Takashi Iwaif8f25ba2009-10-06 08:31:29 +02001387 fix += quirk->value;
1388 cfg = fix->pins;
1389 if (cfg) {
1390 for (; cfg->nid; cfg++)
1391 snd_hda_codec_set_pincfg(codec, cfg->nid, cfg->val);
1392 }
1393 if (fix->verbs)
1394 add_verb(codec->spec, fix->verbs);
Takashi Iwaif95474e2007-07-10 00:47:43 +02001395}
1396
1397/*
Vincent Petryef8ef5f2008-11-23 11:31:41 +08001398 * ALC888
1399 */
1400
1401/*
1402 * 2ch mode
1403 */
1404static struct hda_verb alc888_4ST_ch2_intel_init[] = {
1405/* Mic-in jack as mic in */
1406 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
1407 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
1408/* Line-in jack as Line in */
1409 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
1410 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
1411/* Line-Out as Front */
1412 { 0x17, AC_VERB_SET_CONNECT_SEL, 0x00},
1413 { } /* end */
1414};
1415
1416/*
1417 * 4ch mode
1418 */
1419static struct hda_verb alc888_4ST_ch4_intel_init[] = {
1420/* Mic-in jack as mic in */
1421 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
1422 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
1423/* Line-in jack as Surround */
1424 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
1425 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
1426/* Line-Out as Front */
1427 { 0x17, AC_VERB_SET_CONNECT_SEL, 0x00},
1428 { } /* end */
1429};
1430
1431/*
1432 * 6ch mode
1433 */
1434static struct hda_verb alc888_4ST_ch6_intel_init[] = {
1435/* Mic-in jack as CLFE */
1436 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
1437 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
1438/* Line-in jack as Surround */
1439 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
1440 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
1441/* Line-Out as CLFE (workaround because Mic-in is not loud enough) */
1442 { 0x17, AC_VERB_SET_CONNECT_SEL, 0x03},
1443 { } /* end */
1444};
1445
1446/*
1447 * 8ch mode
1448 */
1449static struct hda_verb alc888_4ST_ch8_intel_init[] = {
1450/* Mic-in jack as CLFE */
1451 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
1452 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
1453/* Line-in jack as Surround */
1454 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
1455 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
1456/* Line-Out as Side */
1457 { 0x17, AC_VERB_SET_CONNECT_SEL, 0x03},
1458 { } /* end */
1459};
1460
1461static struct hda_channel_mode alc888_4ST_8ch_intel_modes[4] = {
1462 { 2, alc888_4ST_ch2_intel_init },
1463 { 4, alc888_4ST_ch4_intel_init },
1464 { 6, alc888_4ST_ch6_intel_init },
1465 { 8, alc888_4ST_ch8_intel_init },
1466};
1467
1468/*
1469 * ALC888 Fujitsu Siemens Amillo xa3530
1470 */
1471
1472static struct hda_verb alc888_fujitsu_xa3530_verbs[] = {
1473/* Front Mic: set to PIN_IN (empty by default) */
1474 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
1475/* Connect Internal HP to Front */
1476 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1477 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1478 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
1479/* Connect Bass HP to Front */
1480 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1481 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1482 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
1483/* Connect Line-Out side jack (SPDIF) to Side */
1484 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1485 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1486 {0x17, AC_VERB_SET_CONNECT_SEL, 0x03},
1487/* Connect Mic jack to CLFE */
1488 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1489 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1490 {0x18, AC_VERB_SET_CONNECT_SEL, 0x02},
1491/* Connect Line-in jack to Surround */
1492 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1493 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1494 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x01},
1495/* Connect HP out jack to Front */
1496 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1497 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1498 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
1499/* Enable unsolicited event for HP jack and Line-out jack */
1500 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
1501 {0x17, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
1502 {}
1503};
1504
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02001505static void alc_automute_amp(struct hda_codec *codec)
Vincent Petryef8ef5f2008-11-23 11:31:41 +08001506{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02001507 struct alc_spec *spec = codec->spec;
Wu Fengguang864f92b2009-11-18 12:38:02 +08001508 unsigned int mute;
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02001509 hda_nid_t nid;
1510 int i;
1511
1512 spec->jack_present = 0;
1513 for (i = 0; i < ARRAY_SIZE(spec->autocfg.hp_pins); i++) {
1514 nid = spec->autocfg.hp_pins[i];
1515 if (!nid)
1516 break;
Wu Fengguang864f92b2009-11-18 12:38:02 +08001517 if (snd_hda_jack_detect(codec, nid)) {
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02001518 spec->jack_present = 1;
1519 break;
1520 }
1521 }
1522
1523 mute = spec->jack_present ? HDA_AMP_MUTE : 0;
Vincent Petryef8ef5f2008-11-23 11:31:41 +08001524 /* Toggle internal speakers muting */
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02001525 for (i = 0; i < ARRAY_SIZE(spec->autocfg.speaker_pins); i++) {
1526 nid = spec->autocfg.speaker_pins[i];
1527 if (!nid)
1528 break;
1529 snd_hda_codec_amp_stereo(codec, nid, HDA_OUTPUT, 0,
1530 HDA_AMP_MUTE, mute);
1531 }
Vincent Petryef8ef5f2008-11-23 11:31:41 +08001532}
1533
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02001534static void alc_automute_amp_unsol_event(struct hda_codec *codec,
1535 unsigned int res)
Vincent Petryef8ef5f2008-11-23 11:31:41 +08001536{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02001537 if (codec->vendor_id == 0x10ec0880)
1538 res >>= 28;
1539 else
1540 res >>= 26;
1541 if (res == ALC880_HP_EVENT)
1542 alc_automute_amp(codec);
Vincent Petryef8ef5f2008-11-23 11:31:41 +08001543}
1544
Takashi Iwai4f5d17062009-08-11 18:17:46 +02001545static void alc889_automute_setup(struct hda_codec *codec)
Wu Fengguang6732bd02009-07-30 09:19:14 +02001546{
1547 struct alc_spec *spec = codec->spec;
1548
1549 spec->autocfg.hp_pins[0] = 0x15;
1550 spec->autocfg.speaker_pins[0] = 0x14;
1551 spec->autocfg.speaker_pins[1] = 0x16;
1552 spec->autocfg.speaker_pins[2] = 0x17;
1553 spec->autocfg.speaker_pins[3] = 0x19;
1554 spec->autocfg.speaker_pins[4] = 0x1a;
Wu Fengguang6732bd02009-07-30 09:19:14 +02001555}
1556
1557static void alc889_intel_init_hook(struct hda_codec *codec)
1558{
1559 alc889_coef_init(codec);
Takashi Iwai4f5d17062009-08-11 18:17:46 +02001560 alc_automute_amp(codec);
Wu Fengguang6732bd02009-07-30 09:19:14 +02001561}
1562
Takashi Iwai4f5d17062009-08-11 18:17:46 +02001563static void alc888_fujitsu_xa3530_setup(struct hda_codec *codec)
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02001564{
1565 struct alc_spec *spec = codec->spec;
1566
1567 spec->autocfg.hp_pins[0] = 0x17; /* line-out */
1568 spec->autocfg.hp_pins[1] = 0x1b; /* hp */
1569 spec->autocfg.speaker_pins[0] = 0x14; /* speaker */
1570 spec->autocfg.speaker_pins[1] = 0x15; /* bass */
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02001571}
Vincent Petryef8ef5f2008-11-23 11:31:41 +08001572
1573/*
Vincent Petry5b2d1ec2008-11-18 22:21:57 +08001574 * ALC888 Acer Aspire 4930G model
1575 */
1576
1577static struct hda_verb alc888_acer_aspire_4930g_verbs[] = {
1578/* Front Mic: set to PIN_IN (empty by default) */
1579 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
1580/* Unselect Front Mic by default in input mixer 3 */
1581 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0xb)},
Vincent Petryef8ef5f2008-11-23 11:31:41 +08001582/* Enable unsolicited event for HP jack */
Vincent Petry5b2d1ec2008-11-18 22:21:57 +08001583 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
1584/* Connect Internal HP to front */
1585 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1586 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1587 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
1588/* Connect HP out to front */
1589 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1590 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1591 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
1592 { }
1593};
1594
Hector Martin3b315d72009-06-02 10:54:19 +02001595/*
Tony Vroond2fd4b02009-06-21 00:40:10 +01001596 * ALC888 Acer Aspire 6530G model
1597 */
1598
1599static struct hda_verb alc888_acer_aspire_6530g_verbs[] = {
1600/* Bias voltage on for external mic port */
1601 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN | PIN_VREF80},
Emilio López320d5922009-06-25 08:18:44 +02001602/* Front Mic: set to PIN_IN (empty by default) */
1603 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
1604/* Unselect Front Mic by default in input mixer 3 */
1605 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0xb)},
Tony Vroond2fd4b02009-06-21 00:40:10 +01001606/* Enable unsolicited event for HP jack */
1607 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
1608/* Enable speaker output */
1609 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1610 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1611/* Enable headphone output */
1612 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT | PIN_HP},
1613 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1614 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
1615 { }
1616};
1617
1618/*
Hector Martin018df412009-06-04 00:13:40 +02001619 * ALC889 Acer Aspire 8930G model
Hector Martin3b315d72009-06-02 10:54:19 +02001620 */
1621
Hector Martin018df412009-06-04 00:13:40 +02001622static struct hda_verb alc889_acer_aspire_8930g_verbs[] = {
Hector Martin3b315d72009-06-02 10:54:19 +02001623/* Front Mic: set to PIN_IN (empty by default) */
1624 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
1625/* Unselect Front Mic by default in input mixer 3 */
1626 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0xb)},
1627/* Enable unsolicited event for HP jack */
1628 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
1629/* Connect Internal Front to Front */
1630 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1631 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1632 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
1633/* Connect Internal Rear to Rear */
1634 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1635 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1636 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x01},
1637/* Connect Internal CLFE to CLFE */
1638 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1639 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1640 {0x16, AC_VERB_SET_CONNECT_SEL, 0x02},
1641/* Connect HP out to Front */
Hector Martin018df412009-06-04 00:13:40 +02001642 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT | PIN_HP},
Hector Martin3b315d72009-06-02 10:54:19 +02001643 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1644 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
1645/* Enable all DACs */
1646/* DAC DISABLE/MUTE 1? */
1647/* setting bits 1-5 disables DAC nids 0x02-0x06 apparently. Init=0x38 */
1648 {0x20, AC_VERB_SET_COEF_INDEX, 0x03},
1649 {0x20, AC_VERB_SET_PROC_COEF, 0x0000},
1650/* DAC DISABLE/MUTE 2? */
1651/* some bit here disables the other DACs. Init=0x4900 */
1652 {0x20, AC_VERB_SET_COEF_INDEX, 0x08},
1653 {0x20, AC_VERB_SET_PROC_COEF, 0x0000},
1654/* Enable amplifiers */
1655 {0x14, AC_VERB_SET_EAPD_BTLENABLE, 0x02},
1656 {0x15, AC_VERB_SET_EAPD_BTLENABLE, 0x02},
Hector Martin018df412009-06-04 00:13:40 +02001657/* DMIC fix
1658 * This laptop has a stereo digital microphone. The mics are only 1cm apart
1659 * which makes the stereo useless. However, either the mic or the ALC889
1660 * makes the signal become a difference/sum signal instead of standard
1661 * stereo, which is annoying. So instead we flip this bit which makes the
1662 * codec replicate the sum signal to both channels, turning it into a
1663 * normal mono mic.
1664 */
1665/* DMIC_CONTROL? Init value = 0x0001 */
1666 {0x20, AC_VERB_SET_COEF_INDEX, 0x0b},
1667 {0x20, AC_VERB_SET_PROC_COEF, 0x0003},
Hector Martin3b315d72009-06-02 10:54:19 +02001668 { }
1669};
1670
Vincent Petryef8ef5f2008-11-23 11:31:41 +08001671static struct hda_input_mux alc888_2_capture_sources[2] = {
Vincent Petry5b2d1ec2008-11-18 22:21:57 +08001672 /* Front mic only available on one ADC */
1673 {
1674 .num_items = 4,
1675 .items = {
1676 { "Mic", 0x0 },
1677 { "Line", 0x2 },
1678 { "CD", 0x4 },
1679 { "Front Mic", 0xb },
1680 },
1681 },
1682 {
1683 .num_items = 3,
1684 .items = {
1685 { "Mic", 0x0 },
1686 { "Line", 0x2 },
1687 { "CD", 0x4 },
1688 },
1689 }
1690};
1691
Tony Vroond2fd4b02009-06-21 00:40:10 +01001692static struct hda_input_mux alc888_acer_aspire_6530_sources[2] = {
1693 /* Interal mic only available on one ADC */
1694 {
Tony Vroon684a8842009-06-26 09:27:50 +01001695 .num_items = 5,
Tony Vroond2fd4b02009-06-21 00:40:10 +01001696 .items = {
1697 { "Ext Mic", 0x0 },
Tony Vroon684a8842009-06-26 09:27:50 +01001698 { "Line In", 0x2 },
Tony Vroond2fd4b02009-06-21 00:40:10 +01001699 { "CD", 0x4 },
Tony Vroon684a8842009-06-26 09:27:50 +01001700 { "Input Mix", 0xa },
Tony Vroond2fd4b02009-06-21 00:40:10 +01001701 { "Int Mic", 0xb },
1702 },
1703 },
1704 {
Tony Vroon684a8842009-06-26 09:27:50 +01001705 .num_items = 4,
Tony Vroond2fd4b02009-06-21 00:40:10 +01001706 .items = {
1707 { "Ext Mic", 0x0 },
Tony Vroon684a8842009-06-26 09:27:50 +01001708 { "Line In", 0x2 },
Tony Vroond2fd4b02009-06-21 00:40:10 +01001709 { "CD", 0x4 },
Tony Vroon684a8842009-06-26 09:27:50 +01001710 { "Input Mix", 0xa },
Tony Vroond2fd4b02009-06-21 00:40:10 +01001711 },
1712 }
1713};
1714
Hector Martin018df412009-06-04 00:13:40 +02001715static struct hda_input_mux alc889_capture_sources[3] = {
1716 /* Digital mic only available on first "ADC" */
1717 {
1718 .num_items = 5,
1719 .items = {
1720 { "Mic", 0x0 },
1721 { "Line", 0x2 },
1722 { "CD", 0x4 },
1723 { "Front Mic", 0xb },
1724 { "Input Mix", 0xa },
1725 },
1726 },
1727 {
1728 .num_items = 4,
1729 .items = {
1730 { "Mic", 0x0 },
1731 { "Line", 0x2 },
1732 { "CD", 0x4 },
1733 { "Input Mix", 0xa },
1734 },
1735 },
1736 {
1737 .num_items = 4,
1738 .items = {
1739 { "Mic", 0x0 },
1740 { "Line", 0x2 },
1741 { "CD", 0x4 },
1742 { "Input Mix", 0xa },
1743 },
1744 }
1745};
1746
Vincent Petryef8ef5f2008-11-23 11:31:41 +08001747static struct snd_kcontrol_new alc888_base_mixer[] = {
Vincent Petry5b2d1ec2008-11-18 22:21:57 +08001748 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
1749 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
1750 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
1751 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
1752 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0,
1753 HDA_OUTPUT),
1754 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
1755 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
1756 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
1757 HDA_CODEC_VOLUME("Side Playback Volume", 0x0f, 0x0, HDA_OUTPUT),
1758 HDA_BIND_MUTE("Side Playback Switch", 0x0f, 2, HDA_INPUT),
1759 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
1760 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
1761 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
1762 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
1763 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
1764 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
1765 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Vincent Petry5b2d1ec2008-11-18 22:21:57 +08001766 { } /* end */
1767};
1768
Takashi Iwai4f5d17062009-08-11 18:17:46 +02001769static void alc888_acer_aspire_4930g_setup(struct hda_codec *codec)
Vincent Petry5b2d1ec2008-11-18 22:21:57 +08001770{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02001771 struct alc_spec *spec = codec->spec;
Vincent Petry5b2d1ec2008-11-18 22:21:57 +08001772
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02001773 spec->autocfg.hp_pins[0] = 0x15;
1774 spec->autocfg.speaker_pins[0] = 0x14;
Vincent Petry5b2d1ec2008-11-18 22:21:57 +08001775}
1776
Takashi Iwai4f5d17062009-08-11 18:17:46 +02001777static void alc888_acer_aspire_6530g_setup(struct hda_codec *codec)
Emilio López320d5922009-06-25 08:18:44 +02001778{
1779 struct alc_spec *spec = codec->spec;
1780
1781 spec->autocfg.hp_pins[0] = 0x15;
1782 spec->autocfg.speaker_pins[0] = 0x14;
1783 spec->autocfg.speaker_pins[1] = 0x16;
1784 spec->autocfg.speaker_pins[2] = 0x17;
Emilio López320d5922009-06-25 08:18:44 +02001785}
1786
Takashi Iwai4f5d17062009-08-11 18:17:46 +02001787static void alc889_acer_aspire_8930g_setup(struct hda_codec *codec)
Hector Martin3b315d72009-06-02 10:54:19 +02001788{
1789 struct alc_spec *spec = codec->spec;
1790
1791 spec->autocfg.hp_pins[0] = 0x15;
1792 spec->autocfg.speaker_pins[0] = 0x14;
1793 spec->autocfg.speaker_pins[1] = 0x16;
1794 spec->autocfg.speaker_pins[2] = 0x1b;
Hector Martin3b315d72009-06-02 10:54:19 +02001795}
1796
Vincent Petry5b2d1ec2008-11-18 22:21:57 +08001797/*
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001798 * ALC880 3-stack model
1799 *
1800 * DAC: Front = 0x02 (0x0c), Surr = 0x05 (0x0f), CLFE = 0x04 (0x0e)
Takashi Iwai9c7f8522006-06-28 15:08:22 +02001801 * Pin assignment: Front = 0x14, Line-In/Surr = 0x1a, Mic/CLFE = 0x18,
1802 * F-Mic = 0x1b, HP = 0x19
Linus Torvalds1da177e2005-04-16 15:20:36 -07001803 */
1804
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001805static hda_nid_t alc880_dac_nids[4] = {
1806 /* front, rear, clfe, rear_surr */
1807 0x02, 0x05, 0x04, 0x03
1808};
1809
1810static hda_nid_t alc880_adc_nids[3] = {
1811 /* ADC0-2 */
1812 0x07, 0x08, 0x09,
1813};
1814
1815/* The datasheet says the node 0x07 is connected from inputs,
1816 * but it shows zero connection in the real implementation on some devices.
Kailang Yangdf694da2005-12-05 19:42:22 +01001817 * Note: this is a 915GAV bug, fixed on 915GLV
Linus Torvalds1da177e2005-04-16 15:20:36 -07001818 */
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001819static hda_nid_t alc880_adc_nids_alt[2] = {
1820 /* ADC1-2 */
1821 0x08, 0x09,
1822};
1823
1824#define ALC880_DIGOUT_NID 0x06
1825#define ALC880_DIGIN_NID 0x0a
1826
1827static struct hda_input_mux alc880_capture_source = {
1828 .num_items = 4,
1829 .items = {
1830 { "Mic", 0x0 },
1831 { "Front Mic", 0x3 },
1832 { "Line", 0x2 },
1833 { "CD", 0x4 },
Linus Torvalds1da177e2005-04-16 15:20:36 -07001834 },
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001835};
1836
1837/* channel source setting (2/6 channel selection for 3-stack) */
1838/* 2ch mode */
1839static struct hda_verb alc880_threestack_ch2_init[] = {
1840 /* set line-in to input, mute it */
1841 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
1842 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
1843 /* set mic-in to input vref 80%, mute it */
1844 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
1845 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
Linus Torvalds1da177e2005-04-16 15:20:36 -07001846 { } /* end */
1847};
1848
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001849/* 6ch mode */
1850static struct hda_verb alc880_threestack_ch6_init[] = {
1851 /* set line-in to output, unmute it */
1852 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
1853 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
1854 /* set mic-in to output, unmute it */
1855 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
1856 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
1857 { } /* end */
1858};
1859
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +01001860static struct hda_channel_mode alc880_threestack_modes[2] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001861 { 2, alc880_threestack_ch2_init },
1862 { 6, alc880_threestack_ch6_init },
1863};
1864
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01001865static struct snd_kcontrol_new alc880_three_stack_mixer[] = {
Takashi Iwai05acb862005-06-10 19:50:25 +02001866 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001867 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Takashi Iwai05acb862005-06-10 19:50:25 +02001868 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0f, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001869 HDA_BIND_MUTE("Surround Playback Switch", 0x0f, 2, HDA_INPUT),
Takashi Iwai05acb862005-06-10 19:50:25 +02001870 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
1871 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01001872 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
1873 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
Linus Torvalds1da177e2005-04-16 15:20:36 -07001874 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
1875 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
1876 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
1877 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
1878 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
1879 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
1880 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x3, HDA_INPUT),
1881 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x3, HDA_INPUT),
Linus Torvalds1da177e2005-04-16 15:20:36 -07001882 HDA_CODEC_MUTE("Headphone Playback Switch", 0x19, 0x0, HDA_OUTPUT),
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001883 {
1884 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
1885 .name = "Channel Mode",
Kailang Yangdf694da2005-12-05 19:42:22 +01001886 .info = alc_ch_mode_info,
1887 .get = alc_ch_mode_get,
1888 .put = alc_ch_mode_put,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001889 },
1890 { } /* end */
1891};
1892
1893/* capture mixer elements */
Takashi Iwaif9e336f2008-10-31 16:37:07 +01001894static int alc_cap_vol_info(struct snd_kcontrol *kcontrol,
1895 struct snd_ctl_elem_info *uinfo)
1896{
1897 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
1898 struct alc_spec *spec = codec->spec;
1899 int err;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02001900
Wu Fengguang5a9e02e2009-01-09 16:45:24 +08001901 mutex_lock(&codec->control_mutex);
Takashi Iwaif9e336f2008-10-31 16:37:07 +01001902 kcontrol->private_value = HDA_COMPOSE_AMP_VAL(spec->adc_nids[0], 3, 0,
1903 HDA_INPUT);
1904 err = snd_hda_mixer_amp_volume_info(kcontrol, uinfo);
Wu Fengguang5a9e02e2009-01-09 16:45:24 +08001905 mutex_unlock(&codec->control_mutex);
Takashi Iwaif9e336f2008-10-31 16:37:07 +01001906 return err;
1907}
Linus Torvalds1da177e2005-04-16 15:20:36 -07001908
Takashi Iwaif9e336f2008-10-31 16:37:07 +01001909static int alc_cap_vol_tlv(struct snd_kcontrol *kcontrol, int op_flag,
1910 unsigned int size, unsigned int __user *tlv)
1911{
1912 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
1913 struct alc_spec *spec = codec->spec;
1914 int err;
1915
Wu Fengguang5a9e02e2009-01-09 16:45:24 +08001916 mutex_lock(&codec->control_mutex);
Takashi Iwaif9e336f2008-10-31 16:37:07 +01001917 kcontrol->private_value = HDA_COMPOSE_AMP_VAL(spec->adc_nids[0], 3, 0,
1918 HDA_INPUT);
1919 err = snd_hda_mixer_amp_tlv(kcontrol, op_flag, size, tlv);
Wu Fengguang5a9e02e2009-01-09 16:45:24 +08001920 mutex_unlock(&codec->control_mutex);
Takashi Iwaif9e336f2008-10-31 16:37:07 +01001921 return err;
1922}
1923
1924typedef int (*getput_call_t)(struct snd_kcontrol *kcontrol,
1925 struct snd_ctl_elem_value *ucontrol);
1926
1927static int alc_cap_getput_caller(struct snd_kcontrol *kcontrol,
1928 struct snd_ctl_elem_value *ucontrol,
1929 getput_call_t func)
1930{
1931 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
1932 struct alc_spec *spec = codec->spec;
1933 unsigned int adc_idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
1934 int err;
1935
Wu Fengguang5a9e02e2009-01-09 16:45:24 +08001936 mutex_lock(&codec->control_mutex);
Takashi Iwaif9e336f2008-10-31 16:37:07 +01001937 kcontrol->private_value = HDA_COMPOSE_AMP_VAL(spec->adc_nids[adc_idx],
1938 3, 0, HDA_INPUT);
1939 err = func(kcontrol, ucontrol);
Wu Fengguang5a9e02e2009-01-09 16:45:24 +08001940 mutex_unlock(&codec->control_mutex);
Takashi Iwaif9e336f2008-10-31 16:37:07 +01001941 return err;
1942}
1943
1944static int alc_cap_vol_get(struct snd_kcontrol *kcontrol,
1945 struct snd_ctl_elem_value *ucontrol)
1946{
1947 return alc_cap_getput_caller(kcontrol, ucontrol,
1948 snd_hda_mixer_amp_volume_get);
1949}
1950
1951static int alc_cap_vol_put(struct snd_kcontrol *kcontrol,
1952 struct snd_ctl_elem_value *ucontrol)
1953{
1954 return alc_cap_getput_caller(kcontrol, ucontrol,
1955 snd_hda_mixer_amp_volume_put);
1956}
1957
1958/* capture mixer elements */
1959#define alc_cap_sw_info snd_ctl_boolean_stereo_info
1960
1961static int alc_cap_sw_get(struct snd_kcontrol *kcontrol,
1962 struct snd_ctl_elem_value *ucontrol)
1963{
1964 return alc_cap_getput_caller(kcontrol, ucontrol,
1965 snd_hda_mixer_amp_switch_get);
1966}
1967
1968static int alc_cap_sw_put(struct snd_kcontrol *kcontrol,
1969 struct snd_ctl_elem_value *ucontrol)
1970{
1971 return alc_cap_getput_caller(kcontrol, ucontrol,
1972 snd_hda_mixer_amp_switch_put);
1973}
1974
Takashi Iwaia23b6882009-03-23 15:21:36 +01001975#define _DEFINE_CAPMIX(num) \
Takashi Iwaif9e336f2008-10-31 16:37:07 +01001976 { \
1977 .iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
1978 .name = "Capture Switch", \
1979 .access = SNDRV_CTL_ELEM_ACCESS_READWRITE, \
1980 .count = num, \
1981 .info = alc_cap_sw_info, \
1982 .get = alc_cap_sw_get, \
1983 .put = alc_cap_sw_put, \
1984 }, \
1985 { \
1986 .iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
1987 .name = "Capture Volume", \
1988 .access = (SNDRV_CTL_ELEM_ACCESS_READWRITE | \
1989 SNDRV_CTL_ELEM_ACCESS_TLV_READ | \
1990 SNDRV_CTL_ELEM_ACCESS_TLV_CALLBACK), \
1991 .count = num, \
1992 .info = alc_cap_vol_info, \
1993 .get = alc_cap_vol_get, \
1994 .put = alc_cap_vol_put, \
1995 .tlv = { .c = alc_cap_vol_tlv }, \
Takashi Iwaia23b6882009-03-23 15:21:36 +01001996 }
1997
1998#define _DEFINE_CAPSRC(num) \
Takashi Iwai3c3e9892008-10-31 17:48:56 +01001999 { \
2000 .iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
2001 /* .name = "Capture Source", */ \
2002 .name = "Input Source", \
2003 .count = num, \
2004 .info = alc_mux_enum_info, \
2005 .get = alc_mux_enum_get, \
2006 .put = alc_mux_enum_put, \
Takashi Iwaia23b6882009-03-23 15:21:36 +01002007 }
2008
2009#define DEFINE_CAPMIX(num) \
2010static struct snd_kcontrol_new alc_capture_mixer ## num[] = { \
2011 _DEFINE_CAPMIX(num), \
2012 _DEFINE_CAPSRC(num), \
2013 { } /* end */ \
2014}
2015
2016#define DEFINE_CAPMIX_NOSRC(num) \
2017static struct snd_kcontrol_new alc_capture_mixer_nosrc ## num[] = { \
2018 _DEFINE_CAPMIX(num), \
2019 { } /* end */ \
Takashi Iwaif9e336f2008-10-31 16:37:07 +01002020}
2021
2022/* up to three ADCs */
2023DEFINE_CAPMIX(1);
2024DEFINE_CAPMIX(2);
2025DEFINE_CAPMIX(3);
Takashi Iwaia23b6882009-03-23 15:21:36 +01002026DEFINE_CAPMIX_NOSRC(1);
2027DEFINE_CAPMIX_NOSRC(2);
2028DEFINE_CAPMIX_NOSRC(3);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002029
2030/*
2031 * ALC880 5-stack model
2032 *
Takashi Iwai9c7f8522006-06-28 15:08:22 +02002033 * DAC: Front = 0x02 (0x0c), Surr = 0x05 (0x0f), CLFE = 0x04 (0x0d),
2034 * Side = 0x02 (0xd)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002035 * Pin assignment: Front = 0x14, Surr = 0x17, CLFE = 0x16
2036 * Line-In/Side = 0x1a, Mic = 0x18, F-Mic = 0x1b, HP = 0x19
2037 */
2038
2039/* additional mixers to alc880_three_stack_mixer */
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01002040static struct snd_kcontrol_new alc880_five_stack_mixer[] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002041 HDA_CODEC_VOLUME("Side Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01002042 HDA_BIND_MUTE("Side Playback Switch", 0x0d, 2, HDA_INPUT),
Linus Torvalds1da177e2005-04-16 15:20:36 -07002043 { } /* end */
2044};
2045
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002046/* channel source setting (6/8 channel selection for 5-stack) */
2047/* 6ch mode */
2048static struct hda_verb alc880_fivestack_ch6_init[] = {
2049 /* set line-in to input, mute it */
2050 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
2051 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
Takashi Iwaidfc0ff62005-05-12 14:31:49 +02002052 { } /* end */
2053};
2054
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002055/* 8ch mode */
2056static struct hda_verb alc880_fivestack_ch8_init[] = {
2057 /* set line-in to output, unmute it */
2058 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
2059 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
2060 { } /* end */
2061};
2062
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +01002063static struct hda_channel_mode alc880_fivestack_modes[2] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002064 { 6, alc880_fivestack_ch6_init },
2065 { 8, alc880_fivestack_ch8_init },
2066};
2067
2068
2069/*
2070 * ALC880 6-stack model
2071 *
Takashi Iwai9c7f8522006-06-28 15:08:22 +02002072 * DAC: Front = 0x02 (0x0c), Surr = 0x03 (0x0d), CLFE = 0x04 (0x0e),
2073 * Side = 0x05 (0x0f)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002074 * Pin assignment: Front = 0x14, Surr = 0x15, CLFE = 0x16, Side = 0x17,
2075 * Mic = 0x18, F-Mic = 0x19, Line = 0x1a, HP = 0x1b
2076 */
2077
2078static hda_nid_t alc880_6st_dac_nids[4] = {
2079 /* front, rear, clfe, rear_surr */
2080 0x02, 0x03, 0x04, 0x05
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02002081};
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002082
2083static struct hda_input_mux alc880_6stack_capture_source = {
2084 .num_items = 4,
2085 .items = {
2086 { "Mic", 0x0 },
2087 { "Front Mic", 0x1 },
2088 { "Line", 0x2 },
2089 { "CD", 0x4 },
2090 },
2091};
2092
2093/* fixed 8-channels */
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +01002094static struct hda_channel_mode alc880_sixstack_modes[1] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002095 { 8, NULL },
2096};
2097
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01002098static struct snd_kcontrol_new alc880_six_stack_mixer[] = {
Takashi Iwai16ded522005-06-10 19:58:24 +02002099 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01002100 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02002101 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01002102 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02002103 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
2104 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01002105 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
2106 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02002107 HDA_CODEC_VOLUME("Side Playback Volume", 0x0f, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01002108 HDA_BIND_MUTE("Side Playback Switch", 0x0f, 2, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02002109 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
2110 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
2111 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
2112 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
2113 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
2114 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
2115 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
2116 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02002117 {
2118 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2119 .name = "Channel Mode",
Kailang Yangdf694da2005-12-05 19:42:22 +01002120 .info = alc_ch_mode_info,
2121 .get = alc_ch_mode_get,
2122 .put = alc_ch_mode_put,
Takashi Iwai16ded522005-06-10 19:58:24 +02002123 },
2124 { } /* end */
2125};
2126
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002127
2128/*
2129 * ALC880 W810 model
2130 *
2131 * W810 has rear IO for:
2132 * Front (DAC 02)
2133 * Surround (DAC 03)
2134 * Center/LFE (DAC 04)
2135 * Digital out (06)
2136 *
2137 * The system also has a pair of internal speakers, and a headphone jack.
2138 * These are both connected to Line2 on the codec, hence to DAC 02.
Kailang Yangea1fb292008-08-26 12:58:38 +02002139 *
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002140 * There is a variable resistor to control the speaker or headphone
2141 * volume. This is a hardware-only device without a software API.
2142 *
2143 * Plugging headphones in will disable the internal speakers. This is
2144 * implemented in hardware, not via the driver using jack sense. In
2145 * a similar fashion, plugging into the rear socket marked "front" will
2146 * disable both the speakers and headphones.
2147 *
2148 * For input, there's a microphone jack, and an "audio in" jack.
2149 * These may not do anything useful with this driver yet, because I
2150 * haven't setup any initialization verbs for these yet...
2151 */
2152
2153static hda_nid_t alc880_w810_dac_nids[3] = {
2154 /* front, rear/surround, clfe */
2155 0x02, 0x03, 0x04
2156};
2157
2158/* fixed 6 channels */
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +01002159static struct hda_channel_mode alc880_w810_modes[1] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002160 { 6, NULL }
2161};
2162
2163/* Pin assignment: Front = 0x14, Surr = 0x15, CLFE = 0x16, HP = 0x1b */
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01002164static struct snd_kcontrol_new alc880_w810_base_mixer[] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002165 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01002166 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002167 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01002168 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002169 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
2170 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01002171 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
2172 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002173 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
2174 { } /* end */
2175};
2176
2177
2178/*
2179 * Z710V model
2180 *
2181 * DAC: Front = 0x02 (0x0c), HP = 0x03 (0x0d)
Takashi Iwai9c7f8522006-06-28 15:08:22 +02002182 * Pin assignment: Front = 0x14, HP = 0x15, Mic = 0x18, Mic2 = 0x19(?),
2183 * Line = 0x1a
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002184 */
2185
2186static hda_nid_t alc880_z71v_dac_nids[1] = {
2187 0x02
2188};
2189#define ALC880_Z71V_HP_DAC 0x03
2190
2191/* fixed 2 channels */
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +01002192static struct hda_channel_mode alc880_2_jack_modes[1] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002193 { 2, NULL }
2194};
2195
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01002196static struct snd_kcontrol_new alc880_z71v_mixer[] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002197 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01002198 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002199 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01002200 HDA_BIND_MUTE("Headphone Playback Switch", 0x0d, 2, HDA_INPUT),
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002201 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
2202 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
2203 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
2204 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
2205 { } /* end */
2206};
2207
2208
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002209/*
2210 * ALC880 F1734 model
2211 *
2212 * DAC: HP = 0x02 (0x0c), Front = 0x03 (0x0d)
2213 * Pin assignment: HP = 0x14, Front = 0x15, Mic = 0x18
2214 */
2215
2216static hda_nid_t alc880_f1734_dac_nids[1] = {
2217 0x03
2218};
2219#define ALC880_F1734_HP_DAC 0x02
2220
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01002221static struct snd_kcontrol_new alc880_f1734_mixer[] = {
Takashi Iwai16ded522005-06-10 19:58:24 +02002222 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01002223 HDA_BIND_MUTE("Headphone Playback Switch", 0x0c, 2, HDA_INPUT),
Takashi Iwai2134ea42008-01-10 16:53:55 +01002224 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
2225 HDA_BIND_MUTE("Speaker Playback Switch", 0x0d, 2, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02002226 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
2227 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
Takashi Iwai937b4162008-02-11 14:52:36 +01002228 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
2229 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02002230 { } /* end */
2231};
2232
Takashi Iwai937b4162008-02-11 14:52:36 +01002233static struct hda_input_mux alc880_f1734_capture_source = {
2234 .num_items = 2,
2235 .items = {
2236 { "Mic", 0x1 },
2237 { "CD", 0x4 },
2238 },
2239};
2240
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002241
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002242/*
2243 * ALC880 ASUS model
2244 *
2245 * DAC: HP/Front = 0x02 (0x0c), Surr = 0x03 (0x0d), CLFE = 0x04 (0x0e)
2246 * Pin assignment: HP/Front = 0x14, Surr = 0x15, CLFE = 0x16,
2247 * Mic = 0x18, Line = 0x1a
2248 */
2249
2250#define alc880_asus_dac_nids alc880_w810_dac_nids /* identical with w810 */
2251#define alc880_asus_modes alc880_threestack_modes /* 2/6 channel mode */
2252
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01002253static struct snd_kcontrol_new alc880_asus_mixer[] = {
Takashi Iwai16ded522005-06-10 19:58:24 +02002254 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01002255 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02002256 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01002257 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02002258 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
2259 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01002260 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
2261 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02002262 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
2263 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
2264 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
2265 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
2266 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
2267 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02002268 {
2269 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2270 .name = "Channel Mode",
Kailang Yangdf694da2005-12-05 19:42:22 +01002271 .info = alc_ch_mode_info,
2272 .get = alc_ch_mode_get,
2273 .put = alc_ch_mode_put,
Takashi Iwai16ded522005-06-10 19:58:24 +02002274 },
2275 { } /* end */
2276};
2277
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002278/*
2279 * ALC880 ASUS W1V model
2280 *
2281 * DAC: HP/Front = 0x02 (0x0c), Surr = 0x03 (0x0d), CLFE = 0x04 (0x0e)
2282 * Pin assignment: HP/Front = 0x14, Surr = 0x15, CLFE = 0x16,
2283 * Mic = 0x18, Line = 0x1a, Line2 = 0x1b
2284 */
2285
2286/* additional mixers to alc880_asus_mixer */
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01002287static struct snd_kcontrol_new alc880_asus_w1v_mixer[] = {
Takashi Iwai16ded522005-06-10 19:58:24 +02002288 HDA_CODEC_VOLUME("Line2 Playback Volume", 0x0b, 0x03, HDA_INPUT),
2289 HDA_CODEC_MUTE("Line2 Playback Switch", 0x0b, 0x03, HDA_INPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02002290 { } /* end */
2291};
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002292
Kailang Yangdf694da2005-12-05 19:42:22 +01002293/* TCL S700 */
2294static struct snd_kcontrol_new alc880_tcl_s700_mixer[] = {
2295 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
2296 HDA_CODEC_MUTE("Front Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
2297 HDA_CODEC_MUTE("Headphone Playback Switch", 0x14, 0x0, HDA_OUTPUT),
2298 HDA_CODEC_VOLUME("CD Playback Volume", 0x0B, 0x04, HDA_INPUT),
2299 HDA_CODEC_MUTE("CD Playback Switch", 0x0B, 0x04, HDA_INPUT),
2300 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0B, 0x0, HDA_INPUT),
2301 HDA_CODEC_MUTE("Mic Playback Switch", 0x0B, 0x0, HDA_INPUT),
2302 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
2303 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
Kailang Yangdf694da2005-12-05 19:42:22 +01002304 { } /* end */
2305};
2306
Kailang Yangccc656c2006-10-17 12:32:26 +02002307/* Uniwill */
2308static struct snd_kcontrol_new alc880_uniwill_mixer[] = {
Takashi Iwai2134ea42008-01-10 16:53:55 +01002309 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
2310 HDA_BIND_MUTE("Headphone Playback Switch", 0x0c, 2, HDA_INPUT),
2311 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
2312 HDA_BIND_MUTE("Speaker Playback Switch", 0x0d, 2, HDA_INPUT),
Kailang Yangccc656c2006-10-17 12:32:26 +02002313 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
2314 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
2315 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
2316 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
2317 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
2318 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
2319 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
2320 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
2321 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
2322 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
2323 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
2324 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Kailang Yangccc656c2006-10-17 12:32:26 +02002325 {
2326 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2327 .name = "Channel Mode",
2328 .info = alc_ch_mode_info,
2329 .get = alc_ch_mode_get,
2330 .put = alc_ch_mode_put,
2331 },
2332 { } /* end */
2333};
2334
Tobin Davis2cf9f0f2007-02-07 16:04:25 +01002335static struct snd_kcontrol_new alc880_fujitsu_mixer[] = {
2336 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
2337 HDA_BIND_MUTE("Headphone Playback Switch", 0x0c, 2, HDA_INPUT),
2338 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
2339 HDA_BIND_MUTE("Speaker Playback Switch", 0x0d, 2, HDA_INPUT),
2340 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
2341 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
2342 HDA_CODEC_VOLUME("Ext Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
2343 HDA_CODEC_MUTE("Ext Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
2344 HDA_CODEC_VOLUME("Int Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
2345 HDA_CODEC_MUTE("Int Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
2346 { } /* end */
2347};
2348
Kailang Yangccc656c2006-10-17 12:32:26 +02002349static struct snd_kcontrol_new alc880_uniwill_p53_mixer[] = {
Takashi Iwai2134ea42008-01-10 16:53:55 +01002350 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
2351 HDA_BIND_MUTE("Headphone Playback Switch", 0x0c, 2, HDA_INPUT),
2352 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
2353 HDA_BIND_MUTE("Speaker Playback Switch", 0x0d, 2, HDA_INPUT),
Kailang Yangccc656c2006-10-17 12:32:26 +02002354 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
2355 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
2356 { } /* end */
2357};
2358
Linus Torvalds1da177e2005-04-16 15:20:36 -07002359/*
Takashi Iwai2134ea42008-01-10 16:53:55 +01002360 * virtual master controls
2361 */
2362
2363/*
2364 * slave controls for virtual master
2365 */
2366static const char *alc_slave_vols[] = {
2367 "Front Playback Volume",
2368 "Surround Playback Volume",
2369 "Center Playback Volume",
2370 "LFE Playback Volume",
2371 "Side Playback Volume",
2372 "Headphone Playback Volume",
2373 "Speaker Playback Volume",
2374 "Mono Playback Volume",
Takashi Iwai2134ea42008-01-10 16:53:55 +01002375 "Line-Out Playback Volume",
Takashi Iwai26f5df22008-11-03 17:39:46 +01002376 "PCM Playback Volume",
Takashi Iwai2134ea42008-01-10 16:53:55 +01002377 NULL,
2378};
2379
2380static const char *alc_slave_sws[] = {
2381 "Front Playback Switch",
2382 "Surround Playback Switch",
2383 "Center Playback Switch",
2384 "LFE Playback Switch",
2385 "Side Playback Switch",
2386 "Headphone Playback Switch",
2387 "Speaker Playback Switch",
2388 "Mono Playback Switch",
Takashi Iwaiedb54a52008-01-29 12:47:02 +01002389 "IEC958 Playback Switch",
Takashi Iwai2134ea42008-01-10 16:53:55 +01002390 NULL,
2391};
2392
2393/*
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002394 * build control elements
Linus Torvalds1da177e2005-04-16 15:20:36 -07002395 */
Takashi Iwai603c4012008-07-30 15:01:44 +02002396
2397static void alc_free_kctls(struct hda_codec *codec);
2398
Takashi Iwai67d634c2009-11-16 15:35:59 +01002399#ifdef CONFIG_SND_HDA_INPUT_BEEP
Takashi Iwai45bdd1c2009-02-06 16:11:25 +01002400/* additional beep mixers; the actual parameters are overwritten at build */
2401static struct snd_kcontrol_new alc_beep_mixer[] = {
2402 HDA_CODEC_VOLUME("Beep Playback Volume", 0, 0, HDA_INPUT),
Jaroslav Kysela123c07a2009-10-21 14:48:23 +02002403 HDA_CODEC_MUTE_BEEP("Beep Playback Switch", 0, 0, HDA_INPUT),
Takashi Iwai45bdd1c2009-02-06 16:11:25 +01002404 { } /* end */
2405};
Takashi Iwai67d634c2009-11-16 15:35:59 +01002406#endif
Takashi Iwai45bdd1c2009-02-06 16:11:25 +01002407
Linus Torvalds1da177e2005-04-16 15:20:36 -07002408static int alc_build_controls(struct hda_codec *codec)
2409{
2410 struct alc_spec *spec = codec->spec;
2411 int err;
2412 int i;
2413
2414 for (i = 0; i < spec->num_mixers; i++) {
2415 err = snd_hda_add_new_ctls(codec, spec->mixers[i]);
2416 if (err < 0)
2417 return err;
2418 }
Takashi Iwaif9e336f2008-10-31 16:37:07 +01002419 if (spec->cap_mixer) {
2420 err = snd_hda_add_new_ctls(codec, spec->cap_mixer);
2421 if (err < 0)
2422 return err;
2423 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002424 if (spec->multiout.dig_out_nid) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +02002425 err = snd_hda_create_spdif_out_ctls(codec,
2426 spec->multiout.dig_out_nid);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002427 if (err < 0)
2428 return err;
Takashi Iwaie64f14f2009-01-20 18:32:55 +01002429 if (!spec->no_analog) {
2430 err = snd_hda_create_spdif_share_sw(codec,
2431 &spec->multiout);
2432 if (err < 0)
2433 return err;
2434 spec->multiout.share_spdif = 1;
2435 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002436 }
2437 if (spec->dig_in_nid) {
2438 err = snd_hda_create_spdif_in_ctls(codec, spec->dig_in_nid);
2439 if (err < 0)
2440 return err;
2441 }
Takashi Iwai2134ea42008-01-10 16:53:55 +01002442
Takashi Iwai67d634c2009-11-16 15:35:59 +01002443#ifdef CONFIG_SND_HDA_INPUT_BEEP
Takashi Iwai45bdd1c2009-02-06 16:11:25 +01002444 /* create beep controls if needed */
2445 if (spec->beep_amp) {
2446 struct snd_kcontrol_new *knew;
2447 for (knew = alc_beep_mixer; knew->name; knew++) {
2448 struct snd_kcontrol *kctl;
2449 kctl = snd_ctl_new1(knew, codec);
2450 if (!kctl)
2451 return -ENOMEM;
2452 kctl->private_value = spec->beep_amp;
Jaroslav Kysela3911a4c2009-11-11 13:43:01 +01002453 err = snd_hda_ctl_add(codec,
2454 get_amp_nid_(spec->beep_amp), kctl);
Takashi Iwai45bdd1c2009-02-06 16:11:25 +01002455 if (err < 0)
2456 return err;
2457 }
2458 }
Takashi Iwai67d634c2009-11-16 15:35:59 +01002459#endif
Takashi Iwai45bdd1c2009-02-06 16:11:25 +01002460
Takashi Iwai2134ea42008-01-10 16:53:55 +01002461 /* if we have no master control, let's create it */
Takashi Iwaie64f14f2009-01-20 18:32:55 +01002462 if (!spec->no_analog &&
2463 !snd_hda_find_mixer_ctl(codec, "Master Playback Volume")) {
Takashi Iwai1c82ed12008-02-18 13:05:50 +01002464 unsigned int vmaster_tlv[4];
Takashi Iwai2134ea42008-01-10 16:53:55 +01002465 snd_hda_set_vmaster_tlv(codec, spec->vmaster_nid,
Takashi Iwai1c82ed12008-02-18 13:05:50 +01002466 HDA_OUTPUT, vmaster_tlv);
Takashi Iwai2134ea42008-01-10 16:53:55 +01002467 err = snd_hda_add_vmaster(codec, "Master Playback Volume",
Takashi Iwai1c82ed12008-02-18 13:05:50 +01002468 vmaster_tlv, alc_slave_vols);
Takashi Iwai2134ea42008-01-10 16:53:55 +01002469 if (err < 0)
2470 return err;
2471 }
Takashi Iwaie64f14f2009-01-20 18:32:55 +01002472 if (!spec->no_analog &&
2473 !snd_hda_find_mixer_ctl(codec, "Master Playback Switch")) {
Takashi Iwai2134ea42008-01-10 16:53:55 +01002474 err = snd_hda_add_vmaster(codec, "Master Playback Switch",
2475 NULL, alc_slave_sws);
2476 if (err < 0)
2477 return err;
2478 }
2479
Takashi Iwai603c4012008-07-30 15:01:44 +02002480 alc_free_kctls(codec); /* no longer needed */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002481 return 0;
2482}
2483
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002484
Linus Torvalds1da177e2005-04-16 15:20:36 -07002485/*
2486 * initialize the codec volumes, etc
2487 */
2488
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002489/*
2490 * generic initialization of ADC, input mixers and output mixers
2491 */
2492static struct hda_verb alc880_volume_init_verbs[] = {
2493 /*
2494 * Unmute ADC0-2 and set the default input to mic-in
2495 */
Takashi Iwai71fe7b82005-05-25 18:11:40 +02002496 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002497 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Takashi Iwai71fe7b82005-05-25 18:11:40 +02002498 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002499 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Takashi Iwai71fe7b82005-05-25 18:11:40 +02002500 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002501 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07002502
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002503 /* Unmute input amps (CD, Line In, Mic 1 & Mic 2) of the analog-loopback
2504 * mixer widget
Takashi Iwai9c7f8522006-06-28 15:08:22 +02002505 * Note: PASD motherboards uses the Line In 2 as the input for front
2506 * panel mic (mic 2)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002507 */
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002508 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +02002509 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
2510 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
2511 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
2512 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
2513 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
2514 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(6)},
2515 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(7)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07002516
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002517 /*
2518 * Set up output mixers (0x0c - 0x0f)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002519 */
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002520 /* set vol=0 to output mixers */
2521 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
2522 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
2523 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
2524 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
2525 /* set up input amps for analog loopback */
2526 /* Amp Indices: DAC = 0, mixer = 1 */
Takashi Iwai05acb862005-06-10 19:50:25 +02002527 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
2528 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Takashi Iwai05acb862005-06-10 19:50:25 +02002529 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
2530 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Takashi Iwai05acb862005-06-10 19:50:25 +02002531 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
2532 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Takashi Iwai05acb862005-06-10 19:50:25 +02002533 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
2534 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07002535
2536 { }
2537};
2538
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002539/*
2540 * 3-stack pin configuration:
2541 * front = 0x14, mic/clfe = 0x18, HP = 0x19, line/surr = 0x1a, f-mic = 0x1b
2542 */
2543static struct hda_verb alc880_pin_3stack_init_verbs[] = {
2544 /*
2545 * preset connection lists of input pins
2546 * 0 = front, 1 = rear_surr, 2 = CLFE, 3 = surround
2547 */
2548 {0x10, AC_VERB_SET_CONNECT_SEL, 0x02}, /* mic/clfe */
2549 {0x11, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
2550 {0x12, AC_VERB_SET_CONNECT_SEL, 0x03}, /* line/surround */
2551
2552 /*
2553 * Set pin mode and muting
2554 */
2555 /* set front pin widgets 0x14 for output */
2556 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2557 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2558 /* Mic1 (rear panel) pin widget for input and vref at 80% */
2559 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2560 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2561 /* Mic2 (as headphone out) for HP output */
2562 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2563 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2564 /* Line In pin widget for input */
2565 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2566 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2567 /* Line2 (as front mic) pin widget for input and vref at 80% */
2568 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2569 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2570 /* CD pin widget for input */
2571 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2572
2573 { }
2574};
2575
2576/*
2577 * 5-stack pin configuration:
2578 * front = 0x14, surround = 0x17, clfe = 0x16, mic = 0x18, HP = 0x19,
2579 * line-in/side = 0x1a, f-mic = 0x1b
2580 */
2581static struct hda_verb alc880_pin_5stack_init_verbs[] = {
2582 /*
2583 * preset connection lists of input pins
2584 * 0 = front, 1 = rear_surr, 2 = CLFE, 3 = surround
2585 */
2586 {0x11, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
2587 {0x12, AC_VERB_SET_CONNECT_SEL, 0x01}, /* line/side */
2588
2589 /*
2590 * Set pin mode and muting
2591 */
2592 /* set pin widgets 0x14-0x17 for output */
Takashi Iwai05acb862005-06-10 19:50:25 +02002593 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2594 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2595 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2596 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002597 /* unmute pins for output (no gain on this amp) */
2598 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2599 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2600 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2601 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2602
Linus Torvalds1da177e2005-04-16 15:20:36 -07002603 /* Mic1 (rear panel) pin widget for input and vref at 80% */
Takashi Iwai16ded522005-06-10 19:58:24 +02002604 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002605 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2606 /* Mic2 (as headphone out) for HP output */
2607 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Takashi Iwai05acb862005-06-10 19:50:25 +02002608 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002609 /* Line In pin widget for input */
2610 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2611 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2612 /* Line2 (as front mic) pin widget for input and vref at 80% */
2613 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2614 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2615 /* CD pin widget for input */
2616 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Linus Torvalds1da177e2005-04-16 15:20:36 -07002617
2618 { }
2619};
2620
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002621/*
2622 * W810 pin configuration:
2623 * front = 0x14, surround = 0x15, clfe = 0x16, HP = 0x1b
2624 */
2625static struct hda_verb alc880_pin_w810_init_verbs[] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002626 /* hphone/speaker input selector: front DAC */
2627 {0x13, AC_VERB_SET_CONNECT_SEL, 0x0},
2628
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002629 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2630 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2631 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2632 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2633 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2634 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2635
2636 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Takashi Iwai05acb862005-06-10 19:50:25 +02002637 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
Linus Torvalds1da177e2005-04-16 15:20:36 -07002638
Linus Torvalds1da177e2005-04-16 15:20:36 -07002639 { }
2640};
2641
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002642/*
2643 * Z71V pin configuration:
2644 * Speaker-out = 0x14, HP = 0x15, Mic = 0x18, Line-in = 0x1a, Mic2 = 0x1b (?)
2645 */
2646static struct hda_verb alc880_pin_z71v_init_verbs[] = {
Takashi Iwai05acb862005-06-10 19:50:25 +02002647 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002648 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Takashi Iwai05acb862005-06-10 19:50:25 +02002649 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002650 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Takashi Iwaidfc0ff62005-05-12 14:31:49 +02002651
Takashi Iwai16ded522005-06-10 19:58:24 +02002652 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002653 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Takashi Iwai16ded522005-06-10 19:58:24 +02002654 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002655 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Takashi Iwaidfc0ff62005-05-12 14:31:49 +02002656
2657 { }
2658};
2659
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002660/*
2661 * 6-stack pin configuration:
Takashi Iwai9c7f8522006-06-28 15:08:22 +02002662 * front = 0x14, surr = 0x15, clfe = 0x16, side = 0x17, mic = 0x18,
2663 * f-mic = 0x19, line = 0x1a, HP = 0x1b
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002664 */
2665static struct hda_verb alc880_pin_6stack_init_verbs[] = {
2666 {0x13, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
2667
Takashi Iwai16ded522005-06-10 19:58:24 +02002668 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002669 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Takashi Iwai16ded522005-06-10 19:58:24 +02002670 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002671 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Takashi Iwai16ded522005-06-10 19:58:24 +02002672 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002673 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Takashi Iwai16ded522005-06-10 19:58:24 +02002674 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002675 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2676
Takashi Iwai16ded522005-06-10 19:58:24 +02002677 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002678 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
Takashi Iwai16ded522005-06-10 19:58:24 +02002679 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002680 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
Takashi Iwai16ded522005-06-10 19:58:24 +02002681 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002682 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
Takashi Iwai16ded522005-06-10 19:58:24 +02002683 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Takashi Iwai16ded522005-06-10 19:58:24 +02002684 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Takashi Iwai16ded522005-06-10 19:58:24 +02002685 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Kailang Yangea1fb292008-08-26 12:58:38 +02002686
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002687 { }
2688};
Takashi Iwai16ded522005-06-10 19:58:24 +02002689
Kailang Yangccc656c2006-10-17 12:32:26 +02002690/*
2691 * Uniwill pin configuration:
2692 * HP = 0x14, InternalSpeaker = 0x15, mic = 0x18, internal mic = 0x19,
2693 * line = 0x1a
2694 */
2695static struct hda_verb alc880_uniwill_init_verbs[] = {
2696 {0x13, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
2697
2698 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2699 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2700 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2701 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2702 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2703 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2704 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2705 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2706 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
2707 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
2708 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
2709 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
2710 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
2711 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
2712
2713 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2714 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2715 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2716 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2717 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2718 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2719 /* {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP}, */
2720 /* {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE}, */
2721 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2722
2723 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
2724 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
2725
2726 { }
2727};
2728
2729/*
2730* Uniwill P53
Kailang Yangea1fb292008-08-26 12:58:38 +02002731* HP = 0x14, InternalSpeaker = 0x15, mic = 0x19,
Kailang Yangccc656c2006-10-17 12:32:26 +02002732 */
2733static struct hda_verb alc880_uniwill_p53_init_verbs[] = {
2734 {0x13, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
2735
2736 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2737 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2738 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2739 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2740 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2741 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2742 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
2743 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
2744 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
2745 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
2746 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
2747 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
2748
2749 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2750 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2751 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2752 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2753 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2754 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2755
2756 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
2757 {0x21, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_DCVOL_EVENT},
2758
2759 { }
2760};
2761
Tobin Davis2cf9f0f2007-02-07 16:04:25 +01002762static struct hda_verb alc880_beep_init_verbs[] = {
2763 { 0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(5) },
2764 { }
2765};
2766
Takashi Iwai458a4fa2007-05-05 12:18:40 +02002767/* auto-toggle front mic */
2768static void alc880_uniwill_mic_automute(struct hda_codec *codec)
2769{
2770 unsigned int present;
2771 unsigned char bits;
Kailang Yangccc656c2006-10-17 12:32:26 +02002772
Wu Fengguang864f92b2009-11-18 12:38:02 +08002773 present = snd_hda_jack_detect(codec, 0x18);
Takashi Iwai47fd8302007-08-10 17:11:07 +02002774 bits = present ? HDA_AMP_MUTE : 0;
2775 snd_hda_codec_amp_stereo(codec, 0x0b, HDA_INPUT, 1, HDA_AMP_MUTE, bits);
Takashi Iwai458a4fa2007-05-05 12:18:40 +02002776}
2777
Takashi Iwai4f5d17062009-08-11 18:17:46 +02002778static void alc880_uniwill_setup(struct hda_codec *codec)
Takashi Iwai458a4fa2007-05-05 12:18:40 +02002779{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02002780 struct alc_spec *spec = codec->spec;
2781
2782 spec->autocfg.hp_pins[0] = 0x14;
2783 spec->autocfg.speaker_pins[0] = 0x15;
2784 spec->autocfg.speaker_pins[0] = 0x16;
Takashi Iwai4f5d17062009-08-11 18:17:46 +02002785}
2786
2787static void alc880_uniwill_init_hook(struct hda_codec *codec)
2788{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02002789 alc_automute_amp(codec);
Takashi Iwai458a4fa2007-05-05 12:18:40 +02002790 alc880_uniwill_mic_automute(codec);
Kailang Yangccc656c2006-10-17 12:32:26 +02002791}
2792
2793static void alc880_uniwill_unsol_event(struct hda_codec *codec,
2794 unsigned int res)
2795{
2796 /* Looks like the unsol event is incompatible with the standard
2797 * definition. 4bit tag is placed at 28 bit!
2798 */
Takashi Iwai458a4fa2007-05-05 12:18:40 +02002799 switch (res >> 28) {
Takashi Iwai458a4fa2007-05-05 12:18:40 +02002800 case ALC880_MIC_EVENT:
2801 alc880_uniwill_mic_automute(codec);
2802 break;
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02002803 default:
2804 alc_automute_amp_unsol_event(codec, res);
2805 break;
Takashi Iwai458a4fa2007-05-05 12:18:40 +02002806 }
Kailang Yangccc656c2006-10-17 12:32:26 +02002807}
2808
Takashi Iwai4f5d17062009-08-11 18:17:46 +02002809static void alc880_uniwill_p53_setup(struct hda_codec *codec)
Kailang Yangccc656c2006-10-17 12:32:26 +02002810{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02002811 struct alc_spec *spec = codec->spec;
Kailang Yangccc656c2006-10-17 12:32:26 +02002812
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02002813 spec->autocfg.hp_pins[0] = 0x14;
2814 spec->autocfg.speaker_pins[0] = 0x15;
Kailang Yangccc656c2006-10-17 12:32:26 +02002815}
2816
2817static void alc880_uniwill_p53_dcvol_automute(struct hda_codec *codec)
2818{
2819 unsigned int present;
Kailang Yangea1fb292008-08-26 12:58:38 +02002820
Kailang Yangccc656c2006-10-17 12:32:26 +02002821 present = snd_hda_codec_read(codec, 0x21, 0,
Takashi Iwai47fd8302007-08-10 17:11:07 +02002822 AC_VERB_GET_VOLUME_KNOB_CONTROL, 0);
2823 present &= HDA_AMP_VOLMASK;
2824 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_OUTPUT, 0,
2825 HDA_AMP_VOLMASK, present);
2826 snd_hda_codec_amp_stereo(codec, 0x0d, HDA_OUTPUT, 0,
2827 HDA_AMP_VOLMASK, present);
Kailang Yangccc656c2006-10-17 12:32:26 +02002828}
Takashi Iwai47fd8302007-08-10 17:11:07 +02002829
Kailang Yangccc656c2006-10-17 12:32:26 +02002830static void alc880_uniwill_p53_unsol_event(struct hda_codec *codec,
2831 unsigned int res)
2832{
2833 /* Looks like the unsol event is incompatible with the standard
2834 * definition. 4bit tag is placed at 28 bit!
2835 */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02002836 if ((res >> 28) == ALC880_DCVOL_EVENT)
Kailang Yangccc656c2006-10-17 12:32:26 +02002837 alc880_uniwill_p53_dcvol_automute(codec);
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02002838 else
2839 alc_automute_amp_unsol_event(codec, res);
Kailang Yangccc656c2006-10-17 12:32:26 +02002840}
2841
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002842/*
2843 * F1734 pin configuration:
2844 * HP = 0x14, speaker-out = 0x15, mic = 0x18
2845 */
2846static struct hda_verb alc880_pin_f1734_init_verbs[] = {
Michael Gruberee7a9c72008-03-10 11:30:59 +01002847 {0x07, AC_VERB_SET_CONNECT_SEL, 0x01},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002848 {0x10, AC_VERB_SET_CONNECT_SEL, 0x02},
2849 {0x11, AC_VERB_SET_CONNECT_SEL, 0x00},
2850 {0x12, AC_VERB_SET_CONNECT_SEL, 0x01},
2851 {0x13, AC_VERB_SET_CONNECT_SEL, 0x00},
2852
2853 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2854 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2855 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2856 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2857
2858 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2859 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
Michael Gruberee7a9c72008-03-10 11:30:59 +01002860 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF50},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002861 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2862 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2863 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2864 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2865 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2866 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Takashi Iwai16ded522005-06-10 19:58:24 +02002867
Takashi Iwai937b4162008-02-11 14:52:36 +01002868 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|ALC880_HP_EVENT},
2869 {0x21, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|ALC880_DCVOL_EVENT},
2870
Takashi Iwai16ded522005-06-10 19:58:24 +02002871 { }
2872};
2873
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002874/*
2875 * ASUS pin configuration:
2876 * HP/front = 0x14, surr = 0x15, clfe = 0x16, mic = 0x18, line = 0x1a
2877 */
2878static struct hda_verb alc880_pin_asus_init_verbs[] = {
2879 {0x10, AC_VERB_SET_CONNECT_SEL, 0x02},
2880 {0x11, AC_VERB_SET_CONNECT_SEL, 0x00},
2881 {0x12, AC_VERB_SET_CONNECT_SEL, 0x01},
2882 {0x13, AC_VERB_SET_CONNECT_SEL, 0x00},
2883
2884 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2885 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2886 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2887 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2888 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2889 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2890 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2891 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2892
2893 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2894 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2895 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2896 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2897 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2898 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2899 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2900 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2901 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Kailang Yangea1fb292008-08-26 12:58:38 +02002902
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002903 { }
2904};
2905
2906/* Enable GPIO mask and set output */
Kailang Yangbc9f98a2007-04-12 13:06:07 +02002907#define alc880_gpio1_init_verbs alc_gpio1_init_verbs
2908#define alc880_gpio2_init_verbs alc_gpio2_init_verbs
David Heidelberger64a8be72009-06-08 16:15:18 +02002909#define alc880_gpio3_init_verbs alc_gpio3_init_verbs
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002910
Kailang Yangdf694da2005-12-05 19:42:22 +01002911/* Clevo m520g init */
2912static struct hda_verb alc880_pin_clevo_init_verbs[] = {
2913 /* headphone output */
2914 {0x11, AC_VERB_SET_CONNECT_SEL, 0x01},
2915 /* line-out */
2916 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2917 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2918 /* Line-in */
2919 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2920 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2921 /* CD */
2922 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2923 {0x1c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2924 /* Mic1 (rear panel) */
2925 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2926 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2927 /* Mic2 (front panel) */
2928 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2929 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2930 /* headphone */
2931 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2932 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2933 /* change to EAPD mode */
2934 {0x20, AC_VERB_SET_COEF_INDEX, 0x07},
2935 {0x20, AC_VERB_SET_PROC_COEF, 0x3060},
2936
2937 { }
2938};
2939
2940static struct hda_verb alc880_pin_tcl_S700_init_verbs[] = {
Takashi Iwai4b146cb2006-07-28 14:42:36 +02002941 /* change to EAPD mode */
2942 {0x20, AC_VERB_SET_COEF_INDEX, 0x07},
2943 {0x20, AC_VERB_SET_PROC_COEF, 0x3060},
2944
Kailang Yangdf694da2005-12-05 19:42:22 +01002945 /* Headphone output */
2946 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2947 /* Front output*/
2948 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2949 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
2950
2951 /* Line In pin widget for input */
2952 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2953 /* CD pin widget for input */
2954 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2955 /* Mic1 (rear panel) pin widget for input and vref at 80% */
2956 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2957
2958 /* change to EAPD mode */
2959 {0x20, AC_VERB_SET_COEF_INDEX, 0x07},
2960 {0x20, AC_VERB_SET_PROC_COEF, 0x3070},
2961
2962 { }
2963};
Takashi Iwaie9edcee2005-06-13 14:16:38 +02002964
2965/*
Takashi Iwaiae6b8132006-03-03 16:47:17 +01002966 * LG m1 express dual
2967 *
2968 * Pin assignment:
2969 * Rear Line-In/Out (blue): 0x14
2970 * Build-in Mic-In: 0x15
2971 * Speaker-out: 0x17
2972 * HP-Out (green): 0x1b
2973 * Mic-In/Out (red): 0x19
2974 * SPDIF-Out: 0x1e
2975 */
2976
2977/* To make 5.1 output working (green=Front, blue=Surr, red=CLFE) */
2978static hda_nid_t alc880_lg_dac_nids[3] = {
2979 0x05, 0x02, 0x03
2980};
2981
2982/* seems analog CD is not working */
2983static struct hda_input_mux alc880_lg_capture_source = {
2984 .num_items = 3,
2985 .items = {
2986 { "Mic", 0x1 },
2987 { "Line", 0x5 },
2988 { "Internal Mic", 0x6 },
2989 },
2990};
2991
2992/* 2,4,6 channel modes */
2993static struct hda_verb alc880_lg_ch2_init[] = {
2994 /* set line-in and mic-in to input */
2995 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
2996 { 0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
2997 { }
2998};
2999
3000static struct hda_verb alc880_lg_ch4_init[] = {
3001 /* set line-in to out and mic-in to input */
3002 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP },
3003 { 0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
3004 { }
3005};
3006
3007static struct hda_verb alc880_lg_ch6_init[] = {
3008 /* set line-in and mic-in to output */
3009 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP },
3010 { 0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP },
3011 { }
3012};
3013
3014static struct hda_channel_mode alc880_lg_ch_modes[3] = {
3015 { 2, alc880_lg_ch2_init },
3016 { 4, alc880_lg_ch4_init },
3017 { 6, alc880_lg_ch6_init },
3018};
3019
3020static struct snd_kcontrol_new alc880_lg_mixer[] = {
Takashi Iwai2134ea42008-01-10 16:53:55 +01003021 HDA_CODEC_VOLUME("Front Playback Volume", 0x0f, 0x0, HDA_OUTPUT),
3022 HDA_BIND_MUTE("Front Playback Switch", 0x0f, 2, HDA_INPUT),
Takashi Iwaiae6b8132006-03-03 16:47:17 +01003023 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
3024 HDA_BIND_MUTE("Surround Playback Switch", 0x0c, 2, HDA_INPUT),
3025 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0d, 1, 0x0, HDA_OUTPUT),
3026 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0d, 2, 0x0, HDA_OUTPUT),
3027 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0d, 1, 2, HDA_INPUT),
3028 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0d, 2, 2, HDA_INPUT),
3029 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
3030 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
3031 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x06, HDA_INPUT),
3032 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x06, HDA_INPUT),
3033 HDA_CODEC_VOLUME("Internal Mic Playback Volume", 0x0b, 0x07, HDA_INPUT),
3034 HDA_CODEC_MUTE("Internal Mic Playback Switch", 0x0b, 0x07, HDA_INPUT),
3035 {
3036 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
3037 .name = "Channel Mode",
3038 .info = alc_ch_mode_info,
3039 .get = alc_ch_mode_get,
3040 .put = alc_ch_mode_put,
3041 },
3042 { } /* end */
3043};
3044
3045static struct hda_verb alc880_lg_init_verbs[] = {
3046 /* set capture source to mic-in */
3047 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
3048 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
3049 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
3050 /* mute all amp mixer inputs */
3051 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(5)},
Takashi Iwaicb53c622007-08-10 17:21:45 +02003052 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(6)},
3053 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(7)},
Takashi Iwaiae6b8132006-03-03 16:47:17 +01003054 /* line-in to input */
3055 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
3056 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
3057 /* built-in mic */
3058 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
3059 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
3060 /* speaker-out */
3061 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
3062 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
3063 /* mic-in to input */
3064 {0x11, AC_VERB_SET_CONNECT_SEL, 0x01},
3065 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
3066 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
3067 /* HP-out */
3068 {0x13, AC_VERB_SET_CONNECT_SEL, 0x03},
3069 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
3070 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
3071 /* jack sense */
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02003072 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
Takashi Iwaiae6b8132006-03-03 16:47:17 +01003073 { }
3074};
3075
3076/* toggle speaker-output according to the hp-jack state */
Takashi Iwai4f5d17062009-08-11 18:17:46 +02003077static void alc880_lg_setup(struct hda_codec *codec)
Takashi Iwaiae6b8132006-03-03 16:47:17 +01003078{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02003079 struct alc_spec *spec = codec->spec;
Takashi Iwaiae6b8132006-03-03 16:47:17 +01003080
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02003081 spec->autocfg.hp_pins[0] = 0x1b;
3082 spec->autocfg.speaker_pins[0] = 0x17;
Takashi Iwaiae6b8132006-03-03 16:47:17 +01003083}
3084
3085/*
Takashi Iwaid6815182006-03-23 16:06:23 +01003086 * LG LW20
3087 *
3088 * Pin assignment:
3089 * Speaker-out: 0x14
3090 * Mic-In: 0x18
Claudio Matsuokae4f41da2007-07-13 11:51:06 +02003091 * Built-in Mic-In: 0x19
3092 * Line-In: 0x1b
3093 * HP-Out: 0x1a
Takashi Iwaid6815182006-03-23 16:06:23 +01003094 * SPDIF-Out: 0x1e
3095 */
3096
Takashi Iwaid6815182006-03-23 16:06:23 +01003097static struct hda_input_mux alc880_lg_lw_capture_source = {
Claudio Matsuokae4f41da2007-07-13 11:51:06 +02003098 .num_items = 3,
Takashi Iwaid6815182006-03-23 16:06:23 +01003099 .items = {
3100 { "Mic", 0x0 },
3101 { "Internal Mic", 0x1 },
Claudio Matsuokae4f41da2007-07-13 11:51:06 +02003102 { "Line In", 0x2 },
Takashi Iwaid6815182006-03-23 16:06:23 +01003103 },
3104};
3105
Claudio Matsuoka0a8c5da2007-07-04 15:17:38 +02003106#define alc880_lg_lw_modes alc880_threestack_modes
3107
Takashi Iwaid6815182006-03-23 16:06:23 +01003108static struct snd_kcontrol_new alc880_lg_lw_mixer[] = {
Claudio Matsuoka0a8c5da2007-07-04 15:17:38 +02003109 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
3110 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
3111 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0f, 0x0, HDA_OUTPUT),
3112 HDA_BIND_MUTE("Surround Playback Switch", 0x0f, 2, HDA_INPUT),
3113 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
3114 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
3115 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
3116 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
3117 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
3118 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
Takashi Iwaid6815182006-03-23 16:06:23 +01003119 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
3120 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
3121 HDA_CODEC_VOLUME("Internal Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
3122 HDA_CODEC_MUTE("Internal Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
Claudio Matsuoka0a8c5da2007-07-04 15:17:38 +02003123 {
3124 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
3125 .name = "Channel Mode",
3126 .info = alc_ch_mode_info,
3127 .get = alc_ch_mode_get,
3128 .put = alc_ch_mode_put,
3129 },
Takashi Iwaid6815182006-03-23 16:06:23 +01003130 { } /* end */
3131};
3132
3133static struct hda_verb alc880_lg_lw_init_verbs[] = {
Claudio Matsuoka0a8c5da2007-07-04 15:17:38 +02003134 {0x13, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
3135 {0x10, AC_VERB_SET_CONNECT_SEL, 0x02}, /* mic/clfe */
3136 {0x12, AC_VERB_SET_CONNECT_SEL, 0x03}, /* line/surround */
3137
Takashi Iwaid6815182006-03-23 16:06:23 +01003138 /* set capture source to mic-in */
3139 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
3140 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
3141 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Takashi Iwaicb53c622007-08-10 17:21:45 +02003142 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(7)},
Takashi Iwaid6815182006-03-23 16:06:23 +01003143 /* speaker-out */
3144 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
3145 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
3146 /* HP-out */
Takashi Iwaid6815182006-03-23 16:06:23 +01003147 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
3148 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
3149 /* mic-in to input */
3150 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
3151 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
3152 /* built-in mic */
3153 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
3154 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
3155 /* jack sense */
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02003156 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
Takashi Iwaid6815182006-03-23 16:06:23 +01003157 { }
3158};
3159
3160/* toggle speaker-output according to the hp-jack state */
Takashi Iwai4f5d17062009-08-11 18:17:46 +02003161static void alc880_lg_lw_setup(struct hda_codec *codec)
Takashi Iwaid6815182006-03-23 16:06:23 +01003162{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02003163 struct alc_spec *spec = codec->spec;
Takashi Iwaid6815182006-03-23 16:06:23 +01003164
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02003165 spec->autocfg.hp_pins[0] = 0x1b;
3166 spec->autocfg.speaker_pins[0] = 0x14;
Takashi Iwaid6815182006-03-23 16:06:23 +01003167}
3168
Takashi Iwaidf99cd32008-04-25 15:25:04 +02003169static struct snd_kcontrol_new alc880_medion_rim_mixer[] = {
3170 HDA_CODEC_VOLUME("Master Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
3171 HDA_BIND_MUTE("Master Playback Switch", 0x0c, 2, HDA_INPUT),
3172 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
3173 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
3174 HDA_CODEC_VOLUME("Internal Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
3175 HDA_CODEC_MUTE("Internal Playback Switch", 0x0b, 0x1, HDA_INPUT),
3176 { } /* end */
3177};
3178
3179static struct hda_input_mux alc880_medion_rim_capture_source = {
3180 .num_items = 2,
3181 .items = {
3182 { "Mic", 0x0 },
3183 { "Internal Mic", 0x1 },
3184 },
3185};
3186
3187static struct hda_verb alc880_medion_rim_init_verbs[] = {
3188 {0x13, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
3189
3190 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
3191 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
3192
3193 /* Mic1 (rear panel) pin widget for input and vref at 80% */
3194 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
3195 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
3196 /* Mic2 (as headphone out) for HP output */
3197 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
3198 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
3199 /* Internal Speaker */
3200 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
3201 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
3202
3203 {0x20, AC_VERB_SET_COEF_INDEX, 0x07},
3204 {0x20, AC_VERB_SET_PROC_COEF, 0x3060},
3205
3206 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
3207 { }
3208};
3209
3210/* toggle speaker-output according to the hp-jack state */
3211static void alc880_medion_rim_automute(struct hda_codec *codec)
3212{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02003213 struct alc_spec *spec = codec->spec;
3214 alc_automute_amp(codec);
3215 /* toggle EAPD */
3216 if (spec->jack_present)
Takashi Iwaidf99cd32008-04-25 15:25:04 +02003217 snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DATA, 0);
3218 else
3219 snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DATA, 2);
3220}
3221
3222static void alc880_medion_rim_unsol_event(struct hda_codec *codec,
3223 unsigned int res)
3224{
3225 /* Looks like the unsol event is incompatible with the standard
3226 * definition. 4bit tag is placed at 28 bit!
3227 */
3228 if ((res >> 28) == ALC880_HP_EVENT)
3229 alc880_medion_rim_automute(codec);
3230}
3231
Takashi Iwai4f5d17062009-08-11 18:17:46 +02003232static void alc880_medion_rim_setup(struct hda_codec *codec)
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02003233{
3234 struct alc_spec *spec = codec->spec;
3235
3236 spec->autocfg.hp_pins[0] = 0x14;
3237 spec->autocfg.speaker_pins[0] = 0x1b;
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02003238}
3239
Takashi Iwaicb53c622007-08-10 17:21:45 +02003240#ifdef CONFIG_SND_HDA_POWER_SAVE
3241static struct hda_amp_list alc880_loopbacks[] = {
3242 { 0x0b, HDA_INPUT, 0 },
3243 { 0x0b, HDA_INPUT, 1 },
3244 { 0x0b, HDA_INPUT, 2 },
3245 { 0x0b, HDA_INPUT, 3 },
3246 { 0x0b, HDA_INPUT, 4 },
3247 { } /* end */
3248};
3249
3250static struct hda_amp_list alc880_lg_loopbacks[] = {
3251 { 0x0b, HDA_INPUT, 1 },
3252 { 0x0b, HDA_INPUT, 6 },
3253 { 0x0b, HDA_INPUT, 7 },
3254 { } /* end */
3255};
3256#endif
3257
Takashi Iwaid6815182006-03-23 16:06:23 +01003258/*
Takashi Iwaiae6b8132006-03-03 16:47:17 +01003259 * Common callbacks
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003260 */
Takashi Iwai16ded522005-06-10 19:58:24 +02003261
Linus Torvalds1da177e2005-04-16 15:20:36 -07003262static int alc_init(struct hda_codec *codec)
3263{
3264 struct alc_spec *spec = codec->spec;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003265 unsigned int i;
3266
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +02003267 alc_fix_pll(codec);
Takashi Iwai4a79ba32009-04-22 16:31:35 +02003268 alc_auto_init_amp(codec, spec->init_amp);
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +02003269
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003270 for (i = 0; i < spec->num_init_verbs; i++)
3271 snd_hda_sequence_write(codec, spec->init_verbs[i]);
Takashi Iwaiae6b8132006-03-03 16:47:17 +01003272
3273 if (spec->init_hook)
3274 spec->init_hook(codec);
3275
Linus Torvalds1da177e2005-04-16 15:20:36 -07003276 return 0;
3277}
3278
Takashi Iwaiae6b8132006-03-03 16:47:17 +01003279static void alc_unsol_event(struct hda_codec *codec, unsigned int res)
3280{
3281 struct alc_spec *spec = codec->spec;
3282
3283 if (spec->unsol_event)
3284 spec->unsol_event(codec, res);
3285}
3286
Takashi Iwaicb53c622007-08-10 17:21:45 +02003287#ifdef CONFIG_SND_HDA_POWER_SAVE
3288static int alc_check_power_status(struct hda_codec *codec, hda_nid_t nid)
3289{
3290 struct alc_spec *spec = codec->spec;
3291 return snd_hda_check_amp_list_power(codec, &spec->loopback, nid);
3292}
3293#endif
3294
Linus Torvalds1da177e2005-04-16 15:20:36 -07003295/*
3296 * Analog playback callbacks
3297 */
3298static int alc880_playback_pcm_open(struct hda_pcm_stream *hinfo,
3299 struct hda_codec *codec,
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01003300 struct snd_pcm_substream *substream)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003301{
3302 struct alc_spec *spec = codec->spec;
Takashi Iwai9a081602008-02-12 18:37:26 +01003303 return snd_hda_multi_out_analog_open(codec, &spec->multiout, substream,
3304 hinfo);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003305}
3306
3307static int alc880_playback_pcm_prepare(struct hda_pcm_stream *hinfo,
3308 struct hda_codec *codec,
3309 unsigned int stream_tag,
3310 unsigned int format,
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01003311 struct snd_pcm_substream *substream)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003312{
3313 struct alc_spec *spec = codec->spec;
Takashi Iwai9c7f8522006-06-28 15:08:22 +02003314 return snd_hda_multi_out_analog_prepare(codec, &spec->multiout,
3315 stream_tag, format, substream);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003316}
3317
3318static int alc880_playback_pcm_cleanup(struct hda_pcm_stream *hinfo,
3319 struct hda_codec *codec,
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01003320 struct snd_pcm_substream *substream)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003321{
3322 struct alc_spec *spec = codec->spec;
3323 return snd_hda_multi_out_analog_cleanup(codec, &spec->multiout);
3324}
3325
3326/*
3327 * Digital out
3328 */
3329static int alc880_dig_playback_pcm_open(struct hda_pcm_stream *hinfo,
3330 struct hda_codec *codec,
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01003331 struct snd_pcm_substream *substream)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003332{
3333 struct alc_spec *spec = codec->spec;
3334 return snd_hda_multi_out_dig_open(codec, &spec->multiout);
3335}
3336
Takashi Iwai6b97eb42007-04-05 14:51:48 +02003337static int alc880_dig_playback_pcm_prepare(struct hda_pcm_stream *hinfo,
3338 struct hda_codec *codec,
3339 unsigned int stream_tag,
3340 unsigned int format,
3341 struct snd_pcm_substream *substream)
3342{
3343 struct alc_spec *spec = codec->spec;
3344 return snd_hda_multi_out_dig_prepare(codec, &spec->multiout,
3345 stream_tag, format, substream);
3346}
3347
Takashi Iwai9b5f12e2009-02-13 11:47:37 +01003348static int alc880_dig_playback_pcm_cleanup(struct hda_pcm_stream *hinfo,
3349 struct hda_codec *codec,
3350 struct snd_pcm_substream *substream)
3351{
3352 struct alc_spec *spec = codec->spec;
3353 return snd_hda_multi_out_dig_cleanup(codec, &spec->multiout);
3354}
3355
Linus Torvalds1da177e2005-04-16 15:20:36 -07003356static int alc880_dig_playback_pcm_close(struct hda_pcm_stream *hinfo,
3357 struct hda_codec *codec,
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01003358 struct snd_pcm_substream *substream)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003359{
3360 struct alc_spec *spec = codec->spec;
3361 return snd_hda_multi_out_dig_close(codec, &spec->multiout);
3362}
3363
3364/*
3365 * Analog capture
3366 */
Takashi Iwai63300792008-01-24 15:31:36 +01003367static int alc880_alt_capture_pcm_prepare(struct hda_pcm_stream *hinfo,
Linus Torvalds1da177e2005-04-16 15:20:36 -07003368 struct hda_codec *codec,
3369 unsigned int stream_tag,
3370 unsigned int format,
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01003371 struct snd_pcm_substream *substream)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003372{
3373 struct alc_spec *spec = codec->spec;
3374
Takashi Iwai63300792008-01-24 15:31:36 +01003375 snd_hda_codec_setup_stream(codec, spec->adc_nids[substream->number + 1],
Linus Torvalds1da177e2005-04-16 15:20:36 -07003376 stream_tag, 0, format);
3377 return 0;
3378}
3379
Takashi Iwai63300792008-01-24 15:31:36 +01003380static int alc880_alt_capture_pcm_cleanup(struct hda_pcm_stream *hinfo,
Linus Torvalds1da177e2005-04-16 15:20:36 -07003381 struct hda_codec *codec,
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01003382 struct snd_pcm_substream *substream)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003383{
3384 struct alc_spec *spec = codec->spec;
3385
Takashi Iwai888afa12008-03-18 09:57:50 +01003386 snd_hda_codec_cleanup_stream(codec,
3387 spec->adc_nids[substream->number + 1]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003388 return 0;
3389}
3390
3391
3392/*
3393 */
3394static struct hda_pcm_stream alc880_pcm_analog_playback = {
3395 .substreams = 1,
3396 .channels_min = 2,
3397 .channels_max = 8,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003398 /* NID is set in alc_build_pcms */
Linus Torvalds1da177e2005-04-16 15:20:36 -07003399 .ops = {
3400 .open = alc880_playback_pcm_open,
3401 .prepare = alc880_playback_pcm_prepare,
3402 .cleanup = alc880_playback_pcm_cleanup
3403 },
3404};
3405
3406static struct hda_pcm_stream alc880_pcm_analog_capture = {
Takashi Iwai63300792008-01-24 15:31:36 +01003407 .substreams = 1,
3408 .channels_min = 2,
3409 .channels_max = 2,
3410 /* NID is set in alc_build_pcms */
3411};
3412
3413static struct hda_pcm_stream alc880_pcm_analog_alt_playback = {
3414 .substreams = 1,
3415 .channels_min = 2,
3416 .channels_max = 2,
3417 /* NID is set in alc_build_pcms */
3418};
3419
3420static struct hda_pcm_stream alc880_pcm_analog_alt_capture = {
3421 .substreams = 2, /* can be overridden */
Linus Torvalds1da177e2005-04-16 15:20:36 -07003422 .channels_min = 2,
3423 .channels_max = 2,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003424 /* NID is set in alc_build_pcms */
Linus Torvalds1da177e2005-04-16 15:20:36 -07003425 .ops = {
Takashi Iwai63300792008-01-24 15:31:36 +01003426 .prepare = alc880_alt_capture_pcm_prepare,
3427 .cleanup = alc880_alt_capture_pcm_cleanup
Linus Torvalds1da177e2005-04-16 15:20:36 -07003428 },
3429};
3430
3431static struct hda_pcm_stream alc880_pcm_digital_playback = {
3432 .substreams = 1,
3433 .channels_min = 2,
3434 .channels_max = 2,
3435 /* NID is set in alc_build_pcms */
3436 .ops = {
3437 .open = alc880_dig_playback_pcm_open,
Takashi Iwai6b97eb42007-04-05 14:51:48 +02003438 .close = alc880_dig_playback_pcm_close,
Takashi Iwai9b5f12e2009-02-13 11:47:37 +01003439 .prepare = alc880_dig_playback_pcm_prepare,
3440 .cleanup = alc880_dig_playback_pcm_cleanup
Linus Torvalds1da177e2005-04-16 15:20:36 -07003441 },
3442};
3443
3444static struct hda_pcm_stream alc880_pcm_digital_capture = {
3445 .substreams = 1,
3446 .channels_min = 2,
3447 .channels_max = 2,
3448 /* NID is set in alc_build_pcms */
3449};
3450
Jonathan Woithe4c5186e2006-02-09 11:53:48 +01003451/* Used by alc_build_pcms to flag that a PCM has no playback stream */
Takashi Iwai63300792008-01-24 15:31:36 +01003452static struct hda_pcm_stream alc_pcm_null_stream = {
Jonathan Woithe4c5186e2006-02-09 11:53:48 +01003453 .substreams = 0,
3454 .channels_min = 0,
3455 .channels_max = 0,
3456};
3457
Linus Torvalds1da177e2005-04-16 15:20:36 -07003458static int alc_build_pcms(struct hda_codec *codec)
3459{
3460 struct alc_spec *spec = codec->spec;
3461 struct hda_pcm *info = spec->pcm_rec;
3462 int i;
3463
3464 codec->num_pcms = 1;
3465 codec->pcm_info = info;
3466
Takashi Iwaie64f14f2009-01-20 18:32:55 +01003467 if (spec->no_analog)
3468 goto skip_analog;
3469
Takashi Iwai812a2cc2009-05-16 10:00:49 +02003470 snprintf(spec->stream_name_analog, sizeof(spec->stream_name_analog),
3471 "%s Analog", codec->chip_name);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003472 info->name = spec->stream_name_analog;
Takashi Iwai812a2cc2009-05-16 10:00:49 +02003473
Takashi Iwai4a471b72005-12-07 13:56:29 +01003474 if (spec->stream_analog_playback) {
Takashi Iwaida3cec32008-08-08 17:12:14 +02003475 if (snd_BUG_ON(!spec->multiout.dac_nids))
3476 return -EINVAL;
Takashi Iwai4a471b72005-12-07 13:56:29 +01003477 info->stream[SNDRV_PCM_STREAM_PLAYBACK] = *(spec->stream_analog_playback);
3478 info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid = spec->multiout.dac_nids[0];
3479 }
3480 if (spec->stream_analog_capture) {
Takashi Iwaida3cec32008-08-08 17:12:14 +02003481 if (snd_BUG_ON(!spec->adc_nids))
3482 return -EINVAL;
Takashi Iwai4a471b72005-12-07 13:56:29 +01003483 info->stream[SNDRV_PCM_STREAM_CAPTURE] = *(spec->stream_analog_capture);
3484 info->stream[SNDRV_PCM_STREAM_CAPTURE].nid = spec->adc_nids[0];
3485 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07003486
Takashi Iwai4a471b72005-12-07 13:56:29 +01003487 if (spec->channel_mode) {
3488 info->stream[SNDRV_PCM_STREAM_PLAYBACK].channels_max = 0;
3489 for (i = 0; i < spec->num_channel_mode; i++) {
3490 if (spec->channel_mode[i].channels > info->stream[SNDRV_PCM_STREAM_PLAYBACK].channels_max) {
3491 info->stream[SNDRV_PCM_STREAM_PLAYBACK].channels_max = spec->channel_mode[i].channels;
3492 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07003493 }
3494 }
3495
Takashi Iwaie64f14f2009-01-20 18:32:55 +01003496 skip_analog:
Takashi Iwaie08a0072006-09-07 17:52:14 +02003497 /* SPDIF for stream index #1 */
Linus Torvalds1da177e2005-04-16 15:20:36 -07003498 if (spec->multiout.dig_out_nid || spec->dig_in_nid) {
Takashi Iwai812a2cc2009-05-16 10:00:49 +02003499 snprintf(spec->stream_name_digital,
3500 sizeof(spec->stream_name_digital),
3501 "%s Digital", codec->chip_name);
Takashi Iwaie08a0072006-09-07 17:52:14 +02003502 codec->num_pcms = 2;
Wu Fengguangb25c9da2009-02-06 15:02:27 +08003503 codec->slave_dig_outs = spec->multiout.slave_dig_outs;
Takashi Iwaic06134d2006-10-11 18:49:13 +02003504 info = spec->pcm_rec + 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003505 info->name = spec->stream_name_digital;
Takashi Iwai8c441982009-01-20 18:30:20 +01003506 if (spec->dig_out_type)
3507 info->pcm_type = spec->dig_out_type;
3508 else
3509 info->pcm_type = HDA_PCM_TYPE_SPDIF;
Takashi Iwai4a471b72005-12-07 13:56:29 +01003510 if (spec->multiout.dig_out_nid &&
3511 spec->stream_digital_playback) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003512 info->stream[SNDRV_PCM_STREAM_PLAYBACK] = *(spec->stream_digital_playback);
3513 info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid = spec->multiout.dig_out_nid;
3514 }
Takashi Iwai4a471b72005-12-07 13:56:29 +01003515 if (spec->dig_in_nid &&
3516 spec->stream_digital_capture) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003517 info->stream[SNDRV_PCM_STREAM_CAPTURE] = *(spec->stream_digital_capture);
3518 info->stream[SNDRV_PCM_STREAM_CAPTURE].nid = spec->dig_in_nid;
3519 }
Takashi Iwai963f8032008-08-11 10:04:40 +02003520 /* FIXME: do we need this for all Realtek codec models? */
3521 codec->spdif_status_reset = 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003522 }
3523
Takashi Iwaie64f14f2009-01-20 18:32:55 +01003524 if (spec->no_analog)
3525 return 0;
3526
Takashi Iwaie08a0072006-09-07 17:52:14 +02003527 /* If the use of more than one ADC is requested for the current
3528 * model, configure a second analog capture-only PCM.
3529 */
3530 /* Additional Analaog capture for index #2 */
Takashi Iwai63300792008-01-24 15:31:36 +01003531 if ((spec->alt_dac_nid && spec->stream_analog_alt_playback) ||
3532 (spec->num_adc_nids > 1 && spec->stream_analog_alt_capture)) {
Takashi Iwaie08a0072006-09-07 17:52:14 +02003533 codec->num_pcms = 3;
Takashi Iwaic06134d2006-10-11 18:49:13 +02003534 info = spec->pcm_rec + 2;
Takashi Iwaie08a0072006-09-07 17:52:14 +02003535 info->name = spec->stream_name_analog;
Takashi Iwai63300792008-01-24 15:31:36 +01003536 if (spec->alt_dac_nid) {
3537 info->stream[SNDRV_PCM_STREAM_PLAYBACK] =
3538 *spec->stream_analog_alt_playback;
3539 info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid =
3540 spec->alt_dac_nid;
3541 } else {
3542 info->stream[SNDRV_PCM_STREAM_PLAYBACK] =
3543 alc_pcm_null_stream;
3544 info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid = 0;
3545 }
3546 if (spec->num_adc_nids > 1) {
3547 info->stream[SNDRV_PCM_STREAM_CAPTURE] =
3548 *spec->stream_analog_alt_capture;
3549 info->stream[SNDRV_PCM_STREAM_CAPTURE].nid =
3550 spec->adc_nids[1];
3551 info->stream[SNDRV_PCM_STREAM_CAPTURE].substreams =
3552 spec->num_adc_nids - 1;
3553 } else {
3554 info->stream[SNDRV_PCM_STREAM_CAPTURE] =
3555 alc_pcm_null_stream;
3556 info->stream[SNDRV_PCM_STREAM_CAPTURE].nid = 0;
Takashi Iwaie08a0072006-09-07 17:52:14 +02003557 }
3558 }
3559
Linus Torvalds1da177e2005-04-16 15:20:36 -07003560 return 0;
3561}
3562
Takashi Iwai603c4012008-07-30 15:01:44 +02003563static void alc_free_kctls(struct hda_codec *codec)
3564{
3565 struct alc_spec *spec = codec->spec;
3566
3567 if (spec->kctls.list) {
3568 struct snd_kcontrol_new *kctl = spec->kctls.list;
3569 int i;
3570 for (i = 0; i < spec->kctls.used; i++)
3571 kfree(kctl[i].name);
3572 }
3573 snd_array_free(&spec->kctls);
3574}
3575
Linus Torvalds1da177e2005-04-16 15:20:36 -07003576static void alc_free(struct hda_codec *codec)
3577{
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003578 struct alc_spec *spec = codec->spec;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003579
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003580 if (!spec)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003581 return;
3582
Takashi Iwai603c4012008-07-30 15:01:44 +02003583 alc_free_kctls(codec);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003584 kfree(spec);
Kusanagi Kouichi680cd532009-02-05 00:00:58 +09003585 snd_hda_detach_beep_device(codec);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003586}
3587
Takashi Iwaie044c392008-10-27 16:56:24 +01003588#ifdef SND_HDA_NEEDS_RESUME
Takashi Iwaie044c392008-10-27 16:56:24 +01003589static int alc_resume(struct hda_codec *codec)
3590{
Takashi Iwaie044c392008-10-27 16:56:24 +01003591 codec->patch_ops.init(codec);
3592 snd_hda_codec_resume_amp(codec);
3593 snd_hda_codec_resume_cache(codec);
3594 return 0;
3595}
Takashi Iwaie044c392008-10-27 16:56:24 +01003596#endif
3597
Linus Torvalds1da177e2005-04-16 15:20:36 -07003598/*
3599 */
3600static struct hda_codec_ops alc_patch_ops = {
3601 .build_controls = alc_build_controls,
3602 .build_pcms = alc_build_pcms,
3603 .init = alc_init,
3604 .free = alc_free,
Takashi Iwaiae6b8132006-03-03 16:47:17 +01003605 .unsol_event = alc_unsol_event,
Takashi Iwaie044c392008-10-27 16:56:24 +01003606#ifdef SND_HDA_NEEDS_RESUME
3607 .resume = alc_resume,
3608#endif
Takashi Iwaicb53c622007-08-10 17:21:45 +02003609#ifdef CONFIG_SND_HDA_POWER_SAVE
3610 .check_power_status = alc_check_power_status,
3611#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07003612};
3613
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003614
3615/*
3616 * Test configuration for debugging
3617 *
3618 * Almost all inputs/outputs are enabled. I/O pins can be configured via
3619 * enum controls.
3620 */
3621#ifdef CONFIG_SND_DEBUG
3622static hda_nid_t alc880_test_dac_nids[4] = {
3623 0x02, 0x03, 0x04, 0x05
3624};
3625
3626static struct hda_input_mux alc880_test_capture_source = {
Takashi Iwaiae6b8132006-03-03 16:47:17 +01003627 .num_items = 7,
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003628 .items = {
3629 { "In-1", 0x0 },
3630 { "In-2", 0x1 },
3631 { "In-3", 0x2 },
3632 { "In-4", 0x3 },
3633 { "CD", 0x4 },
Takashi Iwaiae6b8132006-03-03 16:47:17 +01003634 { "Front", 0x5 },
3635 { "Surround", 0x6 },
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003636 },
3637};
3638
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +01003639static struct hda_channel_mode alc880_test_modes[4] = {
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003640 { 2, NULL },
Takashi Iwaifd2c3262005-05-13 17:18:42 +02003641 { 4, NULL },
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003642 { 6, NULL },
Takashi Iwaifd2c3262005-05-13 17:18:42 +02003643 { 8, NULL },
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003644};
3645
Takashi Iwai9c7f8522006-06-28 15:08:22 +02003646static int alc_test_pin_ctl_info(struct snd_kcontrol *kcontrol,
3647 struct snd_ctl_elem_info *uinfo)
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003648{
3649 static char *texts[] = {
3650 "N/A", "Line Out", "HP Out",
3651 "In Hi-Z", "In 50%", "In Grd", "In 80%", "In 100%"
3652 };
3653 uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
3654 uinfo->count = 1;
3655 uinfo->value.enumerated.items = 8;
3656 if (uinfo->value.enumerated.item >= 8)
3657 uinfo->value.enumerated.item = 7;
3658 strcpy(uinfo->value.enumerated.name, texts[uinfo->value.enumerated.item]);
3659 return 0;
3660}
3661
Takashi Iwai9c7f8522006-06-28 15:08:22 +02003662static int alc_test_pin_ctl_get(struct snd_kcontrol *kcontrol,
3663 struct snd_ctl_elem_value *ucontrol)
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003664{
3665 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
3666 hda_nid_t nid = (hda_nid_t)kcontrol->private_value;
3667 unsigned int pin_ctl, item = 0;
3668
3669 pin_ctl = snd_hda_codec_read(codec, nid, 0,
3670 AC_VERB_GET_PIN_WIDGET_CONTROL, 0);
3671 if (pin_ctl & AC_PINCTL_OUT_EN) {
3672 if (pin_ctl & AC_PINCTL_HP_EN)
3673 item = 2;
3674 else
3675 item = 1;
3676 } else if (pin_ctl & AC_PINCTL_IN_EN) {
3677 switch (pin_ctl & AC_PINCTL_VREFEN) {
3678 case AC_PINCTL_VREF_HIZ: item = 3; break;
3679 case AC_PINCTL_VREF_50: item = 4; break;
3680 case AC_PINCTL_VREF_GRD: item = 5; break;
3681 case AC_PINCTL_VREF_80: item = 6; break;
3682 case AC_PINCTL_VREF_100: item = 7; break;
3683 }
3684 }
3685 ucontrol->value.enumerated.item[0] = item;
3686 return 0;
3687}
3688
Takashi Iwai9c7f8522006-06-28 15:08:22 +02003689static int alc_test_pin_ctl_put(struct snd_kcontrol *kcontrol,
3690 struct snd_ctl_elem_value *ucontrol)
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003691{
3692 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
3693 hda_nid_t nid = (hda_nid_t)kcontrol->private_value;
3694 static unsigned int ctls[] = {
3695 0, AC_PINCTL_OUT_EN, AC_PINCTL_OUT_EN | AC_PINCTL_HP_EN,
3696 AC_PINCTL_IN_EN | AC_PINCTL_VREF_HIZ,
3697 AC_PINCTL_IN_EN | AC_PINCTL_VREF_50,
3698 AC_PINCTL_IN_EN | AC_PINCTL_VREF_GRD,
3699 AC_PINCTL_IN_EN | AC_PINCTL_VREF_80,
3700 AC_PINCTL_IN_EN | AC_PINCTL_VREF_100,
3701 };
3702 unsigned int old_ctl, new_ctl;
3703
3704 old_ctl = snd_hda_codec_read(codec, nid, 0,
3705 AC_VERB_GET_PIN_WIDGET_CONTROL, 0);
3706 new_ctl = ctls[ucontrol->value.enumerated.item[0]];
3707 if (old_ctl != new_ctl) {
Takashi Iwai82beb8f2007-08-10 17:09:26 +02003708 int val;
3709 snd_hda_codec_write_cache(codec, nid, 0,
3710 AC_VERB_SET_PIN_WIDGET_CONTROL,
3711 new_ctl);
Takashi Iwai47fd8302007-08-10 17:11:07 +02003712 val = ucontrol->value.enumerated.item[0] >= 3 ?
3713 HDA_AMP_MUTE : 0;
3714 snd_hda_codec_amp_stereo(codec, nid, HDA_OUTPUT, 0,
3715 HDA_AMP_MUTE, val);
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003716 return 1;
3717 }
3718 return 0;
3719}
3720
Takashi Iwai9c7f8522006-06-28 15:08:22 +02003721static int alc_test_pin_src_info(struct snd_kcontrol *kcontrol,
3722 struct snd_ctl_elem_info *uinfo)
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003723{
3724 static char *texts[] = {
3725 "Front", "Surround", "CLFE", "Side"
3726 };
3727 uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
3728 uinfo->count = 1;
3729 uinfo->value.enumerated.items = 4;
3730 if (uinfo->value.enumerated.item >= 4)
3731 uinfo->value.enumerated.item = 3;
3732 strcpy(uinfo->value.enumerated.name, texts[uinfo->value.enumerated.item]);
3733 return 0;
3734}
3735
Takashi Iwai9c7f8522006-06-28 15:08:22 +02003736static int alc_test_pin_src_get(struct snd_kcontrol *kcontrol,
3737 struct snd_ctl_elem_value *ucontrol)
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003738{
3739 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
3740 hda_nid_t nid = (hda_nid_t)kcontrol->private_value;
3741 unsigned int sel;
3742
3743 sel = snd_hda_codec_read(codec, nid, 0, AC_VERB_GET_CONNECT_SEL, 0);
3744 ucontrol->value.enumerated.item[0] = sel & 3;
3745 return 0;
3746}
3747
Takashi Iwai9c7f8522006-06-28 15:08:22 +02003748static int alc_test_pin_src_put(struct snd_kcontrol *kcontrol,
3749 struct snd_ctl_elem_value *ucontrol)
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003750{
3751 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
3752 hda_nid_t nid = (hda_nid_t)kcontrol->private_value;
3753 unsigned int sel;
3754
3755 sel = snd_hda_codec_read(codec, nid, 0, AC_VERB_GET_CONNECT_SEL, 0) & 3;
3756 if (ucontrol->value.enumerated.item[0] != sel) {
3757 sel = ucontrol->value.enumerated.item[0] & 3;
Takashi Iwai82beb8f2007-08-10 17:09:26 +02003758 snd_hda_codec_write_cache(codec, nid, 0,
3759 AC_VERB_SET_CONNECT_SEL, sel);
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003760 return 1;
3761 }
3762 return 0;
3763}
3764
3765#define PIN_CTL_TEST(xname,nid) { \
3766 .iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
3767 .name = xname, \
3768 .info = alc_test_pin_ctl_info, \
3769 .get = alc_test_pin_ctl_get, \
3770 .put = alc_test_pin_ctl_put, \
3771 .private_value = nid \
3772 }
3773
3774#define PIN_SRC_TEST(xname,nid) { \
3775 .iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
3776 .name = xname, \
3777 .info = alc_test_pin_src_info, \
3778 .get = alc_test_pin_src_get, \
3779 .put = alc_test_pin_src_put, \
3780 .private_value = nid \
3781 }
3782
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01003783static struct snd_kcontrol_new alc880_test_mixer[] = {
Takashi Iwai05acb862005-06-10 19:50:25 +02003784 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
3785 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
3786 HDA_CODEC_VOLUME("CLFE Playback Volume", 0x0e, 0x0, HDA_OUTPUT),
3787 HDA_CODEC_VOLUME("Side Playback Volume", 0x0f, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01003788 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
3789 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
3790 HDA_BIND_MUTE("CLFE Playback Switch", 0x0e, 2, HDA_INPUT),
3791 HDA_BIND_MUTE("Side Playback Switch", 0x0f, 2, HDA_INPUT),
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003792 PIN_CTL_TEST("Front Pin Mode", 0x14),
3793 PIN_CTL_TEST("Surround Pin Mode", 0x15),
3794 PIN_CTL_TEST("CLFE Pin Mode", 0x16),
3795 PIN_CTL_TEST("Side Pin Mode", 0x17),
3796 PIN_CTL_TEST("In-1 Pin Mode", 0x18),
3797 PIN_CTL_TEST("In-2 Pin Mode", 0x19),
3798 PIN_CTL_TEST("In-3 Pin Mode", 0x1a),
3799 PIN_CTL_TEST("In-4 Pin Mode", 0x1b),
3800 PIN_SRC_TEST("In-1 Pin Source", 0x18),
3801 PIN_SRC_TEST("In-2 Pin Source", 0x19),
3802 PIN_SRC_TEST("In-3 Pin Source", 0x1a),
3803 PIN_SRC_TEST("In-4 Pin Source", 0x1b),
3804 HDA_CODEC_VOLUME("In-1 Playback Volume", 0x0b, 0x0, HDA_INPUT),
3805 HDA_CODEC_MUTE("In-1 Playback Switch", 0x0b, 0x0, HDA_INPUT),
3806 HDA_CODEC_VOLUME("In-2 Playback Volume", 0x0b, 0x1, HDA_INPUT),
3807 HDA_CODEC_MUTE("In-2 Playback Switch", 0x0b, 0x1, HDA_INPUT),
3808 HDA_CODEC_VOLUME("In-3 Playback Volume", 0x0b, 0x2, HDA_INPUT),
3809 HDA_CODEC_MUTE("In-3 Playback Switch", 0x0b, 0x2, HDA_INPUT),
3810 HDA_CODEC_VOLUME("In-4 Playback Volume", 0x0b, 0x3, HDA_INPUT),
3811 HDA_CODEC_MUTE("In-4 Playback Switch", 0x0b, 0x3, HDA_INPUT),
3812 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x4, HDA_INPUT),
3813 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x4, HDA_INPUT),
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003814 {
3815 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
3816 .name = "Channel Mode",
Kailang Yangdf694da2005-12-05 19:42:22 +01003817 .info = alc_ch_mode_info,
3818 .get = alc_ch_mode_get,
3819 .put = alc_ch_mode_put,
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003820 },
3821 { } /* end */
3822};
3823
3824static struct hda_verb alc880_test_init_verbs[] = {
3825 /* Unmute inputs of 0x0c - 0x0f */
Takashi Iwai05acb862005-06-10 19:50:25 +02003826 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
3827 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
3828 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
3829 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
3830 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
3831 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
3832 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
3833 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003834 /* Vol output for 0x0c-0x0f */
Takashi Iwai05acb862005-06-10 19:50:25 +02003835 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
3836 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
3837 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
3838 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003839 /* Set output pins 0x14-0x17 */
Takashi Iwai05acb862005-06-10 19:50:25 +02003840 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
3841 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
3842 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
3843 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003844 /* Unmute output pins 0x14-0x17 */
Takashi Iwai05acb862005-06-10 19:50:25 +02003845 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
3846 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
3847 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
3848 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003849 /* Set input pins 0x18-0x1c */
Takashi Iwai16ded522005-06-10 19:58:24 +02003850 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
3851 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Takashi Iwai05acb862005-06-10 19:50:25 +02003852 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
3853 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
3854 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003855 /* Mute input pins 0x18-0x1b */
Takashi Iwai05acb862005-06-10 19:50:25 +02003856 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
3857 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
3858 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
3859 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
Takashi Iwai71fe7b82005-05-25 18:11:40 +02003860 /* ADC set up */
Takashi Iwai05acb862005-06-10 19:50:25 +02003861 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Takashi Iwai71fe7b82005-05-25 18:11:40 +02003862 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
Takashi Iwai05acb862005-06-10 19:50:25 +02003863 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Takashi Iwai71fe7b82005-05-25 18:11:40 +02003864 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
Takashi Iwai05acb862005-06-10 19:50:25 +02003865 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Takashi Iwai71fe7b82005-05-25 18:11:40 +02003866 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
Takashi Iwai05acb862005-06-10 19:50:25 +02003867 /* Analog input/passthru */
3868 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
3869 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
3870 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
3871 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
3872 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003873 { }
3874};
3875#endif
3876
Linus Torvalds1da177e2005-04-16 15:20:36 -07003877/*
3878 */
3879
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003880static const char *alc880_models[ALC880_MODEL_LAST] = {
3881 [ALC880_3ST] = "3stack",
3882 [ALC880_TCL_S700] = "tcl",
3883 [ALC880_3ST_DIG] = "3stack-digout",
3884 [ALC880_CLEVO] = "clevo",
3885 [ALC880_5ST] = "5stack",
3886 [ALC880_5ST_DIG] = "5stack-digout",
3887 [ALC880_W810] = "w810",
3888 [ALC880_Z71V] = "z71v",
3889 [ALC880_6ST] = "6stack",
3890 [ALC880_6ST_DIG] = "6stack-digout",
3891 [ALC880_ASUS] = "asus",
3892 [ALC880_ASUS_W1V] = "asus-w1v",
3893 [ALC880_ASUS_DIG] = "asus-dig",
3894 [ALC880_ASUS_DIG2] = "asus-dig2",
3895 [ALC880_UNIWILL_DIG] = "uniwill",
Tobin Davis2cf9f0f2007-02-07 16:04:25 +01003896 [ALC880_UNIWILL_P53] = "uniwill-p53",
3897 [ALC880_FUJITSU] = "fujitsu",
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003898 [ALC880_F1734] = "F1734",
3899 [ALC880_LG] = "lg",
3900 [ALC880_LG_LW] = "lg-lw",
Takashi Iwaidf99cd32008-04-25 15:25:04 +02003901 [ALC880_MEDION_RIM] = "medion",
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003902#ifdef CONFIG_SND_DEBUG
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003903 [ALC880_TEST] = "test",
Takashi Iwai2fa522b2005-05-12 14:51:12 +02003904#endif
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003905 [ALC880_AUTO] = "auto",
3906};
3907
3908static struct snd_pci_quirk alc880_cfg_tbl[] = {
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003909 SND_PCI_QUIRK(0x1019, 0x0f69, "Coeus G610P", ALC880_W810),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003910 SND_PCI_QUIRK(0x1019, 0xa880, "ECS", ALC880_5ST_DIG),
3911 SND_PCI_QUIRK(0x1019, 0xa884, "Acer APFV", ALC880_6ST),
3912 SND_PCI_QUIRK(0x1025, 0x0070, "ULI", ALC880_3ST_DIG),
3913 SND_PCI_QUIRK(0x1025, 0x0077, "ULI", ALC880_6ST_DIG),
3914 SND_PCI_QUIRK(0x1025, 0x0078, "ULI", ALC880_6ST_DIG),
3915 SND_PCI_QUIRK(0x1025, 0x0087, "ULI", ALC880_6ST_DIG),
3916 SND_PCI_QUIRK(0x1025, 0xe309, "ULI", ALC880_3ST_DIG),
3917 SND_PCI_QUIRK(0x1025, 0xe310, "ULI", ALC880_3ST),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003918 SND_PCI_QUIRK(0x1039, 0x1234, NULL, ALC880_6ST_DIG),
3919 SND_PCI_QUIRK(0x103c, 0x2a09, "HP", ALC880_5ST),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003920 SND_PCI_QUIRK(0x1043, 0x10b3, "ASUS W1V", ALC880_ASUS_W1V),
3921 SND_PCI_QUIRK(0x1043, 0x10c2, "ASUS W6A", ALC880_ASUS_DIG),
3922 SND_PCI_QUIRK(0x1043, 0x10c3, "ASUS Wxx", ALC880_ASUS_DIG),
3923 SND_PCI_QUIRK(0x1043, 0x1113, "ASUS", ALC880_ASUS_DIG),
3924 SND_PCI_QUIRK(0x1043, 0x1123, "ASUS", ALC880_ASUS_DIG),
3925 SND_PCI_QUIRK(0x1043, 0x1173, "ASUS", ALC880_ASUS_DIG),
3926 SND_PCI_QUIRK(0x1043, 0x1964, "ASUS Z71V", ALC880_Z71V),
3927 /* SND_PCI_QUIRK(0x1043, 0x1964, "ASUS", ALC880_ASUS_DIG), */
3928 SND_PCI_QUIRK(0x1043, 0x1973, "ASUS", ALC880_ASUS_DIG),
3929 SND_PCI_QUIRK(0x1043, 0x19b3, "ASUS", ALC880_ASUS_DIG),
Travis Place186c3112008-05-20 11:54:41 +02003930 SND_PCI_QUIRK(0x1043, 0x814e, "ASUS P5GD1 w/SPDIF", ALC880_6ST_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003931 SND_PCI_QUIRK(0x1043, 0x8181, "ASUS P4GPL", ALC880_ASUS_DIG),
3932 SND_PCI_QUIRK(0x1043, 0x8196, "ASUS P5GD1", ALC880_6ST),
3933 SND_PCI_QUIRK(0x1043, 0x81b4, "ASUS", ALC880_6ST),
Takashi Iwaidea0a502009-02-09 17:14:52 +01003934 SND_PCI_QUIRK_VENDOR(0x1043, "ASUS", ALC880_ASUS), /* default ASUS */
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003935 SND_PCI_QUIRK(0x104d, 0x81a0, "Sony", ALC880_3ST),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003936 SND_PCI_QUIRK(0x104d, 0x81d6, "Sony", ALC880_3ST),
3937 SND_PCI_QUIRK(0x107b, 0x3032, "Gateway", ALC880_5ST),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003938 SND_PCI_QUIRK(0x107b, 0x3033, "Gateway", ALC880_5ST),
3939 SND_PCI_QUIRK(0x107b, 0x4039, "Gateway", ALC880_5ST),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003940 SND_PCI_QUIRK(0x1297, 0xc790, "Shuttle ST20G5", ALC880_6ST_DIG),
3941 SND_PCI_QUIRK(0x1458, 0xa102, "Gigabyte K8", ALC880_6ST_DIG),
3942 SND_PCI_QUIRK(0x1462, 0x1150, "MSI", ALC880_6ST_DIG),
3943 SND_PCI_QUIRK(0x1509, 0x925d, "FIC P4M", ALC880_6ST_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003944 SND_PCI_QUIRK(0x1558, 0x0520, "Clevo m520G", ALC880_CLEVO),
3945 SND_PCI_QUIRK(0x1558, 0x0660, "Clevo m655n", ALC880_CLEVO),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003946 SND_PCI_QUIRK(0x1558, 0x5401, "ASUS", ALC880_ASUS_DIG2),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003947 SND_PCI_QUIRK(0x1565, 0x8202, "Biostar", ALC880_5ST_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003948 SND_PCI_QUIRK(0x1584, 0x9050, "Uniwill", ALC880_UNIWILL_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003949 SND_PCI_QUIRK(0x1584, 0x9054, "Uniwlll", ALC880_F1734),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003950 SND_PCI_QUIRK(0x1584, 0x9070, "Uniwill", ALC880_UNIWILL),
3951 SND_PCI_QUIRK(0x1584, 0x9077, "Uniwill P53", ALC880_UNIWILL_P53),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003952 SND_PCI_QUIRK(0x161f, 0x203d, "W810", ALC880_W810),
Takashi Iwaidf99cd32008-04-25 15:25:04 +02003953 SND_PCI_QUIRK(0x161f, 0x205d, "Medion Rim 2150", ALC880_MEDION_RIM),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003954 SND_PCI_QUIRK(0x1695, 0x400d, "EPoX", ALC880_5ST_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003955 SND_PCI_QUIRK(0x1695, 0x4012, "EPox EP-5LDA", ALC880_5ST_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003956 SND_PCI_QUIRK(0x1734, 0x107c, "FSC F1734", ALC880_F1734),
Travis Place1d116042008-06-23 11:42:30 +02003957 SND_PCI_QUIRK(0x1734, 0x1094, "FSC Amilo M1451G", ALC880_FUJITSU),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003958 SND_PCI_QUIRK(0x1734, 0x10ac, "FSC", ALC880_UNIWILL),
Tobin Davis2cf9f0f2007-02-07 16:04:25 +01003959 SND_PCI_QUIRK(0x1734, 0x10b0, "Fujitsu", ALC880_FUJITSU),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003960 SND_PCI_QUIRK(0x1854, 0x0018, "LG LW20", ALC880_LG_LW),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003961 SND_PCI_QUIRK(0x1854, 0x003b, "LG", ALC880_LG),
3962 SND_PCI_QUIRK(0x1854, 0x0068, "LG w1", ALC880_LG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003963 SND_PCI_QUIRK(0x1854, 0x0077, "LG LW25", ALC880_LG_LW),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003964 SND_PCI_QUIRK(0x19db, 0x4188, "TCL S700", ALC880_TCL_S700),
3965 SND_PCI_QUIRK(0x2668, 0x8086, NULL, ALC880_6ST_DIG), /* broken BIOS */
3966 SND_PCI_QUIRK(0x8086, 0x2668, NULL, ALC880_6ST_DIG),
3967 SND_PCI_QUIRK(0x8086, 0xa100, "Intel mobo", ALC880_5ST_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003968 SND_PCI_QUIRK(0x8086, 0xd400, "Intel mobo", ALC880_5ST_DIG),
3969 SND_PCI_QUIRK(0x8086, 0xd401, "Intel mobo", ALC880_5ST_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003970 SND_PCI_QUIRK(0x8086, 0xd402, "Intel mobo", ALC880_3ST_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003971 SND_PCI_QUIRK(0x8086, 0xe224, "Intel mobo", ALC880_5ST_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003972 SND_PCI_QUIRK(0x8086, 0xe305, "Intel mobo", ALC880_3ST_DIG),
3973 SND_PCI_QUIRK(0x8086, 0xe308, "Intel mobo", ALC880_3ST_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01003974 SND_PCI_QUIRK(0x8086, 0xe400, "Intel mobo", ALC880_5ST_DIG),
3975 SND_PCI_QUIRK(0x8086, 0xe401, "Intel mobo", ALC880_5ST_DIG),
3976 SND_PCI_QUIRK(0x8086, 0xe402, "Intel mobo", ALC880_5ST_DIG),
Takashi Iwaidea0a502009-02-09 17:14:52 +01003977 /* default Intel */
3978 SND_PCI_QUIRK_VENDOR(0x8086, "Intel mobo", ALC880_3ST),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01003979 SND_PCI_QUIRK(0xa0a0, 0x0560, "AOpen i915GMm-HFS", ALC880_5ST_DIG),
3980 SND_PCI_QUIRK(0xe803, 0x1019, NULL, ALC880_6ST_DIG),
Linus Torvalds1da177e2005-04-16 15:20:36 -07003981 {}
3982};
3983
Takashi Iwai16ded522005-06-10 19:58:24 +02003984/*
Kailang Yangdf694da2005-12-05 19:42:22 +01003985 * ALC880 codec presets
Takashi Iwai16ded522005-06-10 19:58:24 +02003986 */
Takashi Iwai16ded522005-06-10 19:58:24 +02003987static struct alc_config_preset alc880_presets[] = {
3988 [ALC880_3ST] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02003989 .mixers = { alc880_three_stack_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02003990 .init_verbs = { alc880_volume_init_verbs,
3991 alc880_pin_3stack_init_verbs },
Takashi Iwai16ded522005-06-10 19:58:24 +02003992 .num_dacs = ARRAY_SIZE(alc880_dac_nids),
Takashi Iwai16ded522005-06-10 19:58:24 +02003993 .dac_nids = alc880_dac_nids,
Takashi Iwai16ded522005-06-10 19:58:24 +02003994 .num_channel_mode = ARRAY_SIZE(alc880_threestack_modes),
3995 .channel_mode = alc880_threestack_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02003996 .need_dac_fix = 1,
Takashi Iwai16ded522005-06-10 19:58:24 +02003997 .input_mux = &alc880_capture_source,
3998 },
3999 [ALC880_3ST_DIG] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004000 .mixers = { alc880_three_stack_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004001 .init_verbs = { alc880_volume_init_verbs,
4002 alc880_pin_3stack_init_verbs },
Takashi Iwai16ded522005-06-10 19:58:24 +02004003 .num_dacs = ARRAY_SIZE(alc880_dac_nids),
Takashi Iwai16ded522005-06-10 19:58:24 +02004004 .dac_nids = alc880_dac_nids,
4005 .dig_out_nid = ALC880_DIGOUT_NID,
Takashi Iwai16ded522005-06-10 19:58:24 +02004006 .num_channel_mode = ARRAY_SIZE(alc880_threestack_modes),
4007 .channel_mode = alc880_threestack_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02004008 .need_dac_fix = 1,
Takashi Iwai16ded522005-06-10 19:58:24 +02004009 .input_mux = &alc880_capture_source,
4010 },
Kailang Yangdf694da2005-12-05 19:42:22 +01004011 [ALC880_TCL_S700] = {
4012 .mixers = { alc880_tcl_s700_mixer },
4013 .init_verbs = { alc880_volume_init_verbs,
4014 alc880_pin_tcl_S700_init_verbs,
4015 alc880_gpio2_init_verbs },
4016 .num_dacs = ARRAY_SIZE(alc880_dac_nids),
4017 .dac_nids = alc880_dac_nids,
Takashi Iwaif9e336f2008-10-31 16:37:07 +01004018 .adc_nids = alc880_adc_nids_alt, /* FIXME: correct? */
4019 .num_adc_nids = 1, /* single ADC */
Kailang Yangdf694da2005-12-05 19:42:22 +01004020 .hp_nid = 0x03,
4021 .num_channel_mode = ARRAY_SIZE(alc880_2_jack_modes),
4022 .channel_mode = alc880_2_jack_modes,
4023 .input_mux = &alc880_capture_source,
4024 },
Takashi Iwai16ded522005-06-10 19:58:24 +02004025 [ALC880_5ST] = {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004026 .mixers = { alc880_three_stack_mixer,
4027 alc880_five_stack_mixer},
4028 .init_verbs = { alc880_volume_init_verbs,
4029 alc880_pin_5stack_init_verbs },
Takashi Iwai16ded522005-06-10 19:58:24 +02004030 .num_dacs = ARRAY_SIZE(alc880_dac_nids),
4031 .dac_nids = alc880_dac_nids,
Takashi Iwai16ded522005-06-10 19:58:24 +02004032 .num_channel_mode = ARRAY_SIZE(alc880_fivestack_modes),
4033 .channel_mode = alc880_fivestack_modes,
4034 .input_mux = &alc880_capture_source,
4035 },
4036 [ALC880_5ST_DIG] = {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004037 .mixers = { alc880_three_stack_mixer,
4038 alc880_five_stack_mixer },
4039 .init_verbs = { alc880_volume_init_verbs,
4040 alc880_pin_5stack_init_verbs },
Takashi Iwai16ded522005-06-10 19:58:24 +02004041 .num_dacs = ARRAY_SIZE(alc880_dac_nids),
4042 .dac_nids = alc880_dac_nids,
4043 .dig_out_nid = ALC880_DIGOUT_NID,
Takashi Iwai16ded522005-06-10 19:58:24 +02004044 .num_channel_mode = ARRAY_SIZE(alc880_fivestack_modes),
4045 .channel_mode = alc880_fivestack_modes,
4046 .input_mux = &alc880_capture_source,
4047 },
Takashi Iwaib6482d42005-06-27 15:32:43 +02004048 [ALC880_6ST] = {
4049 .mixers = { alc880_six_stack_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004050 .init_verbs = { alc880_volume_init_verbs,
4051 alc880_pin_6stack_init_verbs },
Takashi Iwaib6482d42005-06-27 15:32:43 +02004052 .num_dacs = ARRAY_SIZE(alc880_6st_dac_nids),
4053 .dac_nids = alc880_6st_dac_nids,
4054 .num_channel_mode = ARRAY_SIZE(alc880_sixstack_modes),
4055 .channel_mode = alc880_sixstack_modes,
4056 .input_mux = &alc880_6stack_capture_source,
4057 },
Takashi Iwai16ded522005-06-10 19:58:24 +02004058 [ALC880_6ST_DIG] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004059 .mixers = { alc880_six_stack_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004060 .init_verbs = { alc880_volume_init_verbs,
4061 alc880_pin_6stack_init_verbs },
Takashi Iwai16ded522005-06-10 19:58:24 +02004062 .num_dacs = ARRAY_SIZE(alc880_6st_dac_nids),
4063 .dac_nids = alc880_6st_dac_nids,
4064 .dig_out_nid = ALC880_DIGOUT_NID,
Takashi Iwai16ded522005-06-10 19:58:24 +02004065 .num_channel_mode = ARRAY_SIZE(alc880_sixstack_modes),
4066 .channel_mode = alc880_sixstack_modes,
4067 .input_mux = &alc880_6stack_capture_source,
4068 },
4069 [ALC880_W810] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004070 .mixers = { alc880_w810_base_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004071 .init_verbs = { alc880_volume_init_verbs,
4072 alc880_pin_w810_init_verbs,
Takashi Iwaib0af0de2005-06-21 14:49:19 +02004073 alc880_gpio2_init_verbs },
Takashi Iwai16ded522005-06-10 19:58:24 +02004074 .num_dacs = ARRAY_SIZE(alc880_w810_dac_nids),
4075 .dac_nids = alc880_w810_dac_nids,
4076 .dig_out_nid = ALC880_DIGOUT_NID,
Takashi Iwai16ded522005-06-10 19:58:24 +02004077 .num_channel_mode = ARRAY_SIZE(alc880_w810_modes),
4078 .channel_mode = alc880_w810_modes,
4079 .input_mux = &alc880_capture_source,
4080 },
4081 [ALC880_Z71V] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004082 .mixers = { alc880_z71v_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004083 .init_verbs = { alc880_volume_init_verbs,
4084 alc880_pin_z71v_init_verbs },
Takashi Iwai16ded522005-06-10 19:58:24 +02004085 .num_dacs = ARRAY_SIZE(alc880_z71v_dac_nids),
4086 .dac_nids = alc880_z71v_dac_nids,
4087 .dig_out_nid = ALC880_DIGOUT_NID,
4088 .hp_nid = 0x03,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004089 .num_channel_mode = ARRAY_SIZE(alc880_2_jack_modes),
4090 .channel_mode = alc880_2_jack_modes,
Takashi Iwai16ded522005-06-10 19:58:24 +02004091 .input_mux = &alc880_capture_source,
4092 },
4093 [ALC880_F1734] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004094 .mixers = { alc880_f1734_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004095 .init_verbs = { alc880_volume_init_verbs,
4096 alc880_pin_f1734_init_verbs },
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004097 .num_dacs = ARRAY_SIZE(alc880_f1734_dac_nids),
4098 .dac_nids = alc880_f1734_dac_nids,
4099 .hp_nid = 0x02,
4100 .num_channel_mode = ARRAY_SIZE(alc880_2_jack_modes),
4101 .channel_mode = alc880_2_jack_modes,
Takashi Iwai937b4162008-02-11 14:52:36 +01004102 .input_mux = &alc880_f1734_capture_source,
4103 .unsol_event = alc880_uniwill_p53_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02004104 .setup = alc880_uniwill_p53_setup,
4105 .init_hook = alc_automute_amp,
Takashi Iwai16ded522005-06-10 19:58:24 +02004106 },
4107 [ALC880_ASUS] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004108 .mixers = { alc880_asus_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004109 .init_verbs = { alc880_volume_init_verbs,
4110 alc880_pin_asus_init_verbs,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004111 alc880_gpio1_init_verbs },
4112 .num_dacs = ARRAY_SIZE(alc880_asus_dac_nids),
4113 .dac_nids = alc880_asus_dac_nids,
4114 .num_channel_mode = ARRAY_SIZE(alc880_asus_modes),
4115 .channel_mode = alc880_asus_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02004116 .need_dac_fix = 1,
Takashi Iwai16ded522005-06-10 19:58:24 +02004117 .input_mux = &alc880_capture_source,
4118 },
4119 [ALC880_ASUS_DIG] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004120 .mixers = { alc880_asus_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004121 .init_verbs = { alc880_volume_init_verbs,
4122 alc880_pin_asus_init_verbs,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004123 alc880_gpio1_init_verbs },
4124 .num_dacs = ARRAY_SIZE(alc880_asus_dac_nids),
4125 .dac_nids = alc880_asus_dac_nids,
Takashi Iwai16ded522005-06-10 19:58:24 +02004126 .dig_out_nid = ALC880_DIGOUT_NID,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004127 .num_channel_mode = ARRAY_SIZE(alc880_asus_modes),
4128 .channel_mode = alc880_asus_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02004129 .need_dac_fix = 1,
Takashi Iwai16ded522005-06-10 19:58:24 +02004130 .input_mux = &alc880_capture_source,
4131 },
Kailang Yangdf694da2005-12-05 19:42:22 +01004132 [ALC880_ASUS_DIG2] = {
4133 .mixers = { alc880_asus_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004134 .init_verbs = { alc880_volume_init_verbs,
4135 alc880_pin_asus_init_verbs,
Kailang Yangdf694da2005-12-05 19:42:22 +01004136 alc880_gpio2_init_verbs }, /* use GPIO2 */
4137 .num_dacs = ARRAY_SIZE(alc880_asus_dac_nids),
4138 .dac_nids = alc880_asus_dac_nids,
4139 .dig_out_nid = ALC880_DIGOUT_NID,
4140 .num_channel_mode = ARRAY_SIZE(alc880_asus_modes),
4141 .channel_mode = alc880_asus_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02004142 .need_dac_fix = 1,
Kailang Yangdf694da2005-12-05 19:42:22 +01004143 .input_mux = &alc880_capture_source,
4144 },
Takashi Iwai16ded522005-06-10 19:58:24 +02004145 [ALC880_ASUS_W1V] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004146 .mixers = { alc880_asus_mixer, alc880_asus_w1v_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004147 .init_verbs = { alc880_volume_init_verbs,
4148 alc880_pin_asus_init_verbs,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004149 alc880_gpio1_init_verbs },
4150 .num_dacs = ARRAY_SIZE(alc880_asus_dac_nids),
4151 .dac_nids = alc880_asus_dac_nids,
Takashi Iwai16ded522005-06-10 19:58:24 +02004152 .dig_out_nid = ALC880_DIGOUT_NID,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004153 .num_channel_mode = ARRAY_SIZE(alc880_asus_modes),
4154 .channel_mode = alc880_asus_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02004155 .need_dac_fix = 1,
Takashi Iwai16ded522005-06-10 19:58:24 +02004156 .input_mux = &alc880_capture_source,
4157 },
4158 [ALC880_UNIWILL_DIG] = {
Takashi Iwai45bdd1c2009-02-06 16:11:25 +01004159 .mixers = { alc880_asus_mixer },
Kailang Yangccc656c2006-10-17 12:32:26 +02004160 .init_verbs = { alc880_volume_init_verbs,
4161 alc880_pin_asus_init_verbs },
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004162 .num_dacs = ARRAY_SIZE(alc880_asus_dac_nids),
4163 .dac_nids = alc880_asus_dac_nids,
Takashi Iwai16ded522005-06-10 19:58:24 +02004164 .dig_out_nid = ALC880_DIGOUT_NID,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004165 .num_channel_mode = ARRAY_SIZE(alc880_asus_modes),
4166 .channel_mode = alc880_asus_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02004167 .need_dac_fix = 1,
Takashi Iwai16ded522005-06-10 19:58:24 +02004168 .input_mux = &alc880_capture_source,
4169 },
Kailang Yangccc656c2006-10-17 12:32:26 +02004170 [ALC880_UNIWILL] = {
4171 .mixers = { alc880_uniwill_mixer },
4172 .init_verbs = { alc880_volume_init_verbs,
4173 alc880_uniwill_init_verbs },
4174 .num_dacs = ARRAY_SIZE(alc880_asus_dac_nids),
4175 .dac_nids = alc880_asus_dac_nids,
4176 .dig_out_nid = ALC880_DIGOUT_NID,
4177 .num_channel_mode = ARRAY_SIZE(alc880_threestack_modes),
4178 .channel_mode = alc880_threestack_modes,
4179 .need_dac_fix = 1,
4180 .input_mux = &alc880_capture_source,
4181 .unsol_event = alc880_uniwill_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02004182 .setup = alc880_uniwill_setup,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02004183 .init_hook = alc880_uniwill_init_hook,
Kailang Yangccc656c2006-10-17 12:32:26 +02004184 },
4185 [ALC880_UNIWILL_P53] = {
4186 .mixers = { alc880_uniwill_p53_mixer },
4187 .init_verbs = { alc880_volume_init_verbs,
4188 alc880_uniwill_p53_init_verbs },
4189 .num_dacs = ARRAY_SIZE(alc880_asus_dac_nids),
4190 .dac_nids = alc880_asus_dac_nids,
4191 .num_channel_mode = ARRAY_SIZE(alc880_w810_modes),
Tobin Davis2cf9f0f2007-02-07 16:04:25 +01004192 .channel_mode = alc880_threestack_modes,
4193 .input_mux = &alc880_capture_source,
4194 .unsol_event = alc880_uniwill_p53_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02004195 .setup = alc880_uniwill_p53_setup,
4196 .init_hook = alc_automute_amp,
Tobin Davis2cf9f0f2007-02-07 16:04:25 +01004197 },
4198 [ALC880_FUJITSU] = {
Takashi Iwai45bdd1c2009-02-06 16:11:25 +01004199 .mixers = { alc880_fujitsu_mixer },
Tobin Davis2cf9f0f2007-02-07 16:04:25 +01004200 .init_verbs = { alc880_volume_init_verbs,
4201 alc880_uniwill_p53_init_verbs,
4202 alc880_beep_init_verbs },
4203 .num_dacs = ARRAY_SIZE(alc880_dac_nids),
4204 .dac_nids = alc880_dac_nids,
Takashi Iwaid53d7d92007-08-20 15:20:02 +02004205 .dig_out_nid = ALC880_DIGOUT_NID,
Tobin Davis2cf9f0f2007-02-07 16:04:25 +01004206 .num_channel_mode = ARRAY_SIZE(alc880_2_jack_modes),
4207 .channel_mode = alc880_2_jack_modes,
Kailang Yangccc656c2006-10-17 12:32:26 +02004208 .input_mux = &alc880_capture_source,
4209 .unsol_event = alc880_uniwill_p53_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02004210 .setup = alc880_uniwill_p53_setup,
4211 .init_hook = alc_automute_amp,
Kailang Yangccc656c2006-10-17 12:32:26 +02004212 },
Kailang Yangdf694da2005-12-05 19:42:22 +01004213 [ALC880_CLEVO] = {
4214 .mixers = { alc880_three_stack_mixer },
4215 .init_verbs = { alc880_volume_init_verbs,
4216 alc880_pin_clevo_init_verbs },
4217 .num_dacs = ARRAY_SIZE(alc880_dac_nids),
4218 .dac_nids = alc880_dac_nids,
4219 .hp_nid = 0x03,
4220 .num_channel_mode = ARRAY_SIZE(alc880_threestack_modes),
4221 .channel_mode = alc880_threestack_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02004222 .need_dac_fix = 1,
Kailang Yangdf694da2005-12-05 19:42:22 +01004223 .input_mux = &alc880_capture_source,
4224 },
Takashi Iwaiae6b8132006-03-03 16:47:17 +01004225 [ALC880_LG] = {
4226 .mixers = { alc880_lg_mixer },
4227 .init_verbs = { alc880_volume_init_verbs,
4228 alc880_lg_init_verbs },
4229 .num_dacs = ARRAY_SIZE(alc880_lg_dac_nids),
4230 .dac_nids = alc880_lg_dac_nids,
4231 .dig_out_nid = ALC880_DIGOUT_NID,
4232 .num_channel_mode = ARRAY_SIZE(alc880_lg_ch_modes),
4233 .channel_mode = alc880_lg_ch_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02004234 .need_dac_fix = 1,
Takashi Iwaiae6b8132006-03-03 16:47:17 +01004235 .input_mux = &alc880_lg_capture_source,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02004236 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02004237 .setup = alc880_lg_setup,
4238 .init_hook = alc_automute_amp,
Takashi Iwaicb53c622007-08-10 17:21:45 +02004239#ifdef CONFIG_SND_HDA_POWER_SAVE
4240 .loopbacks = alc880_lg_loopbacks,
4241#endif
Takashi Iwaiae6b8132006-03-03 16:47:17 +01004242 },
Takashi Iwaid6815182006-03-23 16:06:23 +01004243 [ALC880_LG_LW] = {
4244 .mixers = { alc880_lg_lw_mixer },
4245 .init_verbs = { alc880_volume_init_verbs,
4246 alc880_lg_lw_init_verbs },
Claudio Matsuoka0a8c5da2007-07-04 15:17:38 +02004247 .num_dacs = ARRAY_SIZE(alc880_dac_nids),
Takashi Iwaid6815182006-03-23 16:06:23 +01004248 .dac_nids = alc880_dac_nids,
4249 .dig_out_nid = ALC880_DIGOUT_NID,
Claudio Matsuoka0a8c5da2007-07-04 15:17:38 +02004250 .num_channel_mode = ARRAY_SIZE(alc880_lg_lw_modes),
4251 .channel_mode = alc880_lg_lw_modes,
Takashi Iwaid6815182006-03-23 16:06:23 +01004252 .input_mux = &alc880_lg_lw_capture_source,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02004253 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02004254 .setup = alc880_lg_lw_setup,
4255 .init_hook = alc_automute_amp,
Takashi Iwaid6815182006-03-23 16:06:23 +01004256 },
Takashi Iwaidf99cd32008-04-25 15:25:04 +02004257 [ALC880_MEDION_RIM] = {
4258 .mixers = { alc880_medion_rim_mixer },
4259 .init_verbs = { alc880_volume_init_verbs,
4260 alc880_medion_rim_init_verbs,
4261 alc_gpio2_init_verbs },
4262 .num_dacs = ARRAY_SIZE(alc880_dac_nids),
4263 .dac_nids = alc880_dac_nids,
4264 .dig_out_nid = ALC880_DIGOUT_NID,
4265 .num_channel_mode = ARRAY_SIZE(alc880_2_jack_modes),
4266 .channel_mode = alc880_2_jack_modes,
4267 .input_mux = &alc880_medion_rim_capture_source,
4268 .unsol_event = alc880_medion_rim_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02004269 .setup = alc880_medion_rim_setup,
4270 .init_hook = alc880_medion_rim_automute,
Takashi Iwaidf99cd32008-04-25 15:25:04 +02004271 },
Takashi Iwai16ded522005-06-10 19:58:24 +02004272#ifdef CONFIG_SND_DEBUG
4273 [ALC880_TEST] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004274 .mixers = { alc880_test_mixer },
4275 .init_verbs = { alc880_test_init_verbs },
Takashi Iwai16ded522005-06-10 19:58:24 +02004276 .num_dacs = ARRAY_SIZE(alc880_test_dac_nids),
4277 .dac_nids = alc880_test_dac_nids,
4278 .dig_out_nid = ALC880_DIGOUT_NID,
Takashi Iwai16ded522005-06-10 19:58:24 +02004279 .num_channel_mode = ARRAY_SIZE(alc880_test_modes),
4280 .channel_mode = alc880_test_modes,
4281 .input_mux = &alc880_test_capture_source,
4282 },
4283#endif
4284};
4285
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004286/*
4287 * Automatic parse of I/O pins from the BIOS configuration
4288 */
4289
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004290enum {
4291 ALC_CTL_WIDGET_VOL,
4292 ALC_CTL_WIDGET_MUTE,
4293 ALC_CTL_BIND_MUTE,
4294};
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01004295static struct snd_kcontrol_new alc880_control_templates[] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004296 HDA_CODEC_VOLUME(NULL, 0, 0, 0),
4297 HDA_CODEC_MUTE(NULL, 0, 0, 0),
Takashi Iwai985be542005-11-02 18:26:49 +01004298 HDA_BIND_MUTE(NULL, 0, 0, 0),
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004299};
4300
4301/* add dynamic controls */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004302static int add_control(struct alc_spec *spec, int type, const char *name,
4303 unsigned long val)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004304{
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01004305 struct snd_kcontrol_new *knew;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004306
Takashi Iwai603c4012008-07-30 15:01:44 +02004307 snd_array_init(&spec->kctls, sizeof(*knew), 32);
4308 knew = snd_array_new(&spec->kctls);
4309 if (!knew)
4310 return -ENOMEM;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004311 *knew = alc880_control_templates[type];
Paulo Marques543537b2005-06-23 00:09:02 -07004312 knew->name = kstrdup(name, GFP_KERNEL);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004313 if (!knew->name)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004314 return -ENOMEM;
Jaroslav Kysela4d02d1b2009-11-12 10:15:48 +01004315 if (get_amp_nid_(val))
Takashi Iwai9c96fa52009-11-16 11:25:33 +01004316 knew->subdevice = HDA_SUBDEV_NID_FLAG | get_amp_nid_(val);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004317 knew->private_value = val;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004318 return 0;
4319}
4320
Takashi Iwai0afe5f82009-10-02 09:20:00 +02004321static int add_control_with_pfx(struct alc_spec *spec, int type,
4322 const char *pfx, const char *dir,
4323 const char *sfx, unsigned long val)
4324{
4325 char name[32];
4326 snprintf(name, sizeof(name), "%s %s %s", pfx, dir, sfx);
4327 return add_control(spec, type, name, val);
4328}
4329
4330#define add_pb_vol_ctrl(spec, type, pfx, val) \
4331 add_control_with_pfx(spec, type, pfx, "Playback", "Volume", val)
4332#define add_pb_sw_ctrl(spec, type, pfx, val) \
4333 add_control_with_pfx(spec, type, pfx, "Playback", "Switch", val)
4334
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004335#define alc880_is_fixed_pin(nid) ((nid) >= 0x14 && (nid) <= 0x17)
4336#define alc880_fixed_pin_idx(nid) ((nid) - 0x14)
4337#define alc880_is_multi_pin(nid) ((nid) >= 0x18)
4338#define alc880_multi_pin_idx(nid) ((nid) - 0x18)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004339#define alc880_idx_to_dac(nid) ((nid) + 0x02)
4340#define alc880_dac_to_idx(nid) ((nid) - 0x02)
4341#define alc880_idx_to_mixer(nid) ((nid) + 0x0c)
4342#define alc880_idx_to_selector(nid) ((nid) + 0x10)
4343#define ALC880_PIN_CD_NID 0x1c
4344
4345/* fill in the dac_nids table from the parsed pin configuration */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004346static int alc880_auto_fill_dac_nids(struct alc_spec *spec,
4347 const struct auto_pin_cfg *cfg)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004348{
4349 hda_nid_t nid;
4350 int assigned[4];
4351 int i, j;
4352
4353 memset(assigned, 0, sizeof(assigned));
Takashi Iwaib0af0de2005-06-21 14:49:19 +02004354 spec->multiout.dac_nids = spec->private_dac_nids;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004355
4356 /* check the pins hardwired to audio widget */
4357 for (i = 0; i < cfg->line_outs; i++) {
4358 nid = cfg->line_out_pins[i];
4359 if (alc880_is_fixed_pin(nid)) {
4360 int idx = alc880_fixed_pin_idx(nid);
Libin Yang5014f192005-11-23 15:48:36 +01004361 spec->multiout.dac_nids[i] = alc880_idx_to_dac(idx);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004362 assigned[idx] = 1;
4363 }
4364 }
4365 /* left pins can be connect to any audio widget */
4366 for (i = 0; i < cfg->line_outs; i++) {
4367 nid = cfg->line_out_pins[i];
4368 if (alc880_is_fixed_pin(nid))
4369 continue;
4370 /* search for an empty channel */
4371 for (j = 0; j < cfg->line_outs; j++) {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004372 if (!assigned[j]) {
4373 spec->multiout.dac_nids[i] =
4374 alc880_idx_to_dac(j);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004375 assigned[j] = 1;
4376 break;
4377 }
4378 }
4379 }
4380 spec->multiout.num_dacs = cfg->line_outs;
4381 return 0;
4382}
4383
4384/* add playback controls from the parsed DAC table */
Kailang Yangdf694da2005-12-05 19:42:22 +01004385static int alc880_auto_create_multi_out_ctls(struct alc_spec *spec,
4386 const struct auto_pin_cfg *cfg)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004387{
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004388 static const char *chname[4] = {
4389 "Front", "Surround", NULL /*CLFE*/, "Side"
4390 };
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004391 hda_nid_t nid;
4392 int i, err;
4393
4394 for (i = 0; i < cfg->line_outs; i++) {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004395 if (!spec->multiout.dac_nids[i])
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004396 continue;
4397 nid = alc880_idx_to_mixer(alc880_dac_to_idx(spec->multiout.dac_nids[i]));
4398 if (i == 2) {
4399 /* Center/LFE */
Takashi Iwai0afe5f82009-10-02 09:20:00 +02004400 err = add_pb_vol_ctrl(spec, ALC_CTL_WIDGET_VOL,
4401 "Center",
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004402 HDA_COMPOSE_AMP_VAL(nid, 1, 0,
4403 HDA_OUTPUT));
4404 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004405 return err;
Takashi Iwai0afe5f82009-10-02 09:20:00 +02004406 err = add_pb_vol_ctrl(spec, ALC_CTL_WIDGET_VOL,
4407 "LFE",
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004408 HDA_COMPOSE_AMP_VAL(nid, 2, 0,
4409 HDA_OUTPUT));
4410 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004411 return err;
Takashi Iwai0afe5f82009-10-02 09:20:00 +02004412 err = add_pb_sw_ctrl(spec, ALC_CTL_BIND_MUTE,
4413 "Center",
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004414 HDA_COMPOSE_AMP_VAL(nid, 1, 2,
4415 HDA_INPUT));
4416 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004417 return err;
Takashi Iwai0afe5f82009-10-02 09:20:00 +02004418 err = add_pb_sw_ctrl(spec, ALC_CTL_BIND_MUTE,
4419 "LFE",
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004420 HDA_COMPOSE_AMP_VAL(nid, 2, 2,
4421 HDA_INPUT));
4422 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004423 return err;
4424 } else {
Takashi Iwaicb162b62009-08-25 16:05:03 +02004425 const char *pfx;
4426 if (cfg->line_outs == 1 &&
4427 cfg->line_out_type == AUTO_PIN_SPEAKER_OUT)
4428 pfx = "Speaker";
4429 else
4430 pfx = chname[i];
Takashi Iwai0afe5f82009-10-02 09:20:00 +02004431 err = add_pb_vol_ctrl(spec, ALC_CTL_WIDGET_VOL, pfx,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004432 HDA_COMPOSE_AMP_VAL(nid, 3, 0,
4433 HDA_OUTPUT));
4434 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004435 return err;
Takashi Iwai0afe5f82009-10-02 09:20:00 +02004436 err = add_pb_sw_ctrl(spec, ALC_CTL_BIND_MUTE, pfx,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004437 HDA_COMPOSE_AMP_VAL(nid, 3, 2,
4438 HDA_INPUT));
4439 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004440 return err;
4441 }
4442 }
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004443 return 0;
4444}
4445
Takashi Iwai8d88bc32005-11-17 11:09:23 +01004446/* add playback controls for speaker and HP outputs */
4447static int alc880_auto_create_extra_out(struct alc_spec *spec, hda_nid_t pin,
4448 const char *pfx)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004449{
4450 hda_nid_t nid;
4451 int err;
4452
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004453 if (!pin)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004454 return 0;
4455
4456 if (alc880_is_fixed_pin(pin)) {
4457 nid = alc880_idx_to_dac(alc880_fixed_pin_idx(pin));
Takashi Iwai82bc9552006-03-21 11:24:42 +01004458 /* specify the DAC as the extra output */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004459 if (!spec->multiout.hp_nid)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004460 spec->multiout.hp_nid = nid;
Takashi Iwai82bc9552006-03-21 11:24:42 +01004461 else
4462 spec->multiout.extra_out_nid[0] = nid;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004463 /* control HP volume/switch on the output mixer amp */
4464 nid = alc880_idx_to_mixer(alc880_fixed_pin_idx(pin));
Takashi Iwai0afe5f82009-10-02 09:20:00 +02004465 err = add_pb_vol_ctrl(spec, ALC_CTL_WIDGET_VOL, pfx,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004466 HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_OUTPUT));
4467 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004468 return err;
Takashi Iwai0afe5f82009-10-02 09:20:00 +02004469 err = add_pb_sw_ctrl(spec, ALC_CTL_BIND_MUTE, pfx,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004470 HDA_COMPOSE_AMP_VAL(nid, 3, 2, HDA_INPUT));
4471 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004472 return err;
4473 } else if (alc880_is_multi_pin(pin)) {
4474 /* set manual connection */
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004475 /* we have only a switch on HP-out PIN */
Takashi Iwai0afe5f82009-10-02 09:20:00 +02004476 err = add_pb_sw_ctrl(spec, ALC_CTL_WIDGET_MUTE, pfx,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004477 HDA_COMPOSE_AMP_VAL(pin, 3, 0, HDA_OUTPUT));
4478 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004479 return err;
4480 }
4481 return 0;
4482}
4483
4484/* create input playback/capture controls for the given pin */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004485static int new_analog_input(struct alc_spec *spec, hda_nid_t pin,
4486 const char *ctlname,
Kailang Yangdf694da2005-12-05 19:42:22 +01004487 int idx, hda_nid_t mix_nid)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004488{
Kailang Yangdf694da2005-12-05 19:42:22 +01004489 int err;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004490
Takashi Iwai0afe5f82009-10-02 09:20:00 +02004491 err = add_pb_vol_ctrl(spec, ALC_CTL_WIDGET_VOL, ctlname,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004492 HDA_COMPOSE_AMP_VAL(mix_nid, 3, idx, HDA_INPUT));
4493 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004494 return err;
Takashi Iwai0afe5f82009-10-02 09:20:00 +02004495 err = add_pb_sw_ctrl(spec, ALC_CTL_WIDGET_MUTE, ctlname,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004496 HDA_COMPOSE_AMP_VAL(mix_nid, 3, idx, HDA_INPUT));
4497 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004498 return err;
4499 return 0;
4500}
4501
Takashi Iwai05f5f472009-08-25 13:10:18 +02004502static int alc_is_input_pin(struct hda_codec *codec, hda_nid_t nid)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004503{
Takashi Iwai05f5f472009-08-25 13:10:18 +02004504 unsigned int pincap = snd_hda_query_pin_caps(codec, nid);
4505 return (pincap & AC_PINCAP_IN) != 0;
4506}
4507
4508/* create playback/capture controls for input pins */
4509static int alc_auto_create_input_ctls(struct hda_codec *codec,
4510 const struct auto_pin_cfg *cfg,
4511 hda_nid_t mixer,
4512 hda_nid_t cap1, hda_nid_t cap2)
4513{
4514 struct alc_spec *spec = codec->spec;
Herton Ronaldo Krzesinski61b9b9b2009-01-28 09:16:33 -02004515 struct hda_input_mux *imux = &spec->private_imux[0];
Kailang Yangdf694da2005-12-05 19:42:22 +01004516 int i, err, idx;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004517
4518 for (i = 0; i < AUTO_PIN_LAST; i++) {
Takashi Iwai05f5f472009-08-25 13:10:18 +02004519 hda_nid_t pin;
4520
4521 pin = cfg->input_pins[i];
4522 if (!alc_is_input_pin(codec, pin))
4523 continue;
4524
4525 if (mixer) {
4526 idx = get_connection_index(codec, mixer, pin);
4527 if (idx >= 0) {
4528 err = new_analog_input(spec, pin,
4529 auto_pin_cfg_labels[i],
4530 idx, mixer);
4531 if (err < 0)
4532 return err;
4533 }
4534 }
4535
4536 if (!cap1)
4537 continue;
4538 idx = get_connection_index(codec, cap1, pin);
4539 if (idx < 0 && cap2)
4540 idx = get_connection_index(codec, cap2, pin);
4541 if (idx >= 0) {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004542 imux->items[imux->num_items].label =
4543 auto_pin_cfg_labels[i];
Takashi Iwai05f5f472009-08-25 13:10:18 +02004544 imux->items[imux->num_items].index = idx;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004545 imux->num_items++;
4546 }
4547 }
4548 return 0;
4549}
4550
Takashi Iwai05f5f472009-08-25 13:10:18 +02004551static int alc880_auto_create_input_ctls(struct hda_codec *codec,
4552 const struct auto_pin_cfg *cfg)
4553{
4554 return alc_auto_create_input_ctls(codec, cfg, 0x0b, 0x08, 0x09);
4555}
4556
Takashi Iwaif6c7e542008-02-12 18:32:23 +01004557static void alc_set_pin_output(struct hda_codec *codec, hda_nid_t nid,
4558 unsigned int pin_type)
4559{
4560 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
4561 pin_type);
4562 /* unmute pin */
Takashi Iwaid260cdf2008-02-13 17:19:35 +01004563 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_AMP_GAIN_MUTE,
4564 AMP_OUT_UNMUTE);
Takashi Iwaif6c7e542008-02-12 18:32:23 +01004565}
4566
Kailang Yangdf694da2005-12-05 19:42:22 +01004567static void alc880_auto_set_output_and_unmute(struct hda_codec *codec,
4568 hda_nid_t nid, int pin_type,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004569 int dac_idx)
4570{
Takashi Iwaif6c7e542008-02-12 18:32:23 +01004571 alc_set_pin_output(codec, nid, pin_type);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004572 /* need the manual connection? */
4573 if (alc880_is_multi_pin(nid)) {
4574 struct alc_spec *spec = codec->spec;
4575 int idx = alc880_multi_pin_idx(nid);
4576 snd_hda_codec_write(codec, alc880_idx_to_selector(idx), 0,
4577 AC_VERB_SET_CONNECT_SEL,
4578 alc880_dac_to_idx(spec->multiout.dac_nids[dac_idx]));
4579 }
4580}
4581
Takashi Iwaibaba8ee2007-04-23 17:17:48 +02004582static int get_pin_type(int line_out_type)
4583{
4584 if (line_out_type == AUTO_PIN_HP_OUT)
4585 return PIN_HP;
4586 else
4587 return PIN_OUT;
4588}
4589
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004590static void alc880_auto_init_multi_out(struct hda_codec *codec)
4591{
4592 struct alc_spec *spec = codec->spec;
4593 int i;
Kailang Yangea1fb292008-08-26 12:58:38 +02004594
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004595 for (i = 0; i < spec->autocfg.line_outs; i++) {
4596 hda_nid_t nid = spec->autocfg.line_out_pins[i];
Takashi Iwaibaba8ee2007-04-23 17:17:48 +02004597 int pin_type = get_pin_type(spec->autocfg.line_out_type);
4598 alc880_auto_set_output_and_unmute(codec, nid, pin_type, i);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004599 }
4600}
4601
Takashi Iwai8d88bc32005-11-17 11:09:23 +01004602static void alc880_auto_init_extra_out(struct hda_codec *codec)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004603{
4604 struct alc_spec *spec = codec->spec;
4605 hda_nid_t pin;
4606
Takashi Iwai82bc9552006-03-21 11:24:42 +01004607 pin = spec->autocfg.speaker_pins[0];
Takashi Iwai8d88bc32005-11-17 11:09:23 +01004608 if (pin) /* connect to front */
4609 alc880_auto_set_output_and_unmute(codec, pin, PIN_OUT, 0);
Takashi Iwaieb06ed82006-09-20 17:10:27 +02004610 pin = spec->autocfg.hp_pins[0];
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004611 if (pin) /* connect to front */
4612 alc880_auto_set_output_and_unmute(codec, pin, PIN_HP, 0);
4613}
4614
4615static void alc880_auto_init_analog_input(struct hda_codec *codec)
4616{
4617 struct alc_spec *spec = codec->spec;
4618 int i;
4619
4620 for (i = 0; i < AUTO_PIN_LAST; i++) {
4621 hda_nid_t nid = spec->autocfg.input_pins[i];
Takashi Iwai05f5f472009-08-25 13:10:18 +02004622 if (alc_is_input_pin(codec, nid)) {
Takashi Iwai23f0c042009-02-26 13:03:58 +01004623 alc_set_input_pin(codec, nid, i);
Takashi Iwaie82c0252009-03-23 15:17:38 +01004624 if (nid != ALC880_PIN_CD_NID &&
4625 (get_wcaps(codec, nid) & AC_WCAP_OUT_AMP))
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004626 snd_hda_codec_write(codec, nid, 0,
4627 AC_VERB_SET_AMP_GAIN_MUTE,
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004628 AMP_OUT_MUTE);
4629 }
4630 }
4631}
4632
4633/* parse the BIOS configuration and set up the alc_spec */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004634/* return 1 if successful, 0 if the proper config is not found,
4635 * or a negative error code
4636 */
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004637static int alc880_parse_auto_config(struct hda_codec *codec)
4638{
4639 struct alc_spec *spec = codec->spec;
Takashi Iwai6a05ac42009-02-13 11:19:09 +01004640 int i, err;
Kailang Yangdf694da2005-12-05 19:42:22 +01004641 static hda_nid_t alc880_ignore[] = { 0x1d, 0 };
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004642
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004643 err = snd_hda_parse_pin_def_config(codec, &spec->autocfg,
4644 alc880_ignore);
4645 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004646 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004647 if (!spec->autocfg.line_outs)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004648 return 0; /* can't find valid BIOS pin config */
Kailang Yangdf694da2005-12-05 19:42:22 +01004649
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004650 err = alc880_auto_fill_dac_nids(spec, &spec->autocfg);
4651 if (err < 0)
4652 return err;
4653 err = alc880_auto_create_multi_out_ctls(spec, &spec->autocfg);
4654 if (err < 0)
4655 return err;
4656 err = alc880_auto_create_extra_out(spec,
4657 spec->autocfg.speaker_pins[0],
4658 "Speaker");
4659 if (err < 0)
4660 return err;
4661 err = alc880_auto_create_extra_out(spec, spec->autocfg.hp_pins[0],
4662 "Headphone");
4663 if (err < 0)
4664 return err;
Takashi Iwai05f5f472009-08-25 13:10:18 +02004665 err = alc880_auto_create_input_ctls(codec, &spec->autocfg);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004666 if (err < 0)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004667 return err;
4668
4669 spec->multiout.max_channels = spec->multiout.num_dacs * 2;
4670
Takashi Iwai6a05ac42009-02-13 11:19:09 +01004671 /* check multiple SPDIF-out (for recent codecs) */
4672 for (i = 0; i < spec->autocfg.dig_outs; i++) {
4673 hda_nid_t dig_nid;
4674 err = snd_hda_get_connections(codec,
4675 spec->autocfg.dig_out_pins[i],
4676 &dig_nid, 1);
4677 if (err < 0)
4678 continue;
4679 if (!i)
4680 spec->multiout.dig_out_nid = dig_nid;
4681 else {
4682 spec->multiout.slave_dig_outs = spec->slave_dig_outs;
Roel Kluin71121d9f2009-11-10 20:11:55 +01004683 if (i >= ARRAY_SIZE(spec->slave_dig_outs) - 1)
Takashi Iwai6a05ac42009-02-13 11:19:09 +01004684 break;
Roel Kluin71121d9f2009-11-10 20:11:55 +01004685 spec->slave_dig_outs[i - 1] = dig_nid;
Takashi Iwai6a05ac42009-02-13 11:19:09 +01004686 }
4687 }
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004688 if (spec->autocfg.dig_in_pin)
4689 spec->dig_in_nid = ALC880_DIGIN_NID;
4690
Takashi Iwai603c4012008-07-30 15:01:44 +02004691 if (spec->kctls.list)
Takashi Iwaid88897e2008-10-31 15:01:37 +01004692 add_mixer(spec, spec->kctls.list);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004693
Takashi Iwaid88897e2008-10-31 15:01:37 +01004694 add_verb(spec, alc880_volume_init_verbs);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004695
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02004696 spec->num_mux_defs = 1;
Herton Ronaldo Krzesinski61b9b9b2009-01-28 09:16:33 -02004697 spec->input_mux = &spec->private_imux[0];
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004698
Takashi Iwai4a79ba32009-04-22 16:31:35 +02004699 alc_ssid_check(codec, 0x15, 0x1b, 0x14);
4700
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004701 return 1;
4702}
4703
Takashi Iwaiae6b8132006-03-03 16:47:17 +01004704/* additional initialization for auto-configuration model */
4705static void alc880_auto_init(struct hda_codec *codec)
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004706{
Takashi Iwaif6c7e542008-02-12 18:32:23 +01004707 struct alc_spec *spec = codec->spec;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004708 alc880_auto_init_multi_out(codec);
Takashi Iwai8d88bc32005-11-17 11:09:23 +01004709 alc880_auto_init_extra_out(codec);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004710 alc880_auto_init_analog_input(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +01004711 if (spec->unsol_event)
Kailang Yang7fb0d782008-10-15 11:12:35 +02004712 alc_inithook(codec);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004713}
4714
Takashi Iwaib59bdf32009-08-11 09:47:30 +02004715/* check the ADC/MUX contains all input pins; some ADC/MUX contains only
4716 * one of two digital mic pins, e.g. on ALC272
4717 */
4718static void fixup_automic_adc(struct hda_codec *codec)
Takashi Iwaif9e336f2008-10-31 16:37:07 +01004719{
Takashi Iwaib59bdf32009-08-11 09:47:30 +02004720 struct alc_spec *spec = codec->spec;
4721 int i;
4722
4723 for (i = 0; i < spec->num_adc_nids; i++) {
4724 hda_nid_t cap = spec->capsrc_nids ?
4725 spec->capsrc_nids[i] : spec->adc_nids[i];
4726 int iidx, eidx;
4727
4728 iidx = get_connection_index(codec, cap, spec->int_mic.pin);
4729 if (iidx < 0)
4730 continue;
4731 eidx = get_connection_index(codec, cap, spec->ext_mic.pin);
4732 if (eidx < 0)
4733 continue;
4734 spec->int_mic.mux_idx = iidx;
4735 spec->ext_mic.mux_idx = eidx;
4736 if (spec->capsrc_nids)
4737 spec->capsrc_nids += i;
4738 spec->adc_nids += i;
4739 spec->num_adc_nids = 1;
4740 return;
4741 }
4742 snd_printd(KERN_INFO "hda_codec: %s: "
4743 "No ADC/MUX containing both 0x%x and 0x%x pins\n",
4744 codec->chip_name, spec->int_mic.pin, spec->ext_mic.pin);
4745 spec->auto_mic = 0; /* disable auto-mic to be sure */
4746}
4747
4748static void set_capture_mixer(struct hda_codec *codec)
4749{
4750 struct alc_spec *spec = codec->spec;
Takashi Iwaia23b6882009-03-23 15:21:36 +01004751 static struct snd_kcontrol_new *caps[2][3] = {
4752 { alc_capture_mixer_nosrc1,
4753 alc_capture_mixer_nosrc2,
4754 alc_capture_mixer_nosrc3 },
4755 { alc_capture_mixer1,
4756 alc_capture_mixer2,
4757 alc_capture_mixer3 },
Takashi Iwaif9e336f2008-10-31 16:37:07 +01004758 };
Takashi Iwaia23b6882009-03-23 15:21:36 +01004759 if (spec->num_adc_nids > 0 && spec->num_adc_nids <= 3) {
4760 int mux;
Takashi Iwai2a22d3f2009-08-10 18:54:38 +02004761 if (spec->auto_mic) {
4762 mux = 0;
Takashi Iwaib59bdf32009-08-11 09:47:30 +02004763 fixup_automic_adc(codec);
Takashi Iwai2a22d3f2009-08-10 18:54:38 +02004764 } else if (spec->input_mux && spec->input_mux->num_items > 1)
Takashi Iwaia23b6882009-03-23 15:21:36 +01004765 mux = 1;
4766 else
4767 mux = 0;
4768 spec->cap_mixer = caps[mux][spec->num_adc_nids - 1];
4769 }
Takashi Iwaif9e336f2008-10-31 16:37:07 +01004770}
4771
Takashi Iwai67d634c2009-11-16 15:35:59 +01004772#ifdef CONFIG_SND_HDA_INPUT_BEEP
Takashi Iwai45bdd1c2009-02-06 16:11:25 +01004773#define set_beep_amp(spec, nid, idx, dir) \
4774 ((spec)->beep_amp = HDA_COMPOSE_AMP_VAL(nid, 3, idx, dir))
Takashi Iwai67d634c2009-11-16 15:35:59 +01004775#else
4776#define set_beep_amp(spec, nid, idx, dir) /* NOP */
4777#endif
Takashi Iwai45bdd1c2009-02-06 16:11:25 +01004778
4779/*
4780 * OK, here we have finally the patch for ALC880
4781 */
4782
Linus Torvalds1da177e2005-04-16 15:20:36 -07004783static int patch_alc880(struct hda_codec *codec)
4784{
4785 struct alc_spec *spec;
4786 int board_config;
Kailang Yangdf694da2005-12-05 19:42:22 +01004787 int err;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004788
Takashi Iwaie560d8d2005-09-09 14:21:46 +02004789 spec = kzalloc(sizeof(*spec), GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004790 if (spec == NULL)
4791 return -ENOMEM;
4792
4793 codec->spec = spec;
4794
Takashi Iwaif5fcc132006-11-24 17:07:44 +01004795 board_config = snd_hda_check_board_config(codec, ALC880_MODEL_LAST,
4796 alc880_models,
4797 alc880_cfg_tbl);
4798 if (board_config < 0) {
Takashi Iwai9a11f1a2009-07-28 16:01:20 +02004799 printk(KERN_INFO "hda_codec: %s: BIOS auto-probing.\n",
4800 codec->chip_name);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004801 board_config = ALC880_AUTO;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004802 }
4803
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004804 if (board_config == ALC880_AUTO) {
4805 /* automatic parse from the BIOS config */
4806 err = alc880_parse_auto_config(codec);
4807 if (err < 0) {
4808 alc_free(codec);
4809 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004810 } else if (!err) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +02004811 printk(KERN_INFO
4812 "hda_codec: Cannot set up configuration "
4813 "from BIOS. Using 3-stack mode...\n");
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004814 board_config = ALC880_3ST;
4815 }
4816 }
4817
Kusanagi Kouichi680cd532009-02-05 00:00:58 +09004818 err = snd_hda_attach_beep_device(codec, 0x1);
4819 if (err < 0) {
4820 alc_free(codec);
4821 return err;
4822 }
4823
Kailang Yangdf694da2005-12-05 19:42:22 +01004824 if (board_config != ALC880_AUTO)
Takashi Iwaie9c364c2009-08-11 17:16:13 +02004825 setup_preset(codec, &alc880_presets[board_config]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004826
Linus Torvalds1da177e2005-04-16 15:20:36 -07004827 spec->stream_analog_playback = &alc880_pcm_analog_playback;
4828 spec->stream_analog_capture = &alc880_pcm_analog_capture;
Takashi Iwai63300792008-01-24 15:31:36 +01004829 spec->stream_analog_alt_capture = &alc880_pcm_analog_alt_capture;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004830
Linus Torvalds1da177e2005-04-16 15:20:36 -07004831 spec->stream_digital_playback = &alc880_pcm_digital_playback;
4832 spec->stream_digital_capture = &alc880_pcm_digital_capture;
4833
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004834 if (!spec->adc_nids && spec->input_mux) {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004835 /* check whether NID 0x07 is valid */
Takashi Iwai54d17402005-11-21 16:33:22 +01004836 unsigned int wcap = get_wcaps(codec, alc880_adc_nids[0]);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02004837 /* get type */
Takashi Iwaia22d5432009-07-27 12:54:26 +02004838 wcap = get_wcaps_type(wcap);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004839 if (wcap != AC_WID_AUD_IN) {
4840 spec->adc_nids = alc880_adc_nids_alt;
4841 spec->num_adc_nids = ARRAY_SIZE(alc880_adc_nids_alt);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004842 } else {
4843 spec->adc_nids = alc880_adc_nids;
4844 spec->num_adc_nids = ARRAY_SIZE(alc880_adc_nids);
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004845 }
4846 }
Takashi Iwaib59bdf32009-08-11 09:47:30 +02004847 set_capture_mixer(codec);
Takashi Iwai45bdd1c2009-02-06 16:11:25 +01004848 set_beep_amp(spec, 0x0b, 0x05, HDA_INPUT);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004849
Takashi Iwai2134ea42008-01-10 16:53:55 +01004850 spec->vmaster_nid = 0x0c;
4851
Linus Torvalds1da177e2005-04-16 15:20:36 -07004852 codec->patch_ops = alc_patch_ops;
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004853 if (board_config == ALC880_AUTO)
Takashi Iwaiae6b8132006-03-03 16:47:17 +01004854 spec->init_hook = alc880_auto_init;
Takashi Iwaicb53c622007-08-10 17:21:45 +02004855#ifdef CONFIG_SND_HDA_POWER_SAVE
4856 if (!spec->loopback.amplist)
4857 spec->loopback.amplist = alc880_loopbacks;
4858#endif
Takashi Iwaidaead532008-11-28 12:55:36 +01004859 codec->proc_widget_hook = print_realtek_coef;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004860
4861 return 0;
4862}
4863
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004864
Linus Torvalds1da177e2005-04-16 15:20:36 -07004865/*
4866 * ALC260 support
4867 */
4868
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004869static hda_nid_t alc260_dac_nids[1] = {
4870 /* front */
4871 0x02,
4872};
4873
4874static hda_nid_t alc260_adc_nids[1] = {
4875 /* ADC0 */
4876 0x04,
4877};
4878
Kailang Yangdf694da2005-12-05 19:42:22 +01004879static hda_nid_t alc260_adc_nids_alt[1] = {
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004880 /* ADC1 */
4881 0x05,
4882};
4883
Jonathan Woithed57fdac2006-02-28 11:38:35 +01004884/* NIDs used when simultaneous access to both ADCs makes sense. Note that
4885 * alc260_capture_mixer assumes ADC0 (nid 0x04) is the first ADC.
4886 */
4887static hda_nid_t alc260_dual_adc_nids[2] = {
Jonathan Woithe4c5186e2006-02-09 11:53:48 +01004888 /* ADC0, ADC1 */
4889 0x04, 0x05
4890};
4891
Takashi Iwaie9edcee2005-06-13 14:16:38 +02004892#define ALC260_DIGOUT_NID 0x03
4893#define ALC260_DIGIN_NID 0x06
4894
4895static struct hda_input_mux alc260_capture_source = {
4896 .num_items = 4,
4897 .items = {
4898 { "Mic", 0x0 },
4899 { "Front Mic", 0x1 },
4900 { "Line", 0x2 },
4901 { "CD", 0x4 },
4902 },
4903};
4904
Jonathan Woithe17e7aec2006-02-28 11:35:18 +01004905/* On Fujitsu S702x laptops capture only makes sense from Mic/LineIn jack,
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02004906 * headphone jack and the internal CD lines since these are the only pins at
4907 * which audio can appear. For flexibility, also allow the option of
4908 * recording the mixer output on the second ADC (ADC0 doesn't have a
4909 * connection to the mixer output).
Jonathan Woithea9430dd2005-09-16 19:12:48 +02004910 */
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02004911static struct hda_input_mux alc260_fujitsu_capture_sources[2] = {
4912 {
4913 .num_items = 3,
4914 .items = {
4915 { "Mic/Line", 0x0 },
4916 { "CD", 0x4 },
4917 { "Headphone", 0x2 },
4918 },
Jonathan Woithea9430dd2005-09-16 19:12:48 +02004919 },
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02004920 {
4921 .num_items = 4,
4922 .items = {
4923 { "Mic/Line", 0x0 },
4924 { "CD", 0x4 },
4925 { "Headphone", 0x2 },
4926 { "Mixer", 0x5 },
4927 },
4928 },
4929
Jonathan Woithea9430dd2005-09-16 19:12:48 +02004930};
4931
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02004932/* Acer TravelMate(/Extensa/Aspire) notebooks have similar configuration to
4933 * the Fujitsu S702x, but jacks are marked differently.
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01004934 */
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02004935static struct hda_input_mux alc260_acer_capture_sources[2] = {
4936 {
4937 .num_items = 4,
4938 .items = {
4939 { "Mic", 0x0 },
4940 { "Line", 0x2 },
4941 { "CD", 0x4 },
4942 { "Headphone", 0x5 },
4943 },
4944 },
4945 {
4946 .num_items = 5,
4947 .items = {
4948 { "Mic", 0x0 },
4949 { "Line", 0x2 },
4950 { "CD", 0x4 },
4951 { "Headphone", 0x6 },
4952 { "Mixer", 0x5 },
4953 },
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01004954 },
4955};
Michael Schwingencc959482009-02-22 18:58:45 +01004956
4957/* Maxdata Favorit 100XS */
4958static struct hda_input_mux alc260_favorit100_capture_sources[2] = {
4959 {
4960 .num_items = 2,
4961 .items = {
4962 { "Line/Mic", 0x0 },
4963 { "CD", 0x4 },
4964 },
4965 },
4966 {
4967 .num_items = 3,
4968 .items = {
4969 { "Line/Mic", 0x0 },
4970 { "CD", 0x4 },
4971 { "Mixer", 0x5 },
4972 },
4973 },
4974};
4975
Linus Torvalds1da177e2005-04-16 15:20:36 -07004976/*
4977 * This is just place-holder, so there's something for alc_build_pcms to look
4978 * at when it calculates the maximum number of channels. ALC260 has no mixer
4979 * element which allows changing the channel mode, so the verb list is
4980 * never used.
4981 */
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +01004982static struct hda_channel_mode alc260_modes[1] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07004983 { 2, NULL },
4984};
4985
Kailang Yangdf694da2005-12-05 19:42:22 +01004986
4987/* Mixer combinations
4988 *
4989 * basic: base_output + input + pc_beep + capture
4990 * HP: base_output + input + capture_alt
4991 * HP_3013: hp_3013 + input + capture
4992 * fujitsu: fujitsu + capture
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01004993 * acer: acer + capture
Kailang Yangdf694da2005-12-05 19:42:22 +01004994 */
4995
4996static struct snd_kcontrol_new alc260_base_output_mixer[] = {
Takashi Iwai05acb862005-06-10 19:50:25 +02004997 HDA_CODEC_VOLUME("Front Playback Volume", 0x08, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01004998 HDA_BIND_MUTE("Front Playback Switch", 0x08, 2, HDA_INPUT),
Kailang Yangdf694da2005-12-05 19:42:22 +01004999 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x09, 0x0, HDA_OUTPUT),
5000 HDA_BIND_MUTE("Headphone Playback Switch", 0x09, 2, HDA_INPUT),
5001 HDA_CODEC_VOLUME_MONO("Mono Playback Volume", 0x0a, 1, 0x0, HDA_OUTPUT),
5002 HDA_BIND_MUTE_MONO("Mono Playback Switch", 0x0a, 1, 2, HDA_INPUT),
5003 { } /* end */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02005004};
Kailang Yangdf694da2005-12-05 19:42:22 +01005005
5006static struct snd_kcontrol_new alc260_input_mixer[] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07005007 HDA_CODEC_VOLUME("CD Playback Volume", 0x07, 0x04, HDA_INPUT),
5008 HDA_CODEC_MUTE("CD Playback Switch", 0x07, 0x04, HDA_INPUT),
5009 HDA_CODEC_VOLUME("Line Playback Volume", 0x07, 0x02, HDA_INPUT),
5010 HDA_CODEC_MUTE("Line Playback Switch", 0x07, 0x02, HDA_INPUT),
5011 HDA_CODEC_VOLUME("Mic Playback Volume", 0x07, 0x0, HDA_INPUT),
5012 HDA_CODEC_MUTE("Mic Playback Switch", 0x07, 0x0, HDA_INPUT),
5013 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x07, 0x01, HDA_INPUT),
5014 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x07, 0x01, HDA_INPUT),
Linus Torvalds1da177e2005-04-16 15:20:36 -07005015 { } /* end */
5016};
5017
Takashi Iwaibec15c32008-01-28 18:16:30 +01005018/* update HP, line and mono out pins according to the master switch */
5019static void alc260_hp_master_update(struct hda_codec *codec,
5020 hda_nid_t hp, hda_nid_t line,
5021 hda_nid_t mono)
5022{
5023 struct alc_spec *spec = codec->spec;
5024 unsigned int val = spec->master_sw ? PIN_HP : 0;
5025 /* change HP and line-out pins */
Takashi Iwai30cde0a2008-11-07 14:49:28 +01005026 snd_hda_codec_write(codec, hp, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
Takashi Iwaibec15c32008-01-28 18:16:30 +01005027 val);
Takashi Iwai30cde0a2008-11-07 14:49:28 +01005028 snd_hda_codec_write(codec, line, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
Takashi Iwaibec15c32008-01-28 18:16:30 +01005029 val);
5030 /* mono (speaker) depending on the HP jack sense */
5031 val = (val && !spec->jack_present) ? PIN_OUT : 0;
Takashi Iwai30cde0a2008-11-07 14:49:28 +01005032 snd_hda_codec_write(codec, mono, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
Takashi Iwaibec15c32008-01-28 18:16:30 +01005033 val);
5034}
5035
5036static int alc260_hp_master_sw_get(struct snd_kcontrol *kcontrol,
5037 struct snd_ctl_elem_value *ucontrol)
5038{
5039 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
5040 struct alc_spec *spec = codec->spec;
5041 *ucontrol->value.integer.value = spec->master_sw;
5042 return 0;
5043}
5044
5045static int alc260_hp_master_sw_put(struct snd_kcontrol *kcontrol,
5046 struct snd_ctl_elem_value *ucontrol)
5047{
5048 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
5049 struct alc_spec *spec = codec->spec;
5050 int val = !!*ucontrol->value.integer.value;
5051 hda_nid_t hp, line, mono;
5052
5053 if (val == spec->master_sw)
5054 return 0;
5055 spec->master_sw = val;
5056 hp = (kcontrol->private_value >> 16) & 0xff;
5057 line = (kcontrol->private_value >> 8) & 0xff;
5058 mono = kcontrol->private_value & 0xff;
5059 alc260_hp_master_update(codec, hp, line, mono);
5060 return 1;
5061}
5062
5063static struct snd_kcontrol_new alc260_hp_output_mixer[] = {
5064 {
5065 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
5066 .name = "Master Playback Switch",
5067 .info = snd_ctl_boolean_mono_info,
5068 .get = alc260_hp_master_sw_get,
5069 .put = alc260_hp_master_sw_put,
5070 .private_value = (0x0f << 16) | (0x10 << 8) | 0x11
5071 },
5072 HDA_CODEC_VOLUME("Front Playback Volume", 0x08, 0x0, HDA_OUTPUT),
5073 HDA_BIND_MUTE("Front Playback Switch", 0x08, 2, HDA_INPUT),
5074 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x09, 0x0, HDA_OUTPUT),
5075 HDA_BIND_MUTE("Headphone Playback Switch", 0x09, 2, HDA_INPUT),
5076 HDA_CODEC_VOLUME_MONO("Speaker Playback Volume", 0x0a, 1, 0x0,
5077 HDA_OUTPUT),
5078 HDA_BIND_MUTE_MONO("Speaker Playback Switch", 0x0a, 1, 2, HDA_INPUT),
5079 { } /* end */
5080};
5081
5082static struct hda_verb alc260_hp_unsol_verbs[] = {
5083 {0x10, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
5084 {},
5085};
5086
5087static void alc260_hp_automute(struct hda_codec *codec)
5088{
5089 struct alc_spec *spec = codec->spec;
Takashi Iwaibec15c32008-01-28 18:16:30 +01005090
Wu Fengguang864f92b2009-11-18 12:38:02 +08005091 spec->jack_present = snd_hda_jack_detect(codec, 0x10);
Takashi Iwaibec15c32008-01-28 18:16:30 +01005092 alc260_hp_master_update(codec, 0x0f, 0x10, 0x11);
5093}
5094
5095static void alc260_hp_unsol_event(struct hda_codec *codec, unsigned int res)
5096{
5097 if ((res >> 26) == ALC880_HP_EVENT)
5098 alc260_hp_automute(codec);
5099}
5100
Kailang Yangdf694da2005-12-05 19:42:22 +01005101static struct snd_kcontrol_new alc260_hp_3013_mixer[] = {
Takashi Iwaibec15c32008-01-28 18:16:30 +01005102 {
5103 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
5104 .name = "Master Playback Switch",
5105 .info = snd_ctl_boolean_mono_info,
5106 .get = alc260_hp_master_sw_get,
5107 .put = alc260_hp_master_sw_put,
Takashi Iwai30cde0a2008-11-07 14:49:28 +01005108 .private_value = (0x15 << 16) | (0x10 << 8) | 0x11
Takashi Iwaibec15c32008-01-28 18:16:30 +01005109 },
Kailang Yangdf694da2005-12-05 19:42:22 +01005110 HDA_CODEC_VOLUME("Front Playback Volume", 0x09, 0x0, HDA_OUTPUT),
5111 HDA_CODEC_MUTE("Front Playback Switch", 0x10, 0x0, HDA_OUTPUT),
5112 HDA_CODEC_VOLUME("Aux-In Playback Volume", 0x07, 0x06, HDA_INPUT),
5113 HDA_CODEC_MUTE("Aux-In Playback Switch", 0x07, 0x06, HDA_INPUT),
5114 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x08, 0x0, HDA_OUTPUT),
5115 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
Takashi Iwai86cd9292008-01-28 18:09:56 +01005116 HDA_CODEC_VOLUME_MONO("Speaker Playback Volume", 0x0a, 1, 0x0, HDA_OUTPUT),
5117 HDA_CODEC_MUTE_MONO("Speaker Playback Switch", 0x11, 1, 0x0, HDA_OUTPUT),
Takashi Iwai16ded522005-06-10 19:58:24 +02005118 { } /* end */
5119};
5120
Kailang Yang3f878302008-08-26 13:02:23 +02005121static struct hda_bind_ctls alc260_dc7600_bind_master_vol = {
5122 .ops = &snd_hda_bind_vol,
5123 .values = {
5124 HDA_COMPOSE_AMP_VAL(0x08, 3, 0, HDA_OUTPUT),
5125 HDA_COMPOSE_AMP_VAL(0x09, 3, 0, HDA_OUTPUT),
5126 HDA_COMPOSE_AMP_VAL(0x0a, 3, 0, HDA_OUTPUT),
5127 0
5128 },
5129};
5130
5131static struct hda_bind_ctls alc260_dc7600_bind_switch = {
5132 .ops = &snd_hda_bind_sw,
5133 .values = {
5134 HDA_COMPOSE_AMP_VAL(0x11, 3, 0, HDA_OUTPUT),
5135 HDA_COMPOSE_AMP_VAL(0x15, 3, 0, HDA_OUTPUT),
5136 0
5137 },
5138};
5139
5140static struct snd_kcontrol_new alc260_hp_dc7600_mixer[] = {
5141 HDA_BIND_VOL("Master Playback Volume", &alc260_dc7600_bind_master_vol),
5142 HDA_BIND_SW("LineOut Playback Switch", &alc260_dc7600_bind_switch),
5143 HDA_CODEC_MUTE("Speaker Playback Switch", 0x0f, 0x0, HDA_OUTPUT),
5144 HDA_CODEC_MUTE("Headphone Playback Switch", 0x10, 0x0, HDA_OUTPUT),
5145 { } /* end */
5146};
5147
Takashi Iwaibec15c32008-01-28 18:16:30 +01005148static struct hda_verb alc260_hp_3013_unsol_verbs[] = {
5149 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
5150 {},
5151};
5152
5153static void alc260_hp_3013_automute(struct hda_codec *codec)
5154{
5155 struct alc_spec *spec = codec->spec;
Takashi Iwaibec15c32008-01-28 18:16:30 +01005156
Wu Fengguang864f92b2009-11-18 12:38:02 +08005157 spec->jack_present = snd_hda_jack_detect(codec, 0x15);
Takashi Iwai30cde0a2008-11-07 14:49:28 +01005158 alc260_hp_master_update(codec, 0x15, 0x10, 0x11);
Takashi Iwaibec15c32008-01-28 18:16:30 +01005159}
5160
5161static void alc260_hp_3013_unsol_event(struct hda_codec *codec,
5162 unsigned int res)
5163{
5164 if ((res >> 26) == ALC880_HP_EVENT)
5165 alc260_hp_3013_automute(codec);
5166}
5167
Kailang Yang3f878302008-08-26 13:02:23 +02005168static void alc260_hp_3012_automute(struct hda_codec *codec)
5169{
Wu Fengguang864f92b2009-11-18 12:38:02 +08005170 unsigned int bits = snd_hda_jack_detect(codec, 0x10) ? 0 : PIN_OUT;
Kailang Yang3f878302008-08-26 13:02:23 +02005171
Kailang Yang3f878302008-08-26 13:02:23 +02005172 snd_hda_codec_write(codec, 0x0f, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
5173 bits);
5174 snd_hda_codec_write(codec, 0x11, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
5175 bits);
5176 snd_hda_codec_write(codec, 0x15, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
5177 bits);
5178}
5179
5180static void alc260_hp_3012_unsol_event(struct hda_codec *codec,
5181 unsigned int res)
5182{
5183 if ((res >> 26) == ALC880_HP_EVENT)
5184 alc260_hp_3012_automute(codec);
5185}
5186
5187/* Fujitsu S702x series laptops. ALC260 pin usage: Mic/Line jack = 0x12,
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02005188 * HP jack = 0x14, CD audio = 0x16, internal speaker = 0x10.
5189 */
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01005190static struct snd_kcontrol_new alc260_fujitsu_mixer[] = {
Jonathan Woithea9430dd2005-09-16 19:12:48 +02005191 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x08, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01005192 HDA_BIND_MUTE("Headphone Playback Switch", 0x08, 2, HDA_INPUT),
Jonathan Woithe4c5186e2006-02-09 11:53:48 +01005193 ALC_PIN_MODE("Headphone Jack Mode", 0x14, ALC_PIN_DIR_INOUT),
Jonathan Woithea9430dd2005-09-16 19:12:48 +02005194 HDA_CODEC_VOLUME("CD Playback Volume", 0x07, 0x04, HDA_INPUT),
5195 HDA_CODEC_MUTE("CD Playback Switch", 0x07, 0x04, HDA_INPUT),
5196 HDA_CODEC_VOLUME("Mic/Line Playback Volume", 0x07, 0x0, HDA_INPUT),
5197 HDA_CODEC_MUTE("Mic/Line Playback Switch", 0x07, 0x0, HDA_INPUT),
Jonathan Woithe4c5186e2006-02-09 11:53:48 +01005198 ALC_PIN_MODE("Mic/Line Jack Mode", 0x12, ALC_PIN_DIR_IN),
Takashi Iwai31bffaa2008-02-27 16:10:44 +01005199 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x09, 0x0, HDA_OUTPUT),
5200 HDA_BIND_MUTE("Speaker Playback Switch", 0x09, 2, HDA_INPUT),
Kailang Yangdf694da2005-12-05 19:42:22 +01005201 { } /* end */
5202};
5203
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02005204/* Mixer for Acer TravelMate(/Extensa/Aspire) notebooks. Note that current
5205 * versions of the ALC260 don't act on requests to enable mic bias from NID
5206 * 0x0f (used to drive the headphone jack in these laptops). The ALC260
5207 * datasheet doesn't mention this restriction. At this stage it's not clear
5208 * whether this behaviour is intentional or is a hardware bug in chip
5209 * revisions available in early 2006. Therefore for now allow the
5210 * "Headphone Jack Mode" control to span all choices, but if it turns out
5211 * that the lack of mic bias for this NID is intentional we could change the
5212 * mode from ALC_PIN_DIR_INOUT to ALC_PIN_DIR_INOUT_NOMICBIAS.
5213 *
5214 * In addition, Acer TravelMate(/Extensa/Aspire) notebooks in early 2006
5215 * don't appear to make the mic bias available from the "line" jack, even
5216 * though the NID used for this jack (0x14) can supply it. The theory is
5217 * that perhaps Acer have included blocking capacitors between the ALC260
5218 * and the output jack. If this turns out to be the case for all such
5219 * models the "Line Jack Mode" mode could be changed from ALC_PIN_DIR_INOUT
5220 * to ALC_PIN_DIR_INOUT_NOMICBIAS.
Jonathan Woithebd869482006-11-28 11:35:52 +01005221 *
5222 * The C20x Tablet series have a mono internal speaker which is controlled
5223 * via the chip's Mono sum widget and pin complex, so include the necessary
5224 * controls for such models. On models without a "mono speaker" the control
5225 * won't do anything.
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02005226 */
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01005227static struct snd_kcontrol_new alc260_acer_mixer[] = {
5228 HDA_CODEC_VOLUME("Master Playback Volume", 0x08, 0x0, HDA_OUTPUT),
5229 HDA_BIND_MUTE("Master Playback Switch", 0x08, 2, HDA_INPUT),
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02005230 ALC_PIN_MODE("Headphone Jack Mode", 0x0f, ALC_PIN_DIR_INOUT),
Takashi Iwai31bffaa2008-02-27 16:10:44 +01005231 HDA_CODEC_VOLUME_MONO("Speaker Playback Volume", 0x0a, 1, 0x0,
Jonathan Woithebd869482006-11-28 11:35:52 +01005232 HDA_OUTPUT),
Takashi Iwai31bffaa2008-02-27 16:10:44 +01005233 HDA_BIND_MUTE_MONO("Speaker Playback Switch", 0x0a, 1, 2,
Jonathan Woithebd869482006-11-28 11:35:52 +01005234 HDA_INPUT),
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01005235 HDA_CODEC_VOLUME("CD Playback Volume", 0x07, 0x04, HDA_INPUT),
5236 HDA_CODEC_MUTE("CD Playback Switch", 0x07, 0x04, HDA_INPUT),
5237 HDA_CODEC_VOLUME("Mic Playback Volume", 0x07, 0x0, HDA_INPUT),
5238 HDA_CODEC_MUTE("Mic Playback Switch", 0x07, 0x0, HDA_INPUT),
5239 ALC_PIN_MODE("Mic Jack Mode", 0x12, ALC_PIN_DIR_IN),
5240 HDA_CODEC_VOLUME("Line Playback Volume", 0x07, 0x02, HDA_INPUT),
5241 HDA_CODEC_MUTE("Line Playback Switch", 0x07, 0x02, HDA_INPUT),
5242 ALC_PIN_MODE("Line Jack Mode", 0x14, ALC_PIN_DIR_INOUT),
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01005243 { } /* end */
5244};
5245
Michael Schwingencc959482009-02-22 18:58:45 +01005246/* Maxdata Favorit 100XS: one output and one input (0x12) jack
5247 */
5248static struct snd_kcontrol_new alc260_favorit100_mixer[] = {
5249 HDA_CODEC_VOLUME("Master Playback Volume", 0x08, 0x0, HDA_OUTPUT),
5250 HDA_BIND_MUTE("Master Playback Switch", 0x08, 2, HDA_INPUT),
5251 ALC_PIN_MODE("Output Jack Mode", 0x0f, ALC_PIN_DIR_INOUT),
5252 HDA_CODEC_VOLUME("Line/Mic Playback Volume", 0x07, 0x0, HDA_INPUT),
5253 HDA_CODEC_MUTE("Line/Mic Playback Switch", 0x07, 0x0, HDA_INPUT),
5254 ALC_PIN_MODE("Line/Mic Jack Mode", 0x12, ALC_PIN_DIR_IN),
5255 { } /* end */
5256};
5257
Kailang Yangbc9f98a2007-04-12 13:06:07 +02005258/* Packard bell V7900 ALC260 pin usage: HP = 0x0f, Mic jack = 0x12,
5259 * Line In jack = 0x14, CD audio = 0x16, pc beep = 0x17.
5260 */
5261static struct snd_kcontrol_new alc260_will_mixer[] = {
5262 HDA_CODEC_VOLUME("Master Playback Volume", 0x08, 0x0, HDA_OUTPUT),
5263 HDA_BIND_MUTE("Master Playback Switch", 0x08, 0x2, HDA_INPUT),
5264 HDA_CODEC_VOLUME("Mic Playback Volume", 0x07, 0x0, HDA_INPUT),
5265 HDA_CODEC_MUTE("Mic Playback Switch", 0x07, 0x0, HDA_INPUT),
5266 ALC_PIN_MODE("Mic Jack Mode", 0x12, ALC_PIN_DIR_IN),
5267 HDA_CODEC_VOLUME("Line Playback Volume", 0x07, 0x02, HDA_INPUT),
5268 HDA_CODEC_MUTE("Line Playback Switch", 0x07, 0x02, HDA_INPUT),
5269 ALC_PIN_MODE("Line Jack Mode", 0x14, ALC_PIN_DIR_INOUT),
5270 HDA_CODEC_VOLUME("CD Playback Volume", 0x07, 0x04, HDA_INPUT),
5271 HDA_CODEC_MUTE("CD Playback Switch", 0x07, 0x04, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +02005272 { } /* end */
5273};
5274
5275/* Replacer 672V ALC260 pin usage: Mic jack = 0x12,
5276 * Line In jack = 0x14, ATAPI Mic = 0x13, speaker = 0x0f.
5277 */
5278static struct snd_kcontrol_new alc260_replacer_672v_mixer[] = {
5279 HDA_CODEC_VOLUME("Master Playback Volume", 0x08, 0x0, HDA_OUTPUT),
5280 HDA_BIND_MUTE("Master Playback Switch", 0x08, 0x2, HDA_INPUT),
5281 HDA_CODEC_VOLUME("Mic Playback Volume", 0x07, 0x0, HDA_INPUT),
5282 HDA_CODEC_MUTE("Mic Playback Switch", 0x07, 0x0, HDA_INPUT),
5283 ALC_PIN_MODE("Mic Jack Mode", 0x12, ALC_PIN_DIR_IN),
5284 HDA_CODEC_VOLUME("ATAPI Mic Playback Volume", 0x07, 0x1, HDA_INPUT),
5285 HDA_CODEC_MUTE("ATATI Mic Playback Switch", 0x07, 0x1, HDA_INPUT),
5286 HDA_CODEC_VOLUME("Line Playback Volume", 0x07, 0x02, HDA_INPUT),
5287 HDA_CODEC_MUTE("Line Playback Switch", 0x07, 0x02, HDA_INPUT),
5288 ALC_PIN_MODE("Line Jack Mode", 0x14, ALC_PIN_DIR_INOUT),
5289 { } /* end */
5290};
5291
Kailang Yangdf694da2005-12-05 19:42:22 +01005292/*
5293 * initialization verbs
5294 */
Linus Torvalds1da177e2005-04-16 15:20:36 -07005295static struct hda_verb alc260_init_verbs[] = {
5296 /* Line In pin widget for input */
Takashi Iwai05acb862005-06-10 19:50:25 +02005297 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005298 /* CD pin widget for input */
Takashi Iwai05acb862005-06-10 19:50:25 +02005299 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005300 /* Mic1 (rear panel) pin widget for input and vref at 80% */
Takashi Iwai16ded522005-06-10 19:58:24 +02005301 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005302 /* Mic2 (front panel) pin widget for input and vref at 80% */
Takashi Iwai16ded522005-06-10 19:58:24 +02005303 {0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005304 /* LINE-2 is used for line-out in rear */
Takashi Iwai05acb862005-06-10 19:50:25 +02005305 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005306 /* select line-out */
Jonathan Woithefd56f2d2006-01-24 10:35:46 +01005307 {0x0e, AC_VERB_SET_CONNECT_SEL, 0x00},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005308 /* LINE-OUT pin */
Takashi Iwai05acb862005-06-10 19:50:25 +02005309 {0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005310 /* enable HP */
Takashi Iwai05acb862005-06-10 19:50:25 +02005311 {0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005312 /* enable Mono */
Takashi Iwai05acb862005-06-10 19:50:25 +02005313 {0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5314 /* mute capture amp left and right */
Takashi Iwai16ded522005-06-10 19:58:24 +02005315 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005316 /* set connection select to line in (default select for this ADC) */
5317 {0x04, AC_VERB_SET_CONNECT_SEL, 0x02},
Takashi Iwai16ded522005-06-10 19:58:24 +02005318 /* mute capture amp left and right */
5319 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5320 /* set connection select to line in (default select for this ADC) */
5321 {0x05, AC_VERB_SET_CONNECT_SEL, 0x02},
Takashi Iwai05acb862005-06-10 19:50:25 +02005322 /* set vol=0 Line-Out mixer amp left and right */
5323 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5324 /* unmute pin widget amp left and right (no gain on this amp) */
5325 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5326 /* set vol=0 HP mixer amp left and right */
5327 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5328 /* unmute pin widget amp left and right (no gain on this amp) */
5329 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5330 /* set vol=0 Mono mixer amp left and right */
5331 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5332 /* unmute pin widget amp left and right (no gain on this amp) */
5333 {0x11, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5334 /* unmute LINE-2 out pin */
5335 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02005336 /* Amp Indexes: CD = 0x04, Line In 1 = 0x02, Mic 1 = 0x00 &
5337 * Line In 2 = 0x03
5338 */
Takashi Iwaicb53c622007-08-10 17:21:45 +02005339 /* mute analog inputs */
5340 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5341 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5342 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
5343 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
5344 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005345 /* Amp Indexes: DAC = 0x01 & mixer = 0x00 */
Takashi Iwai05acb862005-06-10 19:50:25 +02005346 /* mute Front out path */
5347 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5348 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5349 /* mute Headphone out path */
5350 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5351 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5352 /* mute Mono out path */
5353 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5354 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07005355 { }
5356};
5357
Takashi Iwai474167d2006-05-17 17:17:43 +02005358#if 0 /* should be identical with alc260_init_verbs? */
Kailang Yangdf694da2005-12-05 19:42:22 +01005359static struct hda_verb alc260_hp_init_verbs[] = {
5360 /* Headphone and output */
5361 {0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc0},
5362 /* mono output */
5363 {0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40},
5364 /* Mic1 (rear panel) pin widget for input and vref at 80% */
5365 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
5366 /* Mic2 (front panel) pin widget for input and vref at 80% */
5367 {0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
5368 /* Line In pin widget for input */
5369 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
5370 /* Line-2 pin widget for output */
5371 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40},
5372 /* CD pin widget for input */
5373 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
5374 /* unmute amp left and right */
5375 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000},
5376 /* set connection select to line in (default select for this ADC) */
5377 {0x04, AC_VERB_SET_CONNECT_SEL, 0x02},
5378 /* unmute Line-Out mixer amp left and right (volume = 0) */
5379 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, 0xb000},
5380 /* mute pin widget amp left and right (no gain on this amp) */
5381 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
5382 /* unmute HP mixer amp left and right (volume = 0) */
5383 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, 0xb000},
5384 /* mute pin widget amp left and right (no gain on this amp) */
5385 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02005386 /* Amp Indexes: CD = 0x04, Line In 1 = 0x02, Mic 1 = 0x00 &
5387 * Line In 2 = 0x03
5388 */
Takashi Iwaicb53c622007-08-10 17:21:45 +02005389 /* mute analog inputs */
5390 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5391 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5392 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
5393 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
5394 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Kailang Yangdf694da2005-12-05 19:42:22 +01005395 /* Amp Indexes: DAC = 0x01 & mixer = 0x00 */
5396 /* Unmute Front out path */
5397 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
5398 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
5399 /* Unmute Headphone out path */
5400 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
5401 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
5402 /* Unmute Mono out path */
5403 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
5404 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
5405 { }
5406};
Takashi Iwai474167d2006-05-17 17:17:43 +02005407#endif
Kailang Yangdf694da2005-12-05 19:42:22 +01005408
5409static struct hda_verb alc260_hp_3013_init_verbs[] = {
5410 /* Line out and output */
5411 {0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40},
5412 /* mono output */
5413 {0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40},
5414 /* Mic1 (rear panel) pin widget for input and vref at 80% */
5415 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
5416 /* Mic2 (front panel) pin widget for input and vref at 80% */
5417 {0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
5418 /* Line In pin widget for input */
5419 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
5420 /* Headphone pin widget for output */
5421 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc0},
5422 /* CD pin widget for input */
5423 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
5424 /* unmute amp left and right */
5425 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000},
5426 /* set connection select to line in (default select for this ADC) */
5427 {0x04, AC_VERB_SET_CONNECT_SEL, 0x02},
5428 /* unmute Line-Out mixer amp left and right (volume = 0) */
5429 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, 0xb000},
5430 /* mute pin widget amp left and right (no gain on this amp) */
5431 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
5432 /* unmute HP mixer amp left and right (volume = 0) */
5433 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, 0xb000},
5434 /* mute pin widget amp left and right (no gain on this amp) */
5435 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02005436 /* Amp Indexes: CD = 0x04, Line In 1 = 0x02, Mic 1 = 0x00 &
5437 * Line In 2 = 0x03
5438 */
Takashi Iwaicb53c622007-08-10 17:21:45 +02005439 /* mute analog inputs */
5440 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5441 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5442 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
5443 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
5444 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Kailang Yangdf694da2005-12-05 19:42:22 +01005445 /* Amp Indexes: DAC = 0x01 & mixer = 0x00 */
5446 /* Unmute Front out path */
5447 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
5448 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
5449 /* Unmute Headphone out path */
5450 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
5451 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
5452 /* Unmute Mono out path */
5453 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
5454 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
5455 { }
5456};
5457
Jonathan Woithea9430dd2005-09-16 19:12:48 +02005458/* Initialisation sequence for ALC260 as configured in Fujitsu S702x
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02005459 * laptops. ALC260 pin usage: Mic/Line jack = 0x12, HP jack = 0x14, CD
5460 * audio = 0x16, internal speaker = 0x10.
Jonathan Woithea9430dd2005-09-16 19:12:48 +02005461 */
5462static struct hda_verb alc260_fujitsu_init_verbs[] = {
5463 /* Disable all GPIOs */
5464 {0x01, AC_VERB_SET_GPIO_MASK, 0},
5465 /* Internal speaker is connected to headphone pin */
5466 {0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
5467 /* Headphone/Line-out jack connects to Line1 pin; make it an output */
5468 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Jonathan Woithef7ace402006-02-28 11:46:14 +01005469 /* Mic/Line-in jack is connected to mic1 pin, so make it an input */
5470 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
5471 /* Ensure all other unused pins are disabled and muted. */
5472 {0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
5473 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Jonathan Woithea9430dd2005-09-16 19:12:48 +02005474 {0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
Jonathan Woithef7ace402006-02-28 11:46:14 +01005475 {0x11, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Jonathan Woithea9430dd2005-09-16 19:12:48 +02005476 {0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
Jonathan Woithef7ace402006-02-28 11:46:14 +01005477 {0x13, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5478 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
5479 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Jonathan Woithea9430dd2005-09-16 19:12:48 +02005480
Jonathan Woithef7ace402006-02-28 11:46:14 +01005481 /* Disable digital (SPDIF) pins */
5482 {0x03, AC_VERB_SET_DIGI_CONVERT_1, 0},
5483 {0x06, AC_VERB_SET_DIGI_CONVERT_1, 0},
Jonathan Woithe4c5186e2006-02-09 11:53:48 +01005484
Kailang Yangea1fb292008-08-26 12:58:38 +02005485 /* Ensure Line1 pin widget takes its input from the OUT1 sum bus
Jonathan Woithef7ace402006-02-28 11:46:14 +01005486 * when acting as an output.
5487 */
5488 {0x0d, AC_VERB_SET_CONNECT_SEL, 0},
5489
5490 /* Start with output sum widgets muted and their output gains at min */
Takashi Iwai8b33a5a2006-02-09 11:57:01 +01005491 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5492 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5493 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5494 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5495 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5496 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5497 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5498 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5499 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
Jonathan Woithea9430dd2005-09-16 19:12:48 +02005500
Jonathan Woithef7ace402006-02-28 11:46:14 +01005501 /* Unmute HP pin widget amp left and right (no equiv mixer ctrl) */
5502 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5503 /* Unmute Line1 pin widget output buffer since it starts as an output.
5504 * If the pin mode is changed by the user the pin mode control will
5505 * take care of enabling the pin's input/output buffers as needed.
5506 * Therefore there's no need to enable the input buffer at this
5507 * stage.
Jonathan Woithecdcd9262006-02-28 11:36:42 +01005508 */
Jonathan Woithef7ace402006-02-28 11:46:14 +01005509 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Kailang Yangea1fb292008-08-26 12:58:38 +02005510 /* Unmute input buffer of pin widget used for Line-in (no equiv
Jonathan Woithecdcd9262006-02-28 11:36:42 +01005511 * mixer ctrl)
5512 */
Jonathan Woithef7ace402006-02-28 11:46:14 +01005513 {0x12, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Jonathan Woithea9430dd2005-09-16 19:12:48 +02005514
Jonathan Woithef7ace402006-02-28 11:46:14 +01005515 /* Mute capture amp left and right */
5516 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Kailang Yangea1fb292008-08-26 12:58:38 +02005517 /* Set ADC connection select to match default mixer setting - line
Jonathan Woithef7ace402006-02-28 11:46:14 +01005518 * in (on mic1 pin)
5519 */
5520 {0x04, AC_VERB_SET_CONNECT_SEL, 0x00},
Jonathan Woithea9430dd2005-09-16 19:12:48 +02005521
Jonathan Woithef7ace402006-02-28 11:46:14 +01005522 /* Do the same for the second ADC: mute capture input amp and
5523 * set ADC connection to line in (on mic1 pin)
5524 */
5525 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5526 {0x05, AC_VERB_SET_CONNECT_SEL, 0x00},
Jonathan Woithe4c5186e2006-02-09 11:53:48 +01005527
Jonathan Woithef7ace402006-02-28 11:46:14 +01005528 /* Mute all inputs to mixer widget (even unconnected ones) */
5529 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)}, /* mic1 pin */
5530 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)}, /* mic2 pin */
5531 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)}, /* line1 pin */
5532 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)}, /* line2 pin */
5533 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)}, /* CD pin */
5534 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(5)}, /* Beep-gen pin */
5535 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(6)}, /* Line-out pin */
5536 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(7)}, /* HP-pin pin */
Takashi Iwai4a471b72005-12-07 13:56:29 +01005537
5538 { }
Jonathan Woithea9430dd2005-09-16 19:12:48 +02005539};
5540
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01005541/* Initialisation sequence for ALC260 as configured in Acer TravelMate and
5542 * similar laptops (adapted from Fujitsu init verbs).
5543 */
5544static struct hda_verb alc260_acer_init_verbs[] = {
5545 /* On TravelMate laptops, GPIO 0 enables the internal speaker and
5546 * the headphone jack. Turn this on and rely on the standard mute
5547 * methods whenever the user wants to turn these outputs off.
5548 */
5549 {0x01, AC_VERB_SET_GPIO_MASK, 0x01},
5550 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x01},
5551 {0x01, AC_VERB_SET_GPIO_DATA, 0x01},
5552 /* Internal speaker/Headphone jack is connected to Line-out pin */
5553 {0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
5554 /* Internal microphone/Mic jack is connected to Mic1 pin */
5555 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF50},
5556 /* Line In jack is connected to Line1 pin */
5557 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Jonathan Woithebd869482006-11-28 11:35:52 +01005558 /* Some Acers (eg: C20x Tablets) use Mono pin for internal speaker */
5559 {0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01005560 /* Ensure all other unused pins are disabled and muted. */
5561 {0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
5562 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01005563 {0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
5564 {0x13, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5565 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
5566 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5567 /* Disable digital (SPDIF) pins */
5568 {0x03, AC_VERB_SET_DIGI_CONVERT_1, 0},
5569 {0x06, AC_VERB_SET_DIGI_CONVERT_1, 0},
5570
Kailang Yangea1fb292008-08-26 12:58:38 +02005571 /* Ensure Mic1 and Line1 pin widgets take input from the OUT1 sum
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01005572 * bus when acting as outputs.
5573 */
5574 {0x0b, AC_VERB_SET_CONNECT_SEL, 0},
5575 {0x0d, AC_VERB_SET_CONNECT_SEL, 0},
5576
5577 /* Start with output sum widgets muted and their output gains at min */
5578 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5579 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5580 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5581 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5582 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5583 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5584 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5585 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5586 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5587
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02005588 /* Unmute Line-out pin widget amp left and right
5589 * (no equiv mixer ctrl)
5590 */
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01005591 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Jonathan Woithebd869482006-11-28 11:35:52 +01005592 /* Unmute mono pin widget amp output (no equiv mixer ctrl) */
5593 {0x11, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01005594 /* Unmute Mic1 and Line1 pin widget input buffers since they start as
5595 * inputs. If the pin mode is changed by the user the pin mode control
5596 * will take care of enabling the pin's input/output buffers as needed.
5597 * Therefore there's no need to enable the input buffer at this
5598 * stage.
5599 */
5600 {0x12, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5601 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5602
5603 /* Mute capture amp left and right */
5604 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5605 /* Set ADC connection select to match default mixer setting - mic
5606 * (on mic1 pin)
5607 */
5608 {0x04, AC_VERB_SET_CONNECT_SEL, 0x00},
5609
5610 /* Do similar with the second ADC: mute capture input amp and
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02005611 * set ADC connection to mic to match ALSA's default state.
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01005612 */
5613 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02005614 {0x05, AC_VERB_SET_CONNECT_SEL, 0x00},
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01005615
5616 /* Mute all inputs to mixer widget (even unconnected ones) */
5617 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)}, /* mic1 pin */
5618 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)}, /* mic2 pin */
5619 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)}, /* line1 pin */
5620 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)}, /* line2 pin */
5621 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)}, /* CD pin */
5622 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(5)}, /* Beep-gen pin */
5623 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(6)}, /* Line-out pin */
5624 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(7)}, /* HP-pin pin */
5625
5626 { }
5627};
5628
Michael Schwingencc959482009-02-22 18:58:45 +01005629/* Initialisation sequence for Maxdata Favorit 100XS
5630 * (adapted from Acer init verbs).
5631 */
5632static struct hda_verb alc260_favorit100_init_verbs[] = {
5633 /* GPIO 0 enables the output jack.
5634 * Turn this on and rely on the standard mute
5635 * methods whenever the user wants to turn these outputs off.
5636 */
5637 {0x01, AC_VERB_SET_GPIO_MASK, 0x01},
5638 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x01},
5639 {0x01, AC_VERB_SET_GPIO_DATA, 0x01},
5640 /* Line/Mic input jack is connected to Mic1 pin */
5641 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF50},
5642 /* Ensure all other unused pins are disabled and muted. */
5643 {0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
5644 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5645 {0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
5646 {0x11, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5647 {0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
5648 {0x13, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5649 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
5650 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5651 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
5652 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5653 /* Disable digital (SPDIF) pins */
5654 {0x03, AC_VERB_SET_DIGI_CONVERT_1, 0},
5655 {0x06, AC_VERB_SET_DIGI_CONVERT_1, 0},
5656
5657 /* Ensure Mic1 and Line1 pin widgets take input from the OUT1 sum
5658 * bus when acting as outputs.
5659 */
5660 {0x0b, AC_VERB_SET_CONNECT_SEL, 0},
5661 {0x0d, AC_VERB_SET_CONNECT_SEL, 0},
5662
5663 /* Start with output sum widgets muted and their output gains at min */
5664 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5665 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5666 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5667 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5668 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5669 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5670 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5671 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5672 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5673
5674 /* Unmute Line-out pin widget amp left and right
5675 * (no equiv mixer ctrl)
5676 */
5677 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5678 /* Unmute Mic1 and Line1 pin widget input buffers since they start as
5679 * inputs. If the pin mode is changed by the user the pin mode control
5680 * will take care of enabling the pin's input/output buffers as needed.
5681 * Therefore there's no need to enable the input buffer at this
5682 * stage.
5683 */
5684 {0x12, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
5685
5686 /* Mute capture amp left and right */
5687 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5688 /* Set ADC connection select to match default mixer setting - mic
5689 * (on mic1 pin)
5690 */
5691 {0x04, AC_VERB_SET_CONNECT_SEL, 0x00},
5692
5693 /* Do similar with the second ADC: mute capture input amp and
5694 * set ADC connection to mic to match ALSA's default state.
5695 */
5696 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5697 {0x05, AC_VERB_SET_CONNECT_SEL, 0x00},
5698
5699 /* Mute all inputs to mixer widget (even unconnected ones) */
5700 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)}, /* mic1 pin */
5701 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)}, /* mic2 pin */
5702 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)}, /* line1 pin */
5703 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)}, /* line2 pin */
5704 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)}, /* CD pin */
5705 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(5)}, /* Beep-gen pin */
5706 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(6)}, /* Line-out pin */
5707 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(7)}, /* HP-pin pin */
5708
5709 { }
5710};
5711
Kailang Yangbc9f98a2007-04-12 13:06:07 +02005712static struct hda_verb alc260_will_verbs[] = {
5713 {0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
5714 {0x0b, AC_VERB_SET_CONNECT_SEL, 0x00},
5715 {0x0d, AC_VERB_SET_CONNECT_SEL, 0x00},
5716 {0x0f, AC_VERB_SET_EAPD_BTLENABLE, 0x02},
5717 {0x1a, AC_VERB_SET_COEF_INDEX, 0x07},
5718 {0x1a, AC_VERB_SET_PROC_COEF, 0x3040},
5719 {}
5720};
5721
5722static struct hda_verb alc260_replacer_672v_verbs[] = {
5723 {0x0f, AC_VERB_SET_EAPD_BTLENABLE, 0x02},
5724 {0x1a, AC_VERB_SET_COEF_INDEX, 0x07},
5725 {0x1a, AC_VERB_SET_PROC_COEF, 0x3050},
5726
5727 {0x01, AC_VERB_SET_GPIO_MASK, 0x01},
5728 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x01},
5729 {0x01, AC_VERB_SET_GPIO_DATA, 0x00},
5730
5731 {0x0f, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
5732 {}
5733};
5734
5735/* toggle speaker-output according to the hp-jack state */
5736static void alc260_replacer_672v_automute(struct hda_codec *codec)
5737{
5738 unsigned int present;
5739
5740 /* speaker --> GPIO Data 0, hp or spdif --> GPIO data 1 */
Wu Fengguang864f92b2009-11-18 12:38:02 +08005741 present = snd_hda_jack_detect(codec, 0x0f);
Kailang Yangbc9f98a2007-04-12 13:06:07 +02005742 if (present) {
Takashi Iwai82beb8f2007-08-10 17:09:26 +02005743 snd_hda_codec_write_cache(codec, 0x01, 0,
5744 AC_VERB_SET_GPIO_DATA, 1);
5745 snd_hda_codec_write_cache(codec, 0x0f, 0,
5746 AC_VERB_SET_PIN_WIDGET_CONTROL,
5747 PIN_HP);
Kailang Yangbc9f98a2007-04-12 13:06:07 +02005748 } else {
Takashi Iwai82beb8f2007-08-10 17:09:26 +02005749 snd_hda_codec_write_cache(codec, 0x01, 0,
5750 AC_VERB_SET_GPIO_DATA, 0);
5751 snd_hda_codec_write_cache(codec, 0x0f, 0,
5752 AC_VERB_SET_PIN_WIDGET_CONTROL,
5753 PIN_OUT);
Kailang Yangbc9f98a2007-04-12 13:06:07 +02005754 }
5755}
5756
5757static void alc260_replacer_672v_unsol_event(struct hda_codec *codec,
5758 unsigned int res)
5759{
5760 if ((res >> 26) == ALC880_HP_EVENT)
5761 alc260_replacer_672v_automute(codec);
5762}
5763
Kailang Yang3f878302008-08-26 13:02:23 +02005764static struct hda_verb alc260_hp_dc7600_verbs[] = {
5765 {0x05, AC_VERB_SET_CONNECT_SEL, 0x01},
5766 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01},
5767 {0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5768 {0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
5769 {0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5770 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5771 {0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
5772 {0x10, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
5773 {0x11, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
5774 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
5775 {}
5776};
5777
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01005778/* Test configuration for debugging, modelled after the ALC880 test
5779 * configuration.
5780 */
5781#ifdef CONFIG_SND_DEBUG
5782static hda_nid_t alc260_test_dac_nids[1] = {
5783 0x02,
5784};
5785static hda_nid_t alc260_test_adc_nids[2] = {
5786 0x04, 0x05,
5787};
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02005788/* For testing the ALC260, each input MUX needs its own definition since
Kailang Yangea1fb292008-08-26 12:58:38 +02005789 * the signal assignments are different. This assumes that the first ADC
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02005790 * is NID 0x04.
Jonathan Woithe17e7aec2006-02-28 11:35:18 +01005791 */
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02005792static struct hda_input_mux alc260_test_capture_sources[2] = {
5793 {
5794 .num_items = 7,
5795 .items = {
5796 { "MIC1 pin", 0x0 },
5797 { "MIC2 pin", 0x1 },
5798 { "LINE1 pin", 0x2 },
5799 { "LINE2 pin", 0x3 },
5800 { "CD pin", 0x4 },
5801 { "LINE-OUT pin", 0x5 },
5802 { "HP-OUT pin", 0x6 },
5803 },
5804 },
5805 {
5806 .num_items = 8,
5807 .items = {
5808 { "MIC1 pin", 0x0 },
5809 { "MIC2 pin", 0x1 },
5810 { "LINE1 pin", 0x2 },
5811 { "LINE2 pin", 0x3 },
5812 { "CD pin", 0x4 },
5813 { "Mixer", 0x5 },
5814 { "LINE-OUT pin", 0x6 },
5815 { "HP-OUT pin", 0x7 },
5816 },
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01005817 },
5818};
5819static struct snd_kcontrol_new alc260_test_mixer[] = {
5820 /* Output driver widgets */
5821 HDA_CODEC_VOLUME_MONO("Mono Playback Volume", 0x0a, 1, 0x0, HDA_OUTPUT),
5822 HDA_BIND_MUTE_MONO("Mono Playback Switch", 0x0a, 1, 2, HDA_INPUT),
5823 HDA_CODEC_VOLUME("LOUT2 Playback Volume", 0x09, 0x0, HDA_OUTPUT),
5824 HDA_BIND_MUTE("LOUT2 Playback Switch", 0x09, 2, HDA_INPUT),
5825 HDA_CODEC_VOLUME("LOUT1 Playback Volume", 0x08, 0x0, HDA_OUTPUT),
5826 HDA_BIND_MUTE("LOUT1 Playback Switch", 0x08, 2, HDA_INPUT),
5827
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02005828 /* Modes for retasking pin widgets
5829 * Note: the ALC260 doesn't seem to act on requests to enable mic
5830 * bias from NIDs 0x0f and 0x10. The ALC260 datasheet doesn't
5831 * mention this restriction. At this stage it's not clear whether
5832 * this behaviour is intentional or is a hardware bug in chip
5833 * revisions available at least up until early 2006. Therefore for
5834 * now allow the "HP-OUT" and "LINE-OUT" Mode controls to span all
5835 * choices, but if it turns out that the lack of mic bias for these
5836 * NIDs is intentional we could change their modes from
5837 * ALC_PIN_DIR_INOUT to ALC_PIN_DIR_INOUT_NOMICBIAS.
5838 */
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01005839 ALC_PIN_MODE("HP-OUT pin mode", 0x10, ALC_PIN_DIR_INOUT),
5840 ALC_PIN_MODE("LINE-OUT pin mode", 0x0f, ALC_PIN_DIR_INOUT),
5841 ALC_PIN_MODE("LINE2 pin mode", 0x15, ALC_PIN_DIR_INOUT),
5842 ALC_PIN_MODE("LINE1 pin mode", 0x14, ALC_PIN_DIR_INOUT),
5843 ALC_PIN_MODE("MIC2 pin mode", 0x13, ALC_PIN_DIR_INOUT),
5844 ALC_PIN_MODE("MIC1 pin mode", 0x12, ALC_PIN_DIR_INOUT),
5845
5846 /* Loopback mixer controls */
5847 HDA_CODEC_VOLUME("MIC1 Playback Volume", 0x07, 0x00, HDA_INPUT),
5848 HDA_CODEC_MUTE("MIC1 Playback Switch", 0x07, 0x00, HDA_INPUT),
5849 HDA_CODEC_VOLUME("MIC2 Playback Volume", 0x07, 0x01, HDA_INPUT),
5850 HDA_CODEC_MUTE("MIC2 Playback Switch", 0x07, 0x01, HDA_INPUT),
5851 HDA_CODEC_VOLUME("LINE1 Playback Volume", 0x07, 0x02, HDA_INPUT),
5852 HDA_CODEC_MUTE("LINE1 Playback Switch", 0x07, 0x02, HDA_INPUT),
5853 HDA_CODEC_VOLUME("LINE2 Playback Volume", 0x07, 0x03, HDA_INPUT),
5854 HDA_CODEC_MUTE("LINE2 Playback Switch", 0x07, 0x03, HDA_INPUT),
5855 HDA_CODEC_VOLUME("CD Playback Volume", 0x07, 0x04, HDA_INPUT),
5856 HDA_CODEC_MUTE("CD Playback Switch", 0x07, 0x04, HDA_INPUT),
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01005857 HDA_CODEC_VOLUME("LINE-OUT loopback Playback Volume", 0x07, 0x06, HDA_INPUT),
5858 HDA_CODEC_MUTE("LINE-OUT loopback Playback Switch", 0x07, 0x06, HDA_INPUT),
5859 HDA_CODEC_VOLUME("HP-OUT loopback Playback Volume", 0x07, 0x7, HDA_INPUT),
5860 HDA_CODEC_MUTE("HP-OUT loopback Playback Switch", 0x07, 0x7, HDA_INPUT),
Jonathan Woithe5c8f8582006-02-28 11:43:27 +01005861
5862 /* Controls for GPIO pins, assuming they are configured as outputs */
5863 ALC_GPIO_DATA_SWITCH("GPIO pin 0", 0x01, 0x01),
5864 ALC_GPIO_DATA_SWITCH("GPIO pin 1", 0x01, 0x02),
5865 ALC_GPIO_DATA_SWITCH("GPIO pin 2", 0x01, 0x04),
5866 ALC_GPIO_DATA_SWITCH("GPIO pin 3", 0x01, 0x08),
5867
Jonathan Woithe92621f12006-02-28 11:47:47 +01005868 /* Switches to allow the digital IO pins to be enabled. The datasheet
5869 * is ambigious as to which NID is which; testing on laptops which
Kailang Yangea1fb292008-08-26 12:58:38 +02005870 * make this output available should provide clarification.
Jonathan Woithe92621f12006-02-28 11:47:47 +01005871 */
5872 ALC_SPDIF_CTRL_SWITCH("SPDIF Playback Switch", 0x03, 0x01),
5873 ALC_SPDIF_CTRL_SWITCH("SPDIF Capture Switch", 0x06, 0x01),
5874
Jonathan Woithef8225f62008-01-08 12:16:54 +01005875 /* A switch allowing EAPD to be enabled. Some laptops seem to use
5876 * this output to turn on an external amplifier.
5877 */
5878 ALC_EAPD_CTRL_SWITCH("LINE-OUT EAPD Enable Switch", 0x0f, 0x02),
5879 ALC_EAPD_CTRL_SWITCH("HP-OUT EAPD Enable Switch", 0x10, 0x02),
5880
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01005881 { } /* end */
5882};
5883static struct hda_verb alc260_test_init_verbs[] = {
Jonathan Woithe5c8f8582006-02-28 11:43:27 +01005884 /* Enable all GPIOs as outputs with an initial value of 0 */
5885 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x0f},
5886 {0x01, AC_VERB_SET_GPIO_DATA, 0x00},
5887 {0x01, AC_VERB_SET_GPIO_MASK, 0x0f},
5888
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01005889 /* Enable retasking pins as output, initially without power amp */
5890 {0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5891 {0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5892 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5893 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5894 {0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5895 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
5896
Jonathan Woithe92621f12006-02-28 11:47:47 +01005897 /* Disable digital (SPDIF) pins initially, but users can enable
5898 * them via a mixer switch. In the case of SPDIF-out, this initverb
5899 * payload also sets the generation to 0, output to be in "consumer"
5900 * PCM format, copyright asserted, no pre-emphasis and no validity
5901 * control.
5902 */
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01005903 {0x03, AC_VERB_SET_DIGI_CONVERT_1, 0},
5904 {0x06, AC_VERB_SET_DIGI_CONVERT_1, 0},
5905
Kailang Yangea1fb292008-08-26 12:58:38 +02005906 /* Ensure mic1, mic2, line1 and line2 pin widgets take input from the
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01005907 * OUT1 sum bus when acting as an output.
5908 */
5909 {0x0b, AC_VERB_SET_CONNECT_SEL, 0},
5910 {0x0c, AC_VERB_SET_CONNECT_SEL, 0},
5911 {0x0d, AC_VERB_SET_CONNECT_SEL, 0},
5912 {0x0e, AC_VERB_SET_CONNECT_SEL, 0},
5913
5914 /* Start with output sum widgets muted and their output gains at min */
5915 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5916 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5917 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5918 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5919 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5920 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5921 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5922 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
5923 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
5924
Jonathan Woithecdcd9262006-02-28 11:36:42 +01005925 /* Unmute retasking pin widget output buffers since the default
5926 * state appears to be output. As the pin mode is changed by the
5927 * user the pin mode control will take care of enabling the pin's
5928 * input/output buffers as needed.
5929 */
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01005930 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5931 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5932 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5933 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5934 {0x13, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5935 {0x12, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5936 /* Also unmute the mono-out pin widget */
5937 {0x11, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
5938
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01005939 /* Mute capture amp left and right */
5940 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Jonathan Woithef7ace402006-02-28 11:46:14 +01005941 /* Set ADC connection select to match default mixer setting (mic1
5942 * pin)
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01005943 */
5944 {0x04, AC_VERB_SET_CONNECT_SEL, 0x00},
5945
5946 /* Do the same for the second ADC: mute capture input amp and
Jonathan Woithef7ace402006-02-28 11:46:14 +01005947 * set ADC connection to mic1 pin
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01005948 */
5949 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
5950 {0x05, AC_VERB_SET_CONNECT_SEL, 0x00},
5951
5952 /* Mute all inputs to mixer widget (even unconnected ones) */
5953 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)}, /* mic1 pin */
5954 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)}, /* mic2 pin */
5955 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)}, /* line1 pin */
5956 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)}, /* line2 pin */
5957 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)}, /* CD pin */
5958 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(5)}, /* Beep-gen pin */
5959 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(6)}, /* Line-out pin */
5960 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(7)}, /* HP-pin pin */
5961
5962 { }
5963};
5964#endif
5965
Takashi Iwai63300792008-01-24 15:31:36 +01005966#define alc260_pcm_analog_playback alc880_pcm_analog_alt_playback
5967#define alc260_pcm_analog_capture alc880_pcm_analog_capture
Linus Torvalds1da177e2005-04-16 15:20:36 -07005968
Takashi Iwaia3bcba32005-12-06 19:05:29 +01005969#define alc260_pcm_digital_playback alc880_pcm_digital_playback
5970#define alc260_pcm_digital_capture alc880_pcm_digital_capture
5971
Kailang Yangdf694da2005-12-05 19:42:22 +01005972/*
5973 * for BIOS auto-configuration
5974 */
5975
5976static int alc260_add_playback_controls(struct alc_spec *spec, hda_nid_t nid,
Takashi Iwai863b4512008-10-21 17:01:47 +02005977 const char *pfx, int *vol_bits)
Kailang Yangdf694da2005-12-05 19:42:22 +01005978{
5979 hda_nid_t nid_vol;
5980 unsigned long vol_val, sw_val;
Kailang Yangdf694da2005-12-05 19:42:22 +01005981 int err;
5982
5983 if (nid >= 0x0f && nid < 0x11) {
5984 nid_vol = nid - 0x7;
5985 vol_val = HDA_COMPOSE_AMP_VAL(nid_vol, 3, 0, HDA_OUTPUT);
5986 sw_val = HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_OUTPUT);
5987 } else if (nid == 0x11) {
5988 nid_vol = nid - 0x7;
5989 vol_val = HDA_COMPOSE_AMP_VAL(nid_vol, 2, 0, HDA_OUTPUT);
5990 sw_val = HDA_COMPOSE_AMP_VAL(nid, 2, 0, HDA_OUTPUT);
5991 } else if (nid >= 0x12 && nid <= 0x15) {
5992 nid_vol = 0x08;
5993 vol_val = HDA_COMPOSE_AMP_VAL(nid_vol, 3, 0, HDA_OUTPUT);
5994 sw_val = HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_OUTPUT);
5995 } else
5996 return 0; /* N/A */
Kailang Yangea1fb292008-08-26 12:58:38 +02005997
Takashi Iwai863b4512008-10-21 17:01:47 +02005998 if (!(*vol_bits & (1 << nid_vol))) {
5999 /* first control for the volume widget */
Takashi Iwai0afe5f82009-10-02 09:20:00 +02006000 err = add_pb_vol_ctrl(spec, ALC_CTL_WIDGET_VOL, pfx, vol_val);
Takashi Iwai863b4512008-10-21 17:01:47 +02006001 if (err < 0)
6002 return err;
6003 *vol_bits |= (1 << nid_vol);
6004 }
Takashi Iwai0afe5f82009-10-02 09:20:00 +02006005 err = add_pb_sw_ctrl(spec, ALC_CTL_WIDGET_MUTE, pfx, sw_val);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006006 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +01006007 return err;
6008 return 1;
6009}
6010
6011/* add playback controls from the parsed DAC table */
6012static int alc260_auto_create_multi_out_ctls(struct alc_spec *spec,
6013 const struct auto_pin_cfg *cfg)
6014{
6015 hda_nid_t nid;
6016 int err;
Takashi Iwai863b4512008-10-21 17:01:47 +02006017 int vols = 0;
Kailang Yangdf694da2005-12-05 19:42:22 +01006018
6019 spec->multiout.num_dacs = 1;
6020 spec->multiout.dac_nids = spec->private_dac_nids;
6021 spec->multiout.dac_nids[0] = 0x02;
6022
6023 nid = cfg->line_out_pins[0];
6024 if (nid) {
Takashi Iwai23112d62009-08-25 16:07:08 +02006025 const char *pfx;
6026 if (!cfg->speaker_pins[0] && !cfg->hp_pins[0])
6027 pfx = "Master";
6028 else if (cfg->line_out_type == AUTO_PIN_SPEAKER_OUT)
6029 pfx = "Speaker";
6030 else
6031 pfx = "Front";
6032 err = alc260_add_playback_controls(spec, nid, pfx, &vols);
Kailang Yangdf694da2005-12-05 19:42:22 +01006033 if (err < 0)
6034 return err;
6035 }
6036
Takashi Iwai82bc9552006-03-21 11:24:42 +01006037 nid = cfg->speaker_pins[0];
Kailang Yangdf694da2005-12-05 19:42:22 +01006038 if (nid) {
Takashi Iwai863b4512008-10-21 17:01:47 +02006039 err = alc260_add_playback_controls(spec, nid, "Speaker", &vols);
Kailang Yangdf694da2005-12-05 19:42:22 +01006040 if (err < 0)
6041 return err;
6042 }
6043
Takashi Iwaieb06ed82006-09-20 17:10:27 +02006044 nid = cfg->hp_pins[0];
Kailang Yangdf694da2005-12-05 19:42:22 +01006045 if (nid) {
Takashi Iwai863b4512008-10-21 17:01:47 +02006046 err = alc260_add_playback_controls(spec, nid, "Headphone",
6047 &vols);
Kailang Yangdf694da2005-12-05 19:42:22 +01006048 if (err < 0)
6049 return err;
6050 }
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006051 return 0;
Kailang Yangdf694da2005-12-05 19:42:22 +01006052}
6053
6054/* create playback/capture controls for input pins */
Takashi Iwai05f5f472009-08-25 13:10:18 +02006055static int alc260_auto_create_input_ctls(struct hda_codec *codec,
Kailang Yangdf694da2005-12-05 19:42:22 +01006056 const struct auto_pin_cfg *cfg)
6057{
Takashi Iwai05f5f472009-08-25 13:10:18 +02006058 return alc_auto_create_input_ctls(codec, cfg, 0x07, 0x04, 0x05);
Kailang Yangdf694da2005-12-05 19:42:22 +01006059}
6060
6061static void alc260_auto_set_output_and_unmute(struct hda_codec *codec,
6062 hda_nid_t nid, int pin_type,
6063 int sel_idx)
6064{
Takashi Iwaif6c7e542008-02-12 18:32:23 +01006065 alc_set_pin_output(codec, nid, pin_type);
Kailang Yangdf694da2005-12-05 19:42:22 +01006066 /* need the manual connection? */
6067 if (nid >= 0x12) {
6068 int idx = nid - 0x12;
6069 snd_hda_codec_write(codec, idx + 0x0b, 0,
6070 AC_VERB_SET_CONNECT_SEL, sel_idx);
Kailang Yangdf694da2005-12-05 19:42:22 +01006071 }
6072}
6073
6074static void alc260_auto_init_multi_out(struct hda_codec *codec)
6075{
6076 struct alc_spec *spec = codec->spec;
6077 hda_nid_t nid;
6078
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006079 nid = spec->autocfg.line_out_pins[0];
Takashi Iwaibaba8ee2007-04-23 17:17:48 +02006080 if (nid) {
6081 int pin_type = get_pin_type(spec->autocfg.line_out_type);
6082 alc260_auto_set_output_and_unmute(codec, nid, pin_type, 0);
6083 }
Kailang Yangea1fb292008-08-26 12:58:38 +02006084
Takashi Iwai82bc9552006-03-21 11:24:42 +01006085 nid = spec->autocfg.speaker_pins[0];
Kailang Yangdf694da2005-12-05 19:42:22 +01006086 if (nid)
6087 alc260_auto_set_output_and_unmute(codec, nid, PIN_OUT, 0);
6088
Takashi Iwaieb06ed82006-09-20 17:10:27 +02006089 nid = spec->autocfg.hp_pins[0];
Kailang Yangdf694da2005-12-05 19:42:22 +01006090 if (nid)
Takashi Iwaibaba8ee2007-04-23 17:17:48 +02006091 alc260_auto_set_output_and_unmute(codec, nid, PIN_HP, 0);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006092}
Kailang Yangdf694da2005-12-05 19:42:22 +01006093
6094#define ALC260_PIN_CD_NID 0x16
6095static void alc260_auto_init_analog_input(struct hda_codec *codec)
6096{
6097 struct alc_spec *spec = codec->spec;
6098 int i;
6099
6100 for (i = 0; i < AUTO_PIN_LAST; i++) {
6101 hda_nid_t nid = spec->autocfg.input_pins[i];
6102 if (nid >= 0x12) {
Takashi Iwai23f0c042009-02-26 13:03:58 +01006103 alc_set_input_pin(codec, nid, i);
Takashi Iwaie82c0252009-03-23 15:17:38 +01006104 if (nid != ALC260_PIN_CD_NID &&
6105 (get_wcaps(codec, nid) & AC_WCAP_OUT_AMP))
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006106 snd_hda_codec_write(codec, nid, 0,
6107 AC_VERB_SET_AMP_GAIN_MUTE,
Kailang Yangdf694da2005-12-05 19:42:22 +01006108 AMP_OUT_MUTE);
6109 }
6110 }
6111}
6112
6113/*
6114 * generic initialization of ADC, input mixers and output mixers
6115 */
6116static struct hda_verb alc260_volume_init_verbs[] = {
6117 /*
6118 * Unmute ADC0-1 and set the default input to mic-in
6119 */
6120 {0x04, AC_VERB_SET_CONNECT_SEL, 0x00},
6121 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
6122 {0x05, AC_VERB_SET_CONNECT_SEL, 0x00},
6123 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Kailang Yangea1fb292008-08-26 12:58:38 +02006124
Kailang Yangdf694da2005-12-05 19:42:22 +01006125 /* Unmute input amps (CD, Line In, Mic 1 & Mic 2) of the analog-loopback
6126 * mixer widget
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006127 * Note: PASD motherboards uses the Line In 2 as the input for
6128 * front panel mic (mic 2)
Kailang Yangdf694da2005-12-05 19:42:22 +01006129 */
6130 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +02006131 /* mute analog inputs */
6132 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
6133 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
6134 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
6135 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
6136 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Kailang Yangdf694da2005-12-05 19:42:22 +01006137
6138 /*
6139 * Set up output mixers (0x08 - 0x0a)
6140 */
6141 /* set vol=0 to output mixers */
6142 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
6143 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
6144 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
6145 /* set up input amps for analog loopback */
6146 /* Amp Indices: DAC = 0, mixer = 1 */
6147 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
6148 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
6149 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
6150 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
6151 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
6152 {0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Kailang Yangea1fb292008-08-26 12:58:38 +02006153
Kailang Yangdf694da2005-12-05 19:42:22 +01006154 { }
6155};
6156
6157static int alc260_parse_auto_config(struct hda_codec *codec)
6158{
6159 struct alc_spec *spec = codec->spec;
Kailang Yangdf694da2005-12-05 19:42:22 +01006160 int err;
6161 static hda_nid_t alc260_ignore[] = { 0x17, 0 };
6162
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006163 err = snd_hda_parse_pin_def_config(codec, &spec->autocfg,
6164 alc260_ignore);
6165 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +01006166 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006167 err = alc260_auto_create_multi_out_ctls(spec, &spec->autocfg);
6168 if (err < 0)
Takashi Iwai4a471b72005-12-07 13:56:29 +01006169 return err;
Takashi Iwai603c4012008-07-30 15:01:44 +02006170 if (!spec->kctls.list)
Kailang Yangdf694da2005-12-05 19:42:22 +01006171 return 0; /* can't find valid BIOS pin config */
Takashi Iwai05f5f472009-08-25 13:10:18 +02006172 err = alc260_auto_create_input_ctls(codec, &spec->autocfg);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006173 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +01006174 return err;
6175
6176 spec->multiout.max_channels = 2;
6177
Takashi Iwai0852d7a2009-02-11 11:35:15 +01006178 if (spec->autocfg.dig_outs)
Kailang Yangdf694da2005-12-05 19:42:22 +01006179 spec->multiout.dig_out_nid = ALC260_DIGOUT_NID;
Takashi Iwai603c4012008-07-30 15:01:44 +02006180 if (spec->kctls.list)
Takashi Iwaid88897e2008-10-31 15:01:37 +01006181 add_mixer(spec, spec->kctls.list);
Kailang Yangdf694da2005-12-05 19:42:22 +01006182
Takashi Iwaid88897e2008-10-31 15:01:37 +01006183 add_verb(spec, alc260_volume_init_verbs);
Kailang Yangdf694da2005-12-05 19:42:22 +01006184
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02006185 spec->num_mux_defs = 1;
Herton Ronaldo Krzesinski61b9b9b2009-01-28 09:16:33 -02006186 spec->input_mux = &spec->private_imux[0];
Kailang Yangdf694da2005-12-05 19:42:22 +01006187
Takashi Iwai4a79ba32009-04-22 16:31:35 +02006188 alc_ssid_check(codec, 0x10, 0x15, 0x0f);
6189
Kailang Yangdf694da2005-12-05 19:42:22 +01006190 return 1;
6191}
6192
Takashi Iwaiae6b8132006-03-03 16:47:17 +01006193/* additional initialization for auto-configuration model */
6194static void alc260_auto_init(struct hda_codec *codec)
Kailang Yangdf694da2005-12-05 19:42:22 +01006195{
Takashi Iwaif6c7e542008-02-12 18:32:23 +01006196 struct alc_spec *spec = codec->spec;
Kailang Yangdf694da2005-12-05 19:42:22 +01006197 alc260_auto_init_multi_out(codec);
6198 alc260_auto_init_analog_input(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +01006199 if (spec->unsol_event)
Kailang Yang7fb0d782008-10-15 11:12:35 +02006200 alc_inithook(codec);
Kailang Yangdf694da2005-12-05 19:42:22 +01006201}
6202
Takashi Iwaicb53c622007-08-10 17:21:45 +02006203#ifdef CONFIG_SND_HDA_POWER_SAVE
6204static struct hda_amp_list alc260_loopbacks[] = {
6205 { 0x07, HDA_INPUT, 0 },
6206 { 0x07, HDA_INPUT, 1 },
6207 { 0x07, HDA_INPUT, 2 },
6208 { 0x07, HDA_INPUT, 3 },
6209 { 0x07, HDA_INPUT, 4 },
6210 { } /* end */
6211};
6212#endif
6213
Kailang Yangdf694da2005-12-05 19:42:22 +01006214/*
6215 * ALC260 configurations
6216 */
Takashi Iwaif5fcc132006-11-24 17:07:44 +01006217static const char *alc260_models[ALC260_MODEL_LAST] = {
6218 [ALC260_BASIC] = "basic",
6219 [ALC260_HP] = "hp",
6220 [ALC260_HP_3013] = "hp-3013",
Takashi Iwai2922c9a2008-08-27 18:12:42 +02006221 [ALC260_HP_DC7600] = "hp-dc7600",
Takashi Iwaif5fcc132006-11-24 17:07:44 +01006222 [ALC260_FUJITSU_S702X] = "fujitsu",
6223 [ALC260_ACER] = "acer",
Kailang Yangbc9f98a2007-04-12 13:06:07 +02006224 [ALC260_WILL] = "will",
6225 [ALC260_REPLACER_672V] = "replacer",
Michael Schwingencc959482009-02-22 18:58:45 +01006226 [ALC260_FAVORIT100] = "favorit100",
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01006227#ifdef CONFIG_SND_DEBUG
Takashi Iwaif5fcc132006-11-24 17:07:44 +01006228 [ALC260_TEST] = "test",
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01006229#endif
Takashi Iwaif5fcc132006-11-24 17:07:44 +01006230 [ALC260_AUTO] = "auto",
6231};
6232
6233static struct snd_pci_quirk alc260_cfg_tbl[] = {
Jonathan Woithebd869482006-11-28 11:35:52 +01006234 SND_PCI_QUIRK(0x1025, 0x007b, "Acer C20x", ALC260_ACER),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01006235 SND_PCI_QUIRK(0x1025, 0x008f, "Acer", ALC260_ACER),
Michael Schwingencc959482009-02-22 18:58:45 +01006236 SND_PCI_QUIRK(0x1509, 0x4540, "Favorit 100XS", ALC260_FAVORIT100),
Takashi Iwai9720b712007-03-13 21:46:23 +01006237 SND_PCI_QUIRK(0x103c, 0x2808, "HP d5700", ALC260_HP_3013),
Takashi Iwai4ac55982009-11-10 16:08:45 +01006238 SND_PCI_QUIRK(0x103c, 0x280a, "HP d5750", ALC260_AUTO), /* no quirk */
Takashi Iwaif5fcc132006-11-24 17:07:44 +01006239 SND_PCI_QUIRK(0x103c, 0x3010, "HP", ALC260_HP_3013),
Jaroslav Kysela34ec8a02008-07-10 14:49:19 +02006240 SND_PCI_QUIRK(0x103c, 0x3011, "HP", ALC260_HP_3013),
Kailang Yang3f878302008-08-26 13:02:23 +02006241 SND_PCI_QUIRK(0x103c, 0x3012, "HP", ALC260_HP_DC7600),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01006242 SND_PCI_QUIRK(0x103c, 0x3013, "HP", ALC260_HP_3013),
6243 SND_PCI_QUIRK(0x103c, 0x3014, "HP", ALC260_HP),
6244 SND_PCI_QUIRK(0x103c, 0x3015, "HP", ALC260_HP),
6245 SND_PCI_QUIRK(0x103c, 0x3016, "HP", ALC260_HP),
6246 SND_PCI_QUIRK(0x104d, 0x81bb, "Sony VAIO", ALC260_BASIC),
6247 SND_PCI_QUIRK(0x104d, 0x81cc, "Sony VAIO", ALC260_BASIC),
6248 SND_PCI_QUIRK(0x104d, 0x81cd, "Sony VAIO", ALC260_BASIC),
6249 SND_PCI_QUIRK(0x10cf, 0x1326, "Fujitsu S702X", ALC260_FUJITSU_S702X),
6250 SND_PCI_QUIRK(0x152d, 0x0729, "CTL U553W", ALC260_BASIC),
Kailang Yangbc9f98a2007-04-12 13:06:07 +02006251 SND_PCI_QUIRK(0x161f, 0x2057, "Replacer 672V", ALC260_REPLACER_672V),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01006252 SND_PCI_QUIRK(0x1631, 0xc017, "PB V7900", ALC260_WILL),
Takashi Iwai16ded522005-06-10 19:58:24 +02006253 {}
6254};
6255
Kailang Yangdf694da2005-12-05 19:42:22 +01006256static struct alc_config_preset alc260_presets[] = {
6257 [ALC260_BASIC] = {
6258 .mixers = { alc260_base_output_mixer,
Takashi Iwai45bdd1c2009-02-06 16:11:25 +01006259 alc260_input_mixer },
Kailang Yangdf694da2005-12-05 19:42:22 +01006260 .init_verbs = { alc260_init_verbs },
6261 .num_dacs = ARRAY_SIZE(alc260_dac_nids),
6262 .dac_nids = alc260_dac_nids,
Takashi Iwaif9e336f2008-10-31 16:37:07 +01006263 .num_adc_nids = ARRAY_SIZE(alc260_dual_adc_nids),
Kailang Yangdf694da2005-12-05 19:42:22 +01006264 .adc_nids = alc260_adc_nids,
6265 .num_channel_mode = ARRAY_SIZE(alc260_modes),
6266 .channel_mode = alc260_modes,
6267 .input_mux = &alc260_capture_source,
6268 },
6269 [ALC260_HP] = {
Takashi Iwaibec15c32008-01-28 18:16:30 +01006270 .mixers = { alc260_hp_output_mixer,
Takashi Iwaif9e336f2008-10-31 16:37:07 +01006271 alc260_input_mixer },
Takashi Iwaibec15c32008-01-28 18:16:30 +01006272 .init_verbs = { alc260_init_verbs,
6273 alc260_hp_unsol_verbs },
Kailang Yangdf694da2005-12-05 19:42:22 +01006274 .num_dacs = ARRAY_SIZE(alc260_dac_nids),
6275 .dac_nids = alc260_dac_nids,
Takashi Iwaif9e336f2008-10-31 16:37:07 +01006276 .num_adc_nids = ARRAY_SIZE(alc260_adc_nids_alt),
6277 .adc_nids = alc260_adc_nids_alt,
Kailang Yangdf694da2005-12-05 19:42:22 +01006278 .num_channel_mode = ARRAY_SIZE(alc260_modes),
6279 .channel_mode = alc260_modes,
6280 .input_mux = &alc260_capture_source,
Takashi Iwaibec15c32008-01-28 18:16:30 +01006281 .unsol_event = alc260_hp_unsol_event,
6282 .init_hook = alc260_hp_automute,
Kailang Yangdf694da2005-12-05 19:42:22 +01006283 },
Kailang Yang3f878302008-08-26 13:02:23 +02006284 [ALC260_HP_DC7600] = {
6285 .mixers = { alc260_hp_dc7600_mixer,
Takashi Iwaif9e336f2008-10-31 16:37:07 +01006286 alc260_input_mixer },
Kailang Yang3f878302008-08-26 13:02:23 +02006287 .init_verbs = { alc260_init_verbs,
6288 alc260_hp_dc7600_verbs },
6289 .num_dacs = ARRAY_SIZE(alc260_dac_nids),
6290 .dac_nids = alc260_dac_nids,
Takashi Iwaif9e336f2008-10-31 16:37:07 +01006291 .num_adc_nids = ARRAY_SIZE(alc260_adc_nids_alt),
6292 .adc_nids = alc260_adc_nids_alt,
Kailang Yang3f878302008-08-26 13:02:23 +02006293 .num_channel_mode = ARRAY_SIZE(alc260_modes),
6294 .channel_mode = alc260_modes,
6295 .input_mux = &alc260_capture_source,
6296 .unsol_event = alc260_hp_3012_unsol_event,
6297 .init_hook = alc260_hp_3012_automute,
6298 },
Kailang Yangdf694da2005-12-05 19:42:22 +01006299 [ALC260_HP_3013] = {
6300 .mixers = { alc260_hp_3013_mixer,
Takashi Iwaif9e336f2008-10-31 16:37:07 +01006301 alc260_input_mixer },
Takashi Iwaibec15c32008-01-28 18:16:30 +01006302 .init_verbs = { alc260_hp_3013_init_verbs,
6303 alc260_hp_3013_unsol_verbs },
Kailang Yangdf694da2005-12-05 19:42:22 +01006304 .num_dacs = ARRAY_SIZE(alc260_dac_nids),
6305 .dac_nids = alc260_dac_nids,
Takashi Iwaif9e336f2008-10-31 16:37:07 +01006306 .num_adc_nids = ARRAY_SIZE(alc260_adc_nids_alt),
6307 .adc_nids = alc260_adc_nids_alt,
Kailang Yangdf694da2005-12-05 19:42:22 +01006308 .num_channel_mode = ARRAY_SIZE(alc260_modes),
6309 .channel_mode = alc260_modes,
6310 .input_mux = &alc260_capture_source,
Takashi Iwaibec15c32008-01-28 18:16:30 +01006311 .unsol_event = alc260_hp_3013_unsol_event,
6312 .init_hook = alc260_hp_3013_automute,
Kailang Yangdf694da2005-12-05 19:42:22 +01006313 },
6314 [ALC260_FUJITSU_S702X] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +01006315 .mixers = { alc260_fujitsu_mixer },
Kailang Yangdf694da2005-12-05 19:42:22 +01006316 .init_verbs = { alc260_fujitsu_init_verbs },
6317 .num_dacs = ARRAY_SIZE(alc260_dac_nids),
6318 .dac_nids = alc260_dac_nids,
Jonathan Woithed57fdac2006-02-28 11:38:35 +01006319 .num_adc_nids = ARRAY_SIZE(alc260_dual_adc_nids),
6320 .adc_nids = alc260_dual_adc_nids,
Kailang Yangdf694da2005-12-05 19:42:22 +01006321 .num_channel_mode = ARRAY_SIZE(alc260_modes),
6322 .channel_mode = alc260_modes,
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02006323 .num_mux_defs = ARRAY_SIZE(alc260_fujitsu_capture_sources),
6324 .input_mux = alc260_fujitsu_capture_sources,
Kailang Yangdf694da2005-12-05 19:42:22 +01006325 },
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01006326 [ALC260_ACER] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +01006327 .mixers = { alc260_acer_mixer },
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01006328 .init_verbs = { alc260_acer_init_verbs },
6329 .num_dacs = ARRAY_SIZE(alc260_dac_nids),
6330 .dac_nids = alc260_dac_nids,
6331 .num_adc_nids = ARRAY_SIZE(alc260_dual_adc_nids),
6332 .adc_nids = alc260_dual_adc_nids,
6333 .num_channel_mode = ARRAY_SIZE(alc260_modes),
6334 .channel_mode = alc260_modes,
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02006335 .num_mux_defs = ARRAY_SIZE(alc260_acer_capture_sources),
6336 .input_mux = alc260_acer_capture_sources,
Jonathan Woithe0bfc90e2006-02-28 11:45:11 +01006337 },
Michael Schwingencc959482009-02-22 18:58:45 +01006338 [ALC260_FAVORIT100] = {
6339 .mixers = { alc260_favorit100_mixer },
6340 .init_verbs = { alc260_favorit100_init_verbs },
6341 .num_dacs = ARRAY_SIZE(alc260_dac_nids),
6342 .dac_nids = alc260_dac_nids,
6343 .num_adc_nids = ARRAY_SIZE(alc260_dual_adc_nids),
6344 .adc_nids = alc260_dual_adc_nids,
6345 .num_channel_mode = ARRAY_SIZE(alc260_modes),
6346 .channel_mode = alc260_modes,
6347 .num_mux_defs = ARRAY_SIZE(alc260_favorit100_capture_sources),
6348 .input_mux = alc260_favorit100_capture_sources,
6349 },
Kailang Yangbc9f98a2007-04-12 13:06:07 +02006350 [ALC260_WILL] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +01006351 .mixers = { alc260_will_mixer },
Kailang Yangbc9f98a2007-04-12 13:06:07 +02006352 .init_verbs = { alc260_init_verbs, alc260_will_verbs },
6353 .num_dacs = ARRAY_SIZE(alc260_dac_nids),
6354 .dac_nids = alc260_dac_nids,
6355 .num_adc_nids = ARRAY_SIZE(alc260_adc_nids),
6356 .adc_nids = alc260_adc_nids,
6357 .dig_out_nid = ALC260_DIGOUT_NID,
6358 .num_channel_mode = ARRAY_SIZE(alc260_modes),
6359 .channel_mode = alc260_modes,
6360 .input_mux = &alc260_capture_source,
6361 },
6362 [ALC260_REPLACER_672V] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +01006363 .mixers = { alc260_replacer_672v_mixer },
Kailang Yangbc9f98a2007-04-12 13:06:07 +02006364 .init_verbs = { alc260_init_verbs, alc260_replacer_672v_verbs },
6365 .num_dacs = ARRAY_SIZE(alc260_dac_nids),
6366 .dac_nids = alc260_dac_nids,
6367 .num_adc_nids = ARRAY_SIZE(alc260_adc_nids),
6368 .adc_nids = alc260_adc_nids,
6369 .dig_out_nid = ALC260_DIGOUT_NID,
6370 .num_channel_mode = ARRAY_SIZE(alc260_modes),
6371 .channel_mode = alc260_modes,
6372 .input_mux = &alc260_capture_source,
6373 .unsol_event = alc260_replacer_672v_unsol_event,
6374 .init_hook = alc260_replacer_672v_automute,
6375 },
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01006376#ifdef CONFIG_SND_DEBUG
6377 [ALC260_TEST] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +01006378 .mixers = { alc260_test_mixer },
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01006379 .init_verbs = { alc260_test_init_verbs },
6380 .num_dacs = ARRAY_SIZE(alc260_test_dac_nids),
6381 .dac_nids = alc260_test_dac_nids,
6382 .num_adc_nids = ARRAY_SIZE(alc260_test_adc_nids),
6383 .adc_nids = alc260_test_adc_nids,
6384 .num_channel_mode = ARRAY_SIZE(alc260_modes),
6385 .channel_mode = alc260_modes,
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +02006386 .num_mux_defs = ARRAY_SIZE(alc260_test_capture_sources),
6387 .input_mux = alc260_test_capture_sources,
Jonathan Woithe7cf51e42006-02-09 12:01:26 +01006388 },
6389#endif
Kailang Yangdf694da2005-12-05 19:42:22 +01006390};
6391
Linus Torvalds1da177e2005-04-16 15:20:36 -07006392static int patch_alc260(struct hda_codec *codec)
6393{
6394 struct alc_spec *spec;
Kailang Yangdf694da2005-12-05 19:42:22 +01006395 int err, board_config;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006396
Takashi Iwaie560d8d2005-09-09 14:21:46 +02006397 spec = kzalloc(sizeof(*spec), GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006398 if (spec == NULL)
6399 return -ENOMEM;
6400
6401 codec->spec = spec;
6402
Takashi Iwaif5fcc132006-11-24 17:07:44 +01006403 board_config = snd_hda_check_board_config(codec, ALC260_MODEL_LAST,
6404 alc260_models,
6405 alc260_cfg_tbl);
6406 if (board_config < 0) {
Takashi Iwai9a11f1a2009-07-28 16:01:20 +02006407 snd_printd(KERN_INFO "hda_codec: %s: BIOS auto-probing.\n",
Takashi Iwai6c627f32009-05-18 12:33:36 +02006408 codec->chip_name);
Kailang Yangdf694da2005-12-05 19:42:22 +01006409 board_config = ALC260_AUTO;
Takashi Iwai16ded522005-06-10 19:58:24 +02006410 }
6411
Kailang Yangdf694da2005-12-05 19:42:22 +01006412 if (board_config == ALC260_AUTO) {
6413 /* automatic parse from the BIOS config */
6414 err = alc260_parse_auto_config(codec);
6415 if (err < 0) {
6416 alc_free(codec);
6417 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02006418 } else if (!err) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +02006419 printk(KERN_INFO
6420 "hda_codec: Cannot set up configuration "
6421 "from BIOS. Using base mode...\n");
Kailang Yangdf694da2005-12-05 19:42:22 +01006422 board_config = ALC260_BASIC;
6423 }
Takashi Iwai16ded522005-06-10 19:58:24 +02006424 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07006425
Kusanagi Kouichi680cd532009-02-05 00:00:58 +09006426 err = snd_hda_attach_beep_device(codec, 0x1);
6427 if (err < 0) {
6428 alc_free(codec);
6429 return err;
6430 }
6431
Kailang Yangdf694da2005-12-05 19:42:22 +01006432 if (board_config != ALC260_AUTO)
Takashi Iwaie9c364c2009-08-11 17:16:13 +02006433 setup_preset(codec, &alc260_presets[board_config]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006434
Linus Torvalds1da177e2005-04-16 15:20:36 -07006435 spec->stream_analog_playback = &alc260_pcm_analog_playback;
6436 spec->stream_analog_capture = &alc260_pcm_analog_capture;
6437
Takashi Iwaia3bcba32005-12-06 19:05:29 +01006438 spec->stream_digital_playback = &alc260_pcm_digital_playback;
6439 spec->stream_digital_capture = &alc260_pcm_digital_capture;
6440
Takashi Iwai4ef0ef12008-11-03 17:47:49 +01006441 if (!spec->adc_nids && spec->input_mux) {
6442 /* check whether NID 0x04 is valid */
6443 unsigned int wcap = get_wcaps(codec, 0x04);
Takashi Iwaia22d5432009-07-27 12:54:26 +02006444 wcap = get_wcaps_type(wcap);
Takashi Iwai4ef0ef12008-11-03 17:47:49 +01006445 /* get type */
6446 if (wcap != AC_WID_AUD_IN || spec->input_mux->num_items == 1) {
6447 spec->adc_nids = alc260_adc_nids_alt;
6448 spec->num_adc_nids = ARRAY_SIZE(alc260_adc_nids_alt);
6449 } else {
6450 spec->adc_nids = alc260_adc_nids;
6451 spec->num_adc_nids = ARRAY_SIZE(alc260_adc_nids);
6452 }
6453 }
Takashi Iwaib59bdf32009-08-11 09:47:30 +02006454 set_capture_mixer(codec);
Takashi Iwai45bdd1c2009-02-06 16:11:25 +01006455 set_beep_amp(spec, 0x07, 0x05, HDA_INPUT);
Takashi Iwaif9e336f2008-10-31 16:37:07 +01006456
Takashi Iwai2134ea42008-01-10 16:53:55 +01006457 spec->vmaster_nid = 0x08;
6458
Linus Torvalds1da177e2005-04-16 15:20:36 -07006459 codec->patch_ops = alc_patch_ops;
Kailang Yangdf694da2005-12-05 19:42:22 +01006460 if (board_config == ALC260_AUTO)
Takashi Iwaiae6b8132006-03-03 16:47:17 +01006461 spec->init_hook = alc260_auto_init;
Takashi Iwaicb53c622007-08-10 17:21:45 +02006462#ifdef CONFIG_SND_HDA_POWER_SAVE
6463 if (!spec->loopback.amplist)
6464 spec->loopback.amplist = alc260_loopbacks;
6465#endif
Takashi Iwaidaead532008-11-28 12:55:36 +01006466 codec->proc_widget_hook = print_realtek_coef;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006467
6468 return 0;
6469}
6470
Takashi Iwaie9edcee2005-06-13 14:16:38 +02006471
Linus Torvalds1da177e2005-04-16 15:20:36 -07006472/*
Takashi Iwai49535502009-06-30 15:28:30 +02006473 * ALC882/883/885/888/889 support
Linus Torvalds1da177e2005-04-16 15:20:36 -07006474 *
6475 * ALC882 is almost identical with ALC880 but has cleaner and more flexible
6476 * configuration. Each pin widget can choose any input DACs and a mixer.
6477 * Each ADC is connected from a mixer of all inputs. This makes possible
6478 * 6-channel independent captures.
6479 *
6480 * In addition, an independent DAC for the multi-playback (not used in this
6481 * driver yet).
6482 */
Kailang Yangdf694da2005-12-05 19:42:22 +01006483#define ALC882_DIGOUT_NID 0x06
6484#define ALC882_DIGIN_NID 0x0a
Takashi Iwai49535502009-06-30 15:28:30 +02006485#define ALC883_DIGOUT_NID ALC882_DIGOUT_NID
6486#define ALC883_DIGIN_NID ALC882_DIGIN_NID
6487#define ALC1200_DIGOUT_NID 0x10
6488
Linus Torvalds1da177e2005-04-16 15:20:36 -07006489
Takashi Iwaid2a6d7d2005-11-17 11:06:29 +01006490static struct hda_channel_mode alc882_ch_modes[1] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07006491 { 8, NULL }
6492};
6493
Takashi Iwai49535502009-06-30 15:28:30 +02006494/* DACs */
Linus Torvalds1da177e2005-04-16 15:20:36 -07006495static hda_nid_t alc882_dac_nids[4] = {
6496 /* front, rear, clfe, rear_surr */
6497 0x02, 0x03, 0x04, 0x05
6498};
Takashi Iwai49535502009-06-30 15:28:30 +02006499#define alc883_dac_nids alc882_dac_nids
Linus Torvalds1da177e2005-04-16 15:20:36 -07006500
Takashi Iwai49535502009-06-30 15:28:30 +02006501/* ADCs */
Kailang Yangdf694da2005-12-05 19:42:22 +01006502#define alc882_adc_nids alc880_adc_nids
6503#define alc882_adc_nids_alt alc880_adc_nids_alt
Takashi Iwai49535502009-06-30 15:28:30 +02006504#define alc883_adc_nids alc882_adc_nids_alt
6505static hda_nid_t alc883_adc_nids_alt[1] = { 0x08 };
6506static hda_nid_t alc883_adc_nids_rev[2] = { 0x09, 0x08 };
6507#define alc889_adc_nids alc880_adc_nids
Linus Torvalds1da177e2005-04-16 15:20:36 -07006508
Takashi Iwaie1406342008-02-11 18:32:32 +01006509static hda_nid_t alc882_capsrc_nids[3] = { 0x24, 0x23, 0x22 };
6510static hda_nid_t alc882_capsrc_nids_alt[2] = { 0x23, 0x22 };
Takashi Iwai49535502009-06-30 15:28:30 +02006511#define alc883_capsrc_nids alc882_capsrc_nids_alt
6512static hda_nid_t alc883_capsrc_nids_rev[2] = { 0x22, 0x23 };
6513#define alc889_capsrc_nids alc882_capsrc_nids
Takashi Iwaie1406342008-02-11 18:32:32 +01006514
Linus Torvalds1da177e2005-04-16 15:20:36 -07006515/* input MUX */
6516/* FIXME: should be a matrix-type input source selection */
6517
6518static struct hda_input_mux alc882_capture_source = {
6519 .num_items = 4,
6520 .items = {
6521 { "Mic", 0x0 },
6522 { "Front Mic", 0x1 },
6523 { "Line", 0x2 },
6524 { "CD", 0x4 },
6525 },
6526};
Kacper Szczesniak41d55452009-05-07 12:47:43 +02006527
Takashi Iwai49535502009-06-30 15:28:30 +02006528#define alc883_capture_source alc882_capture_source
6529
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02006530static struct hda_input_mux alc889_capture_source = {
6531 .num_items = 3,
6532 .items = {
6533 { "Front Mic", 0x0 },
6534 { "Mic", 0x3 },
6535 { "Line", 0x2 },
6536 },
6537};
6538
Kacper Szczesniak41d55452009-05-07 12:47:43 +02006539static struct hda_input_mux mb5_capture_source = {
6540 .num_items = 3,
6541 .items = {
6542 { "Mic", 0x1 },
6543 { "Line", 0x2 },
6544 { "CD", 0x4 },
6545 },
6546};
6547
Takashi Iwai49535502009-06-30 15:28:30 +02006548static struct hda_input_mux alc883_3stack_6ch_intel = {
6549 .num_items = 4,
6550 .items = {
6551 { "Mic", 0x1 },
6552 { "Front Mic", 0x0 },
6553 { "Line", 0x2 },
6554 { "CD", 0x4 },
6555 },
6556};
6557
6558static struct hda_input_mux alc883_lenovo_101e_capture_source = {
6559 .num_items = 2,
6560 .items = {
6561 { "Mic", 0x1 },
6562 { "Line", 0x2 },
6563 },
6564};
6565
6566static struct hda_input_mux alc883_lenovo_nb0763_capture_source = {
6567 .num_items = 4,
6568 .items = {
6569 { "Mic", 0x0 },
6570 { "iMic", 0x1 },
6571 { "Line", 0x2 },
6572 { "CD", 0x4 },
6573 },
6574};
6575
6576static struct hda_input_mux alc883_fujitsu_pi2515_capture_source = {
6577 .num_items = 2,
6578 .items = {
6579 { "Mic", 0x0 },
6580 { "Int Mic", 0x1 },
6581 },
6582};
6583
6584static struct hda_input_mux alc883_lenovo_sky_capture_source = {
6585 .num_items = 3,
6586 .items = {
6587 { "Mic", 0x0 },
6588 { "Front Mic", 0x1 },
6589 { "Line", 0x4 },
6590 },
6591};
6592
6593static struct hda_input_mux alc883_asus_eee1601_capture_source = {
6594 .num_items = 2,
6595 .items = {
6596 { "Mic", 0x0 },
6597 { "Line", 0x2 },
6598 },
6599};
6600
6601static struct hda_input_mux alc889A_mb31_capture_source = {
6602 .num_items = 2,
6603 .items = {
6604 { "Mic", 0x0 },
6605 /* Front Mic (0x01) unused */
6606 { "Line", 0x2 },
6607 /* Line 2 (0x03) unused */
6608 /* CD (0x04) unsused? */
6609 },
6610};
6611
6612/*
6613 * 2ch mode
6614 */
6615static struct hda_channel_mode alc883_3ST_2ch_modes[1] = {
6616 { 2, NULL }
6617};
6618
Kailang Yangdf694da2005-12-05 19:42:22 +01006619/*
Kailang Yang272a5272007-05-14 11:00:38 +02006620 * 2ch mode
6621 */
6622static struct hda_verb alc882_3ST_ch2_init[] = {
6623 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
6624 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6625 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
6626 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6627 { } /* end */
6628};
6629
6630/*
Takashi Iwai49535502009-06-30 15:28:30 +02006631 * 4ch mode
6632 */
6633static struct hda_verb alc882_3ST_ch4_init[] = {
6634 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
6635 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6636 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6637 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6638 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x01 },
6639 { } /* end */
6640};
6641
6642/*
Kailang Yang272a5272007-05-14 11:00:38 +02006643 * 6ch mode
6644 */
6645static struct hda_verb alc882_3ST_ch6_init[] = {
6646 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6647 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6648 { 0x18, AC_VERB_SET_CONNECT_SEL, 0x02 },
6649 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6650 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6651 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x01 },
6652 { } /* end */
6653};
6654
Takashi Iwai49535502009-06-30 15:28:30 +02006655static struct hda_channel_mode alc882_3ST_6ch_modes[3] = {
Kailang Yang272a5272007-05-14 11:00:38 +02006656 { 2, alc882_3ST_ch2_init },
Takashi Iwai49535502009-06-30 15:28:30 +02006657 { 4, alc882_3ST_ch4_init },
Kailang Yang272a5272007-05-14 11:00:38 +02006658 { 6, alc882_3ST_ch6_init },
6659};
6660
Takashi Iwai49535502009-06-30 15:28:30 +02006661#define alc883_3ST_6ch_modes alc882_3ST_6ch_modes
6662
Kailang Yang272a5272007-05-14 11:00:38 +02006663/*
ddiaz@cenditel.gob.vea65cc602009-09-05 16:28:06 -04306664 * 2ch mode
6665 */
6666static struct hda_verb alc883_3ST_ch2_clevo_init[] = {
6667 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP },
6668 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
6669 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6670 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
6671 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6672 { } /* end */
6673};
6674
6675/*
6676 * 4ch mode
6677 */
6678static struct hda_verb alc883_3ST_ch4_clevo_init[] = {
6679 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6680 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
6681 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6682 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6683 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6684 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x01 },
6685 { } /* end */
6686};
6687
6688/*
6689 * 6ch mode
6690 */
6691static struct hda_verb alc883_3ST_ch6_clevo_init[] = {
6692 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6693 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6694 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6695 { 0x18, AC_VERB_SET_CONNECT_SEL, 0x02 },
6696 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6697 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6698 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x01 },
6699 { } /* end */
6700};
6701
6702static struct hda_channel_mode alc883_3ST_6ch_clevo_modes[3] = {
6703 { 2, alc883_3ST_ch2_clevo_init },
6704 { 4, alc883_3ST_ch4_clevo_init },
6705 { 6, alc883_3ST_ch6_clevo_init },
6706};
6707
6708
6709/*
Kailang Yangdf694da2005-12-05 19:42:22 +01006710 * 6ch mode
6711 */
6712static struct hda_verb alc882_sixstack_ch6_init[] = {
6713 { 0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
6714 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6715 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6716 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6717 { } /* end */
6718};
6719
6720/*
6721 * 8ch mode
6722 */
6723static struct hda_verb alc882_sixstack_ch8_init[] = {
6724 { 0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6725 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6726 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6727 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6728 { } /* end */
6729};
6730
6731static struct hda_channel_mode alc882_sixstack_modes[2] = {
6732 { 6, alc882_sixstack_ch6_init },
6733 { 8, alc882_sixstack_ch8_init },
6734};
6735
Takashi Iwai87350ad2007-08-16 18:19:38 +02006736/*
Sasha Alexandrdef319f2009-06-16 16:00:15 -04006737 * macbook pro ALC885 can switch LineIn to LineOut without losing Mic
Takashi Iwai87350ad2007-08-16 18:19:38 +02006738 */
6739
6740/*
6741 * 2ch mode
6742 */
6743static struct hda_verb alc885_mbp_ch2_init[] = {
6744 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
6745 { 0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
6746 { 0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
6747 { } /* end */
6748};
6749
6750/*
Takashi Iwaia3f730a2009-08-31 08:15:26 +02006751 * 4ch mode
Takashi Iwai87350ad2007-08-16 18:19:38 +02006752 */
Takashi Iwaia3f730a2009-08-31 08:15:26 +02006753static struct hda_verb alc885_mbp_ch4_init[] = {
Takashi Iwai87350ad2007-08-16 18:19:38 +02006754 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6755 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
6756 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x01 },
6757 { 0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
6758 { 0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
6759 { } /* end */
6760};
6761
Takashi Iwaia3f730a2009-08-31 08:15:26 +02006762static struct hda_channel_mode alc885_mbp_4ch_modes[2] = {
Takashi Iwai87350ad2007-08-16 18:19:38 +02006763 { 2, alc885_mbp_ch2_init },
Takashi Iwaia3f730a2009-08-31 08:15:26 +02006764 { 4, alc885_mbp_ch4_init },
Takashi Iwai87350ad2007-08-16 18:19:38 +02006765};
6766
Kacper Szczesniak92b9de82009-06-02 00:55:19 +02006767/*
6768 * 2ch
6769 * Speakers/Woofer/HP = Front
6770 * LineIn = Input
6771 */
6772static struct hda_verb alc885_mb5_ch2_init[] = {
6773 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
6774 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
6775 { } /* end */
6776};
6777
6778/*
6779 * 6ch mode
6780 * Speakers/HP = Front
6781 * Woofer = LFE
6782 * LineIn = Surround
6783 */
6784static struct hda_verb alc885_mb5_ch6_init[] = {
6785 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
6786 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
6787 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01},
6788 { } /* end */
6789};
6790
6791static struct hda_channel_mode alc885_mb5_6ch_modes[2] = {
6792 { 2, alc885_mb5_ch2_init },
6793 { 6, alc885_mb5_ch6_init },
6794};
Takashi Iwai87350ad2007-08-16 18:19:38 +02006795
Takashi Iwai49535502009-06-30 15:28:30 +02006796
6797/*
6798 * 2ch mode
6799 */
6800static struct hda_verb alc883_4ST_ch2_init[] = {
6801 { 0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6802 { 0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6803 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
6804 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6805 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
6806 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6807 { } /* end */
6808};
6809
6810/*
6811 * 4ch mode
6812 */
6813static struct hda_verb alc883_4ST_ch4_init[] = {
6814 { 0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6815 { 0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6816 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
6817 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6818 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6819 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6820 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x01 },
6821 { } /* end */
6822};
6823
6824/*
6825 * 6ch mode
6826 */
6827static struct hda_verb alc883_4ST_ch6_init[] = {
6828 { 0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6829 { 0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6830 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6831 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6832 { 0x18, AC_VERB_SET_CONNECT_SEL, 0x02 },
6833 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6834 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6835 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x01 },
6836 { } /* end */
6837};
6838
6839/*
6840 * 8ch mode
6841 */
6842static struct hda_verb alc883_4ST_ch8_init[] = {
6843 { 0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6844 { 0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6845 { 0x17, AC_VERB_SET_CONNECT_SEL, 0x03 },
6846 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6847 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6848 { 0x18, AC_VERB_SET_CONNECT_SEL, 0x02 },
6849 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6850 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6851 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x01 },
6852 { } /* end */
6853};
6854
6855static struct hda_channel_mode alc883_4ST_8ch_modes[4] = {
6856 { 2, alc883_4ST_ch2_init },
6857 { 4, alc883_4ST_ch4_init },
6858 { 6, alc883_4ST_ch6_init },
6859 { 8, alc883_4ST_ch8_init },
6860};
6861
6862
6863/*
6864 * 2ch mode
6865 */
6866static struct hda_verb alc883_3ST_ch2_intel_init[] = {
6867 { 0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
6868 { 0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6869 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
6870 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6871 { } /* end */
6872};
6873
6874/*
6875 * 4ch mode
6876 */
6877static struct hda_verb alc883_3ST_ch4_intel_init[] = {
6878 { 0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
6879 { 0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6880 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6881 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6882 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x01 },
6883 { } /* end */
6884};
6885
6886/*
6887 * 6ch mode
6888 */
6889static struct hda_verb alc883_3ST_ch6_intel_init[] = {
6890 { 0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6891 { 0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6892 { 0x19, AC_VERB_SET_CONNECT_SEL, 0x02 },
6893 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6894 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
6895 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x01 },
6896 { } /* end */
6897};
6898
6899static struct hda_channel_mode alc883_3ST_6ch_intel_modes[3] = {
6900 { 2, alc883_3ST_ch2_intel_init },
6901 { 4, alc883_3ST_ch4_intel_init },
6902 { 6, alc883_3ST_ch6_intel_init },
6903};
6904
6905/*
Wu Fengguangdd7714c2009-07-30 14:36:35 +08006906 * 2ch mode
6907 */
6908static struct hda_verb alc889_ch2_intel_init[] = {
6909 { 0x14, AC_VERB_SET_CONNECT_SEL, 0x00 },
6910 { 0x19, AC_VERB_SET_CONNECT_SEL, 0x00 },
6911 { 0x16, AC_VERB_SET_CONNECT_SEL, 0x00 },
6912 { 0x17, AC_VERB_SET_CONNECT_SEL, 0x00 },
6913 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
6914 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6915 { } /* end */
6916};
6917
6918/*
Takashi Iwai49535502009-06-30 15:28:30 +02006919 * 6ch mode
6920 */
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02006921static struct hda_verb alc889_ch6_intel_init[] = {
Wu Fengguangdd7714c2009-07-30 14:36:35 +08006922 { 0x14, AC_VERB_SET_CONNECT_SEL, 0x00 },
6923 { 0x19, AC_VERB_SET_CONNECT_SEL, 0x01 },
6924 { 0x16, AC_VERB_SET_CONNECT_SEL, 0x02 },
6925 { 0x17, AC_VERB_SET_CONNECT_SEL, 0x03 },
6926 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02006927 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
6928 { } /* end */
6929};
6930
6931/*
6932 * 8ch mode
6933 */
6934static struct hda_verb alc889_ch8_intel_init[] = {
Wu Fengguangdd7714c2009-07-30 14:36:35 +08006935 { 0x14, AC_VERB_SET_CONNECT_SEL, 0x00 },
6936 { 0x19, AC_VERB_SET_CONNECT_SEL, 0x01 },
6937 { 0x16, AC_VERB_SET_CONNECT_SEL, 0x02 },
6938 { 0x17, AC_VERB_SET_CONNECT_SEL, 0x03 },
6939 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x03 },
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02006940 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6941 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02006942 { } /* end */
6943};
6944
Wu Fengguangdd7714c2009-07-30 14:36:35 +08006945static struct hda_channel_mode alc889_8ch_intel_modes[3] = {
6946 { 2, alc889_ch2_intel_init },
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02006947 { 6, alc889_ch6_intel_init },
6948 { 8, alc889_ch8_intel_init },
6949};
6950
6951/*
6952 * 6ch mode
6953 */
Takashi Iwai49535502009-06-30 15:28:30 +02006954static struct hda_verb alc883_sixstack_ch6_init[] = {
6955 { 0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
6956 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6957 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6958 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6959 { } /* end */
6960};
6961
6962/*
6963 * 8ch mode
6964 */
6965static struct hda_verb alc883_sixstack_ch8_init[] = {
6966 { 0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6967 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6968 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6969 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
6970 { } /* end */
6971};
6972
6973static struct hda_channel_mode alc883_sixstack_modes[2] = {
6974 { 6, alc883_sixstack_ch6_init },
6975 { 8, alc883_sixstack_ch8_init },
6976};
6977
6978
Linus Torvalds1da177e2005-04-16 15:20:36 -07006979/* Pin assignment: Front=0x14, Rear=0x15, CLFE=0x16, Side=0x17
6980 * Mic=0x18, Front Mic=0x19, Line-In=0x1a, HP=0x1b
6981 */
Takashi Iwaic8b6bf9b2005-11-17 14:57:47 +01006982static struct snd_kcontrol_new alc882_base_mixer[] = {
Takashi Iwai05acb862005-06-10 19:50:25 +02006983 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01006984 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Takashi Iwai05acb862005-06-10 19:50:25 +02006985 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01006986 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
Takashi Iwai05acb862005-06-10 19:50:25 +02006987 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
6988 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01006989 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
6990 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
Takashi Iwai05acb862005-06-10 19:50:25 +02006991 HDA_CODEC_VOLUME("Side Playback Volume", 0x0f, 0x0, HDA_OUTPUT),
Takashi Iwai985be542005-11-02 18:26:49 +01006992 HDA_BIND_MUTE("Side Playback Switch", 0x0f, 2, HDA_INPUT),
Linus Torvalds1da177e2005-04-16 15:20:36 -07006993 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
6994 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
6995 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
6996 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
6997 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
6998 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01006999 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Linus Torvalds1da177e2005-04-16 15:20:36 -07007000 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
7001 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01007002 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
Linus Torvalds1da177e2005-04-16 15:20:36 -07007003 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Linus Torvalds1da177e2005-04-16 15:20:36 -07007004 { } /* end */
7005};
7006
Takashi Iwai87350ad2007-08-16 18:19:38 +02007007static struct snd_kcontrol_new alc885_mbp3_mixer[] = {
Takashi Iwaia3f730a2009-08-31 08:15:26 +02007008 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0c, 0x00, HDA_OUTPUT),
7009 HDA_BIND_MUTE ("Speaker Playback Switch", 0x0c, 0x02, HDA_INPUT),
7010 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0e, 0x00, HDA_OUTPUT),
7011 HDA_BIND_MUTE ("Headphone Playback Switch", 0x0e, 0x02, HDA_INPUT),
7012 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x00, HDA_OUTPUT),
Takashi Iwai2134ea42008-01-10 16:53:55 +01007013 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
7014 HDA_CODEC_MUTE ("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
Takashi Iwai87350ad2007-08-16 18:19:38 +02007015 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x00, HDA_INPUT),
7016 HDA_CODEC_MUTE ("Mic Playback Switch", 0x0b, 0x00, HDA_INPUT),
Takashi Iwai2134ea42008-01-10 16:53:55 +01007017 HDA_CODEC_VOLUME("Line Boost", 0x1a, 0x00, HDA_INPUT),
Takashi Iwai87350ad2007-08-16 18:19:38 +02007018 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0x00, HDA_INPUT),
7019 { } /* end */
7020};
Kacper Szczesniak41d55452009-05-07 12:47:43 +02007021
7022static struct snd_kcontrol_new alc885_mb5_mixer[] = {
Kacper Szczesniak92b9de82009-06-02 00:55:19 +02007023 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x00, HDA_OUTPUT),
7024 HDA_BIND_MUTE ("Front Playback Switch", 0x0c, 0x02, HDA_INPUT),
7025 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x00, HDA_OUTPUT),
7026 HDA_BIND_MUTE ("Surround Playback Switch", 0x0d, 0x02, HDA_INPUT),
7027 HDA_CODEC_VOLUME("LFE Playback Volume", 0x0e, 0x00, HDA_OUTPUT),
7028 HDA_BIND_MUTE ("LFE Playback Switch", 0x0e, 0x02, HDA_INPUT),
7029 HDA_CODEC_VOLUME("HP Playback Volume", 0x0f, 0x00, HDA_OUTPUT),
7030 HDA_BIND_MUTE ("HP Playback Switch", 0x0f, 0x02, HDA_INPUT),
Kacper Szczesniak41d55452009-05-07 12:47:43 +02007031 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
7032 HDA_CODEC_MUTE ("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
7033 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
7034 HDA_CODEC_MUTE ("Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
7035 HDA_CODEC_VOLUME("Line Boost", 0x15, 0x00, HDA_INPUT),
7036 HDA_CODEC_VOLUME("Mic Boost", 0x19, 0x00, HDA_INPUT),
7037 { } /* end */
7038};
Kacper Szczesniak92b9de82009-06-02 00:55:19 +02007039
Kailang Yangbdd148a2007-05-08 15:19:08 +02007040static struct snd_kcontrol_new alc882_w2jc_mixer[] = {
7041 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7042 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
7043 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
7044 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
7045 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
7046 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
7047 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
7048 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
7049 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Kailang Yangbdd148a2007-05-08 15:19:08 +02007050 { } /* end */
7051};
7052
Kailang Yang272a5272007-05-14 11:00:38 +02007053static struct snd_kcontrol_new alc882_targa_mixer[] = {
7054 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7055 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
7056 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
7057 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
7058 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
7059 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
7060 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
7061 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
7062 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Takashi Iwai96fe7cc2007-09-07 10:57:44 +02007063 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Kailang Yang272a5272007-05-14 11:00:38 +02007064 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
7065 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Takashi Iwai96fe7cc2007-09-07 10:57:44 +02007066 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
Kailang Yang272a5272007-05-14 11:00:38 +02007067 { } /* end */
7068};
7069
7070/* Pin assignment: Front=0x14, HP = 0x15, Front = 0x16, ???
7071 * Front Mic=0x18, Line In = 0x1a, Line In = 0x1b, CD = 0x1c
7072 */
7073static struct snd_kcontrol_new alc882_asus_a7j_mixer[] = {
7074 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7075 HDA_CODEC_MUTE("Front Playback Switch", 0x14, 0x0, HDA_OUTPUT),
7076 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
7077 HDA_CODEC_MUTE("Mobile Front Playback Switch", 0x16, 0x0, HDA_OUTPUT),
7078 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
7079 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
7080 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
7081 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
7082 HDA_CODEC_VOLUME("Mobile Line Playback Volume", 0x0b, 0x03, HDA_INPUT),
7083 HDA_CODEC_MUTE("Mobile Line Playback Switch", 0x0b, 0x03, HDA_INPUT),
7084 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
7085 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Takashi Iwai96fe7cc2007-09-07 10:57:44 +02007086 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Kailang Yang272a5272007-05-14 11:00:38 +02007087 { } /* end */
7088};
7089
Takashi Iwai914759b2007-09-06 14:52:04 +02007090static struct snd_kcontrol_new alc882_asus_a7m_mixer[] = {
7091 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7092 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
7093 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
7094 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
7095 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
7096 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
7097 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
7098 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
7099 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
7100 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Takashi Iwai914759b2007-09-06 14:52:04 +02007101 { } /* end */
7102};
7103
Kailang Yangdf694da2005-12-05 19:42:22 +01007104static struct snd_kcontrol_new alc882_chmode_mixer[] = {
7105 {
7106 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
7107 .name = "Channel Mode",
7108 .info = alc_ch_mode_info,
7109 .get = alc_ch_mode_get,
7110 .put = alc_ch_mode_put,
7111 },
7112 { } /* end */
7113};
7114
Takashi Iwai49535502009-06-30 15:28:30 +02007115static struct hda_verb alc882_base_init_verbs[] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07007116 /* Front mixer: unmute input/output amp left and right (volume = 0) */
Takashi Iwai05acb862005-06-10 19:50:25 +02007117 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7118 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7119 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07007120 /* Rear mixer */
Takashi Iwai05acb862005-06-10 19:50:25 +02007121 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7122 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7123 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07007124 /* CLFE mixer */
Takashi Iwai05acb862005-06-10 19:50:25 +02007125 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7126 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7127 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07007128 /* Side mixer */
Takashi Iwai05acb862005-06-10 19:50:25 +02007129 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7130 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7131 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07007132
Takashi Iwaicb638122009-07-03 10:56:10 +02007133 /* mute analog input loopbacks */
7134 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7135 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
7136 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
7137 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
7138 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
7139
Takashi Iwaie9edcee2005-06-13 14:16:38 +02007140 /* Front Pin: output 0 (0x0c) */
Takashi Iwai05acb862005-06-10 19:50:25 +02007141 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwai05acb862005-06-10 19:50:25 +02007142 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Linus Torvalds1da177e2005-04-16 15:20:36 -07007143 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02007144 /* Rear Pin: output 1 (0x0d) */
Takashi Iwai05acb862005-06-10 19:50:25 +02007145 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwai05acb862005-06-10 19:50:25 +02007146 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Linus Torvalds1da177e2005-04-16 15:20:36 -07007147 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02007148 /* CLFE Pin: output 2 (0x0e) */
Takashi Iwai05acb862005-06-10 19:50:25 +02007149 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwai05acb862005-06-10 19:50:25 +02007150 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Linus Torvalds1da177e2005-04-16 15:20:36 -07007151 {0x16, AC_VERB_SET_CONNECT_SEL, 0x02},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02007152 /* Side Pin: output 3 (0x0f) */
Takashi Iwai05acb862005-06-10 19:50:25 +02007153 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwai05acb862005-06-10 19:50:25 +02007154 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Linus Torvalds1da177e2005-04-16 15:20:36 -07007155 {0x17, AC_VERB_SET_CONNECT_SEL, 0x03},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02007156 /* Mic (rear) pin: input vref at 80% */
Takashi Iwai16ded522005-06-10 19:58:24 +02007157 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02007158 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
7159 /* Front Mic pin: input vref at 80% */
Takashi Iwai16ded522005-06-10 19:58:24 +02007160 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02007161 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
7162 /* Line In pin: input */
Takashi Iwai05acb862005-06-10 19:50:25 +02007163 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Takashi Iwaie9edcee2005-06-13 14:16:38 +02007164 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
7165 /* Line-2 In: Headphone output (output 0 - 0x0c) */
7166 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
7167 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7168 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
Linus Torvalds1da177e2005-04-16 15:20:36 -07007169 /* CD pin widget for input */
Takashi Iwai05acb862005-06-10 19:50:25 +02007170 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Linus Torvalds1da177e2005-04-16 15:20:36 -07007171
7172 /* FIXME: use matrix-type input source selection */
7173 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
Linus Torvalds1da177e2005-04-16 15:20:36 -07007174 /* Input mixer2 */
Takashi Iwai05acb862005-06-10 19:50:25 +02007175 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7176 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
7177 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
7178 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Linus Torvalds1da177e2005-04-16 15:20:36 -07007179 /* Input mixer3 */
Takashi Iwai05acb862005-06-10 19:50:25 +02007180 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7181 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
7182 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
7183 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Takashi Iwai05acb862005-06-10 19:50:25 +02007184 /* ADC2: mute amp left and right */
7185 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Takashi Iwai71fe7b82005-05-25 18:11:40 +02007186 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
Takashi Iwai05acb862005-06-10 19:50:25 +02007187 /* ADC3: mute amp left and right */
7188 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Takashi Iwai71fe7b82005-05-25 18:11:40 +02007189 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
Linus Torvalds1da177e2005-04-16 15:20:36 -07007190
7191 { }
7192};
7193
Takashi Iwai49535502009-06-30 15:28:30 +02007194static struct hda_verb alc882_adc1_init_verbs[] = {
7195 /* Input mixer1: unmute Mic, F-Mic, Line, CD inputs */
7196 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7197 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
7198 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
7199 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
7200 /* ADC1: mute amp left and right */
7201 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7202 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
7203 { }
7204};
7205
Takashi Iwai4b146cb2006-07-28 14:42:36 +02007206static struct hda_verb alc882_eapd_verbs[] = {
7207 /* change to EAPD mode */
7208 {0x20, AC_VERB_SET_COEF_INDEX, 0x07},
Andrew L. Neporadab373bde2006-11-07 11:37:08 +01007209 {0x20, AC_VERB_SET_PROC_COEF, 0x3060},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02007210 { }
Takashi Iwai4b146cb2006-07-28 14:42:36 +02007211};
7212
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02007213static struct hda_verb alc889_eapd_verbs[] = {
7214 {0x14, AC_VERB_SET_EAPD_BTLENABLE, 2},
7215 {0x15, AC_VERB_SET_EAPD_BTLENABLE, 2},
7216 { }
7217};
7218
Wu Fengguang6732bd02009-07-30 09:19:14 +02007219static struct hda_verb alc_hp15_unsol_verbs[] = {
7220 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
7221 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
7222 {}
7223};
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02007224
7225static struct hda_verb alc885_init_verbs[] = {
7226 /* Front mixer: unmute input/output amp left and right (volume = 0) */
7227 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7228 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7229 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
7230 /* Rear mixer */
7231 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7232 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7233 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
7234 /* CLFE mixer */
7235 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7236 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7237 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
7238 /* Side mixer */
7239 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7240 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7241 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
7242
7243 /* mute analog input loopbacks */
7244 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7245 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
7246 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
7247
7248 /* Front HP Pin: output 0 (0x0c) */
Wu Fengguang6732bd02009-07-30 09:19:14 +02007249 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02007250 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7251 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
7252 /* Front Pin: output 0 (0x0c) */
7253 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7254 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7255 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
7256 /* Rear Pin: output 1 (0x0d) */
7257 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7258 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7259 {0x19, AC_VERB_SET_CONNECT_SEL, 0x01},
7260 /* CLFE Pin: output 2 (0x0e) */
7261 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7262 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7263 {0x16, AC_VERB_SET_CONNECT_SEL, 0x02},
7264 /* Side Pin: output 3 (0x0f) */
7265 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7266 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7267 {0x17, AC_VERB_SET_CONNECT_SEL, 0x03},
7268 /* Mic (rear) pin: input vref at 80% */
7269 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
7270 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
7271 /* Front Mic pin: input vref at 80% */
7272 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
7273 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
7274 /* Line In pin: input */
7275 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
7276 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
7277
7278 /* Mixer elements: 0x18, , 0x1a, 0x1b */
7279 /* Input mixer1 */
7280 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(3)},
7281 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7282 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
7283 /* Input mixer2 */
7284 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7285 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
7286 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
7287 /* Input mixer3 */
7288 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
7289 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7290 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
7291 /* ADC2: mute amp left and right */
7292 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7293 /* ADC3: mute amp left and right */
7294 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7295
7296 { }
7297};
7298
7299static struct hda_verb alc885_init_input_verbs[] = {
7300 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7301 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
7302 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(3)},
7303 { }
7304};
7305
7306
7307/* Unmute Selector 24h and set the default input to front mic */
7308static struct hda_verb alc889_init_input_verbs[] = {
7309 {0x24, AC_VERB_SET_CONNECT_SEL, 0x00},
7310 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7311 { }
7312};
7313
7314
Takashi Iwai49535502009-06-30 15:28:30 +02007315#define alc883_init_verbs alc882_base_init_verbs
7316
Tobin Davis9102cd12006-12-15 10:02:12 +01007317/* Mac Pro test */
7318static struct snd_kcontrol_new alc882_macpro_mixer[] = {
7319 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7320 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
7321 HDA_CODEC_MUTE("Headphone Playback Switch", 0x18, 0x0, HDA_OUTPUT),
7322 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x01, HDA_INPUT),
7323 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x01, HDA_INPUT),
Takashi Iwai45bdd1c2009-02-06 16:11:25 +01007324 /* FIXME: this looks suspicious...
Jaroslav Kyselad355c82a2009-11-03 15:47:25 +01007325 HDA_CODEC_VOLUME("Beep Playback Volume", 0x0b, 0x02, HDA_INPUT),
7326 HDA_CODEC_MUTE("Beep Playback Switch", 0x0b, 0x02, HDA_INPUT),
Takashi Iwai45bdd1c2009-02-06 16:11:25 +01007327 */
Tobin Davis9102cd12006-12-15 10:02:12 +01007328 { } /* end */
7329};
7330
7331static struct hda_verb alc882_macpro_init_verbs[] = {
7332 /* Front mixer: unmute input/output amp left and right (volume = 0) */
7333 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7334 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7335 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
7336 /* Front Pin: output 0 (0x0c) */
7337 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7338 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7339 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
7340 /* Front Mic pin: input vref at 80% */
7341 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
7342 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
7343 /* Speaker: output */
7344 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7345 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7346 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x04},
7347 /* Headphone output (output 0 - 0x0c) */
7348 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
7349 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7350 {0x18, AC_VERB_SET_CONNECT_SEL, 0x00},
7351
7352 /* FIXME: use matrix-type input source selection */
7353 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
7354 /* Input mixer1: unmute Mic, F-Mic, Line, CD inputs */
7355 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7356 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
7357 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
7358 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
7359 /* Input mixer2 */
7360 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7361 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
7362 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
7363 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
7364 /* Input mixer3 */
7365 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7366 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
7367 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
7368 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
7369 /* ADC1: mute amp left and right */
7370 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7371 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
7372 /* ADC2: mute amp left and right */
7373 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7374 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
7375 /* ADC3: mute amp left and right */
7376 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7377 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
7378
7379 { }
7380};
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02007381
Kacper Szczesniak41d55452009-05-07 12:47:43 +02007382/* Macbook 5,1 */
7383static struct hda_verb alc885_mb5_init_verbs[] = {
Kacper Szczesniak92b9de82009-06-02 00:55:19 +02007384 /* DACs */
7385 {0x02, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7386 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7387 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7388 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Kacper Szczesniak41d55452009-05-07 12:47:43 +02007389 /* Front mixer */
Kacper Szczesniak41d55452009-05-07 12:47:43 +02007390 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7391 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7392 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Kacper Szczesniak92b9de82009-06-02 00:55:19 +02007393 /* Surround mixer */
7394 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7395 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7396 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
7397 /* LFE mixer */
7398 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7399 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7400 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
7401 /* HP mixer */
7402 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7403 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7404 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
7405 /* Front Pin (0x0c) */
Kacper Szczesniak41d55452009-05-07 12:47:43 +02007406 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT | 0x01},
7407 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Kacper Szczesniak92b9de82009-06-02 00:55:19 +02007408 {0x18, AC_VERB_SET_CONNECT_SEL, 0x00},
7409 /* LFE Pin (0x0e) */
7410 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT | 0x01},
7411 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7412 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x02},
7413 /* HP Pin (0x0f) */
Kacper Szczesniak41d55452009-05-07 12:47:43 +02007414 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7415 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Kacper Szczesniak92b9de82009-06-02 00:55:19 +02007416 {0x14, AC_VERB_SET_CONNECT_SEL, 0x03},
Kacper Szczesniak41d55452009-05-07 12:47:43 +02007417 /* Front Mic pin: input vref at 80% */
7418 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
7419 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
7420 /* Line In pin */
7421 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
7422 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
7423
7424 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7425 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
7426 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
7427 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
7428 { }
7429};
7430
Takashi Iwai87350ad2007-08-16 18:19:38 +02007431/* Macbook Pro rev3 */
7432static struct hda_verb alc885_mbp3_init_verbs[] = {
7433 /* Front mixer: unmute input/output amp left and right (volume = 0) */
7434 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7435 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7436 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
7437 /* Rear mixer */
7438 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7439 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7440 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Takashi Iwaia3f730a2009-08-31 08:15:26 +02007441 /* HP mixer */
7442 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7443 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7444 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Takashi Iwai87350ad2007-08-16 18:19:38 +02007445 /* Front Pin: output 0 (0x0c) */
7446 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7447 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7448 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
Takashi Iwaia3f730a2009-08-31 08:15:26 +02007449 /* HP Pin: output 0 (0x0e) */
Takashi Iwai87350ad2007-08-16 18:19:38 +02007450 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc4},
Takashi Iwaia3f730a2009-08-31 08:15:26 +02007451 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7452 {0x15, AC_VERB_SET_CONNECT_SEL, 0x02},
Takashi Iwai87350ad2007-08-16 18:19:38 +02007453 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
7454 /* Mic (rear) pin: input vref at 80% */
7455 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
7456 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
7457 /* Front Mic pin: input vref at 80% */
7458 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
7459 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
7460 /* Line In pin: use output 1 when in LineOut mode */
7461 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
7462 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
7463 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x01},
7464
7465 /* FIXME: use matrix-type input source selection */
7466 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
7467 /* Input mixer1: unmute Mic, F-Mic, Line, CD inputs */
7468 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7469 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
7470 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
7471 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
7472 /* Input mixer2 */
7473 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7474 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
7475 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
7476 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
7477 /* Input mixer3 */
7478 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7479 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
7480 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
7481 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
7482 /* ADC1: mute amp left and right */
7483 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7484 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
7485 /* ADC2: mute amp left and right */
7486 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7487 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
7488 /* ADC3: mute amp left and right */
7489 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7490 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
7491
7492 { }
7493};
7494
Nicola Fagnanic54728d2007-07-19 23:28:52 +02007495/* iMac 24 mixer. */
7496static struct snd_kcontrol_new alc885_imac24_mixer[] = {
7497 HDA_CODEC_VOLUME("Master Playback Volume", 0x0c, 0x00, HDA_OUTPUT),
7498 HDA_CODEC_MUTE("Master Playback Switch", 0x0c, 0x00, HDA_INPUT),
7499 { } /* end */
7500};
7501
7502/* iMac 24 init verbs. */
7503static struct hda_verb alc885_imac24_init_verbs[] = {
7504 /* Internal speakers: output 0 (0x0c) */
7505 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7506 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7507 {0x18, AC_VERB_SET_CONNECT_SEL, 0x00},
7508 /* Internal speakers: output 0 (0x0c) */
7509 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7510 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7511 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x00},
7512 /* Headphone: output 0 (0x0c) */
7513 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
7514 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
7515 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
7516 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
7517 /* Front Mic: input vref at 80% */
7518 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
7519 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
7520 { }
7521};
7522
7523/* Toggle speaker-output according to the hp-jack state */
Takashi Iwai4f5d17062009-08-11 18:17:46 +02007524static void alc885_imac24_setup(struct hda_codec *codec)
Nicola Fagnanic54728d2007-07-19 23:28:52 +02007525{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02007526 struct alc_spec *spec = codec->spec;
Nicola Fagnanic54728d2007-07-19 23:28:52 +02007527
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02007528 spec->autocfg.hp_pins[0] = 0x14;
7529 spec->autocfg.speaker_pins[0] = 0x18;
7530 spec->autocfg.speaker_pins[1] = 0x1a;
Nicola Fagnanic54728d2007-07-19 23:28:52 +02007531}
7532
Takashi Iwai4f5d17062009-08-11 18:17:46 +02007533static void alc885_mbp3_setup(struct hda_codec *codec)
Nicola Fagnanic54728d2007-07-19 23:28:52 +02007534{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02007535 struct alc_spec *spec = codec->spec;
Nicola Fagnanic54728d2007-07-19 23:28:52 +02007536
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02007537 spec->autocfg.hp_pins[0] = 0x15;
7538 spec->autocfg.speaker_pins[0] = 0x14;
Takashi Iwai87350ad2007-08-16 18:19:38 +02007539}
7540
7541
Kailang Yang272a5272007-05-14 11:00:38 +02007542static struct hda_verb alc882_targa_verbs[] = {
7543 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7544 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7545
7546 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
7547 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Kailang Yangea1fb292008-08-26 12:58:38 +02007548
Kailang Yang272a5272007-05-14 11:00:38 +02007549 {0x18, AC_VERB_SET_CONNECT_SEL, 0x02}, /* mic/clfe */
7550 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x01}, /* line/surround */
7551 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
7552
7553 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
Kailang Yang272a5272007-05-14 11:00:38 +02007554 { } /* end */
7555};
7556
7557/* toggle speaker-output according to the hp-jack state */
7558static void alc882_targa_automute(struct hda_codec *codec)
7559{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02007560 struct alc_spec *spec = codec->spec;
7561 alc_automute_amp(codec);
Takashi Iwai82beb8f2007-08-10 17:09:26 +02007562 snd_hda_codec_write_cache(codec, 1, 0, AC_VERB_SET_GPIO_DATA,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02007563 spec->jack_present ? 1 : 3);
7564}
7565
Takashi Iwai4f5d17062009-08-11 18:17:46 +02007566static void alc882_targa_setup(struct hda_codec *codec)
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02007567{
7568 struct alc_spec *spec = codec->spec;
7569
7570 spec->autocfg.hp_pins[0] = 0x14;
7571 spec->autocfg.speaker_pins[0] = 0x1b;
Kailang Yang272a5272007-05-14 11:00:38 +02007572}
7573
7574static void alc882_targa_unsol_event(struct hda_codec *codec, unsigned int res)
7575{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02007576 if ((res >> 26) == ALC880_HP_EVENT)
Kailang Yang272a5272007-05-14 11:00:38 +02007577 alc882_targa_automute(codec);
Kailang Yang272a5272007-05-14 11:00:38 +02007578}
7579
7580static struct hda_verb alc882_asus_a7j_verbs[] = {
7581 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7582 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7583
7584 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
7585 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7586 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Kailang Yangea1fb292008-08-26 12:58:38 +02007587
Kailang Yang272a5272007-05-14 11:00:38 +02007588 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00}, /* Front */
7589 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
7590 {0x16, AC_VERB_SET_CONNECT_SEL, 0x00}, /* Front */
7591
7592 {0x18, AC_VERB_SET_CONNECT_SEL, 0x02}, /* mic/clfe */
7593 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x01}, /* line/surround */
7594 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
7595 { } /* end */
7596};
7597
Takashi Iwai914759b2007-09-06 14:52:04 +02007598static struct hda_verb alc882_asus_a7m_verbs[] = {
7599 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7600 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7601
7602 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
7603 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
7604 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Kailang Yangea1fb292008-08-26 12:58:38 +02007605
Takashi Iwai914759b2007-09-06 14:52:04 +02007606 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00}, /* Front */
7607 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
7608 {0x16, AC_VERB_SET_CONNECT_SEL, 0x00}, /* Front */
7609
7610 {0x18, AC_VERB_SET_CONNECT_SEL, 0x02}, /* mic/clfe */
7611 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x01}, /* line/surround */
7612 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP */
7613 { } /* end */
7614};
7615
Tobin Davis9102cd12006-12-15 10:02:12 +01007616static void alc882_gpio_mute(struct hda_codec *codec, int pin, int muted)
7617{
7618 unsigned int gpiostate, gpiomask, gpiodir;
7619
7620 gpiostate = snd_hda_codec_read(codec, codec->afg, 0,
7621 AC_VERB_GET_GPIO_DATA, 0);
7622
7623 if (!muted)
7624 gpiostate |= (1 << pin);
7625 else
7626 gpiostate &= ~(1 << pin);
7627
7628 gpiomask = snd_hda_codec_read(codec, codec->afg, 0,
7629 AC_VERB_GET_GPIO_MASK, 0);
7630 gpiomask |= (1 << pin);
7631
7632 gpiodir = snd_hda_codec_read(codec, codec->afg, 0,
7633 AC_VERB_GET_GPIO_DIRECTION, 0);
7634 gpiodir |= (1 << pin);
7635
7636
7637 snd_hda_codec_write(codec, codec->afg, 0,
7638 AC_VERB_SET_GPIO_MASK, gpiomask);
7639 snd_hda_codec_write(codec, codec->afg, 0,
7640 AC_VERB_SET_GPIO_DIRECTION, gpiodir);
7641
7642 msleep(1);
7643
7644 snd_hda_codec_write(codec, codec->afg, 0,
7645 AC_VERB_SET_GPIO_DATA, gpiostate);
7646}
7647
Takashi Iwai7debbe52007-08-16 15:01:03 +02007648/* set up GPIO at initialization */
7649static void alc885_macpro_init_hook(struct hda_codec *codec)
7650{
7651 alc882_gpio_mute(codec, 0, 0);
7652 alc882_gpio_mute(codec, 1, 0);
7653}
7654
7655/* set up GPIO and update auto-muting at initialization */
7656static void alc885_imac24_init_hook(struct hda_codec *codec)
7657{
7658 alc885_macpro_init_hook(codec);
Takashi Iwai4f5d17062009-08-11 18:17:46 +02007659 alc_automute_amp(codec);
Takashi Iwai7debbe52007-08-16 15:01:03 +02007660}
7661
Kailang Yangdf694da2005-12-05 19:42:22 +01007662/*
7663 * generic initialization of ADC, input mixers and output mixers
7664 */
Takashi Iwai49535502009-06-30 15:28:30 +02007665static struct hda_verb alc883_auto_init_verbs[] = {
Kailang Yangdf694da2005-12-05 19:42:22 +01007666 /*
7667 * Unmute ADC0-2 and set the default input to mic-in
7668 */
Kailang Yangdf694da2005-12-05 19:42:22 +01007669 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
7670 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7671 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
7672 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7673
Takashi Iwaicb53c622007-08-10 17:21:45 +02007674 /* Mute input amps (CD, Line In, Mic 1 & Mic 2) of the analog-loopback
Kailang Yangdf694da2005-12-05 19:42:22 +01007675 * mixer widget
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02007676 * Note: PASD motherboards uses the Line In 2 as the input for
7677 * front panel mic (mic 2)
Kailang Yangdf694da2005-12-05 19:42:22 +01007678 */
7679 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +02007680 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
7681 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
7682 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
7683 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
7684 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Kailang Yangdf694da2005-12-05 19:42:22 +01007685
7686 /*
7687 * Set up output mixers (0x0c - 0x0f)
7688 */
7689 /* set vol=0 to output mixers */
7690 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7691 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7692 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7693 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
7694 /* set up input amps for analog loopback */
7695 /* Amp Indices: DAC = 0, mixer = 1 */
7696 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7697 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7698 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7699 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7700 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7701 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7702 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7703 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7704 {0x26, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
7705 {0x26, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
7706
7707 /* FIXME: use matrix-type input source selection */
7708 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
Kailang Yangdf694da2005-12-05 19:42:22 +01007709 /* Input mixer2 */
7710 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
7711 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
7712 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
7713 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
7714 /* Input mixer3 */
7715 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
7716 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
7717 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
7718 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
7719
7720 { }
7721};
7722
Torben Schulzeb4c41d2009-05-18 15:02:35 +02007723/* 2ch mode (Speaker:front, Subwoofer:CLFE, Line:input, Headphones:front) */
7724static struct hda_verb alc889A_mb31_ch2_init[] = {
7725 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP as front */
7726 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE}, /* Subwoofer on */
7727 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN}, /* Line as input */
7728 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE}, /* Line off */
7729 { } /* end */
7730};
7731
7732/* 4ch mode (Speaker:front, Subwoofer:CLFE, Line:CLFE, Headphones:front) */
7733static struct hda_verb alc889A_mb31_ch4_init[] = {
7734 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00}, /* HP as front */
7735 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE}, /* Subwoofer on */
7736 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT}, /* Line as output */
7737 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE}, /* Line on */
7738 { } /* end */
7739};
7740
7741/* 5ch mode (Speaker:front, Subwoofer:CLFE, Line:input, Headphones:rear) */
7742static struct hda_verb alc889A_mb31_ch5_init[] = {
7743 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01}, /* HP as rear */
7744 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE}, /* Subwoofer on */
7745 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN}, /* Line as input */
7746 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE}, /* Line off */
7747 { } /* end */
7748};
7749
7750/* 6ch mode (Speaker:front, Subwoofer:off, Line:CLFE, Headphones:Rear) */
7751static struct hda_verb alc889A_mb31_ch6_init[] = {
7752 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01}, /* HP as front */
7753 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE}, /* Subwoofer off */
7754 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT}, /* Line as output */
7755 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE}, /* Line on */
7756 { } /* end */
7757};
7758
7759static struct hda_channel_mode alc889A_mb31_6ch_modes[4] = {
7760 { 2, alc889A_mb31_ch2_init },
7761 { 4, alc889A_mb31_ch4_init },
7762 { 5, alc889A_mb31_ch5_init },
7763 { 6, alc889A_mb31_ch6_init },
7764};
7765
Andrew L. Neporadab373bde2006-11-07 11:37:08 +01007766static struct hda_verb alc883_medion_eapd_verbs[] = {
7767 /* eanable EAPD on medion laptop */
7768 {0x20, AC_VERB_SET_COEF_INDEX, 0x07},
7769 {0x20, AC_VERB_SET_PROC_COEF, 0x3070},
7770 { }
7771};
7772
Takashi Iwai49535502009-06-30 15:28:30 +02007773#define alc883_base_mixer alc882_base_mixer
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007774
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +01007775static struct snd_kcontrol_new alc883_mitac_mixer[] = {
7776 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7777 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
7778 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
7779 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
7780 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
7781 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
7782 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
7783 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
7784 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
7785 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
7786 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
7787 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
7788 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +01007789 { } /* end */
7790};
7791
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01007792static struct snd_kcontrol_new alc883_clevo_m720_mixer[] = {
Jiang zhe368c7a92008-03-04 11:20:33 +01007793 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7794 HDA_BIND_MUTE("Headphone Playback Switch", 0x0c, 2, HDA_INPUT),
7795 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
7796 HDA_BIND_MUTE("Speaker Playback Switch", 0x0d, 2, HDA_INPUT),
7797 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
7798 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
7799 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
7800 HDA_CODEC_VOLUME("Int Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
7801 HDA_CODEC_VOLUME("Int Mic Boost", 0x19, 0, HDA_INPUT),
7802 HDA_CODEC_MUTE("Int Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Jiang zhe368c7a92008-03-04 11:20:33 +01007803 { } /* end */
7804};
7805
Jiang zhefb97dc62008-03-06 11:07:11 +01007806static struct snd_kcontrol_new alc883_2ch_fujitsu_pi2515_mixer[] = {
7807 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7808 HDA_BIND_MUTE("Headphone Playback Switch", 0x0c, 2, HDA_INPUT),
7809 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
7810 HDA_BIND_MUTE("Speaker Playback Switch", 0x0d, 2, HDA_INPUT),
7811 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
7812 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
7813 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
7814 HDA_CODEC_VOLUME("Int Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
7815 HDA_CODEC_VOLUME("Int Mic Boost", 0x19, 0, HDA_INPUT),
7816 HDA_CODEC_MUTE("Int Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Jiang zhefb97dc62008-03-06 11:07:11 +01007817 { } /* end */
7818};
7819
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007820static struct snd_kcontrol_new alc883_3ST_2ch_mixer[] = {
7821 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7822 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
7823 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
7824 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
7825 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
7826 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
7827 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
7828 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01007829 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007830 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
7831 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01007832 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007833 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007834 { } /* end */
7835};
7836
7837static struct snd_kcontrol_new alc883_3ST_6ch_mixer[] = {
7838 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7839 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
7840 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
7841 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
7842 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
7843 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
7844 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
7845 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
7846 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
7847 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
7848 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
7849 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
7850 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
7851 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01007852 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007853 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
7854 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01007855 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007856 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +02007857 { } /* end */
7858};
7859
Jiang zhe17bba1b2008-06-04 12:11:07 +02007860static struct snd_kcontrol_new alc883_3ST_6ch_intel_mixer[] = {
7861 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7862 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
7863 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
7864 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
7865 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0,
7866 HDA_OUTPUT),
7867 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
7868 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
7869 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
7870 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
7871 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
7872 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
7873 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
7874 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
7875 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
7876 HDA_CODEC_VOLUME("Mic Boost", 0x19, 0, HDA_INPUT),
7877 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
7878 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
7879 HDA_CODEC_VOLUME("Front Mic Boost", 0x18, 0, HDA_INPUT),
7880 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Jiang zhe17bba1b2008-06-04 12:11:07 +02007881 { } /* end */
7882};
7883
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02007884static struct snd_kcontrol_new alc885_8ch_intel_mixer[] = {
7885 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7886 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
7887 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
7888 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
7889 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0,
7890 HDA_OUTPUT),
7891 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
7892 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
7893 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
7894 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0f, 0x0, HDA_OUTPUT),
7895 HDA_BIND_MUTE("Speaker Playback Switch", 0x0f, 2, HDA_INPUT),
7896 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
7897 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
7898 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
7899 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x3, HDA_INPUT),
7900 HDA_CODEC_VOLUME("Mic Boost", 0x1b, 0, HDA_INPUT),
7901 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x3, HDA_INPUT),
7902 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
7903 HDA_CODEC_VOLUME("Front Mic Boost", 0x18, 0, HDA_INPUT),
7904 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
7905 { } /* end */
7906};
7907
Takashi Iwaid1d985f2006-11-23 19:27:12 +01007908static struct snd_kcontrol_new alc883_fivestack_mixer[] = {
Tobin Davisc07584c2006-10-13 12:32:16 +02007909 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwai0701e062008-06-27 16:30:57 +02007910 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Tobin Davisc07584c2006-10-13 12:32:16 +02007911 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
Takashi Iwai0701e062008-06-27 16:30:57 +02007912 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
Tobin Davisc07584c2006-10-13 12:32:16 +02007913 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
7914 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
Takashi Iwai0701e062008-06-27 16:30:57 +02007915 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
7916 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
Tobin Davisc07584c2006-10-13 12:32:16 +02007917 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
7918 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
7919 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
7920 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
7921 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
7922 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01007923 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Tobin Davisc07584c2006-10-13 12:32:16 +02007924 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
7925 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01007926 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
Tobin Davisc07584c2006-10-13 12:32:16 +02007927 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Tobin Davisc07584c2006-10-13 12:32:16 +02007928 { } /* end */
7929};
7930
Sasha Alexandrc2592492009-06-16 14:52:54 -04007931static struct snd_kcontrol_new alc883_targa_mixer[] = {
Kailang Yangccc656c2006-10-17 12:32:26 +02007932 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwaib99dba32009-09-17 18:23:00 +02007933 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Kailang Yangccc656c2006-10-17 12:32:26 +02007934 HDA_CODEC_MUTE("Headphone Playback Switch", 0x14, 0x0, HDA_OUTPUT),
Takashi Iwaib99dba32009-09-17 18:23:00 +02007935 HDA_CODEC_MUTE("Speaker Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
Kailang Yangccc656c2006-10-17 12:32:26 +02007936 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
7937 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
7938 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
7939 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
7940 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
7941 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
7942 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
7943 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
7944 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
7945 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
7946 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01007947 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Kailang Yangccc656c2006-10-17 12:32:26 +02007948 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Kailang Yangccc656c2006-10-17 12:32:26 +02007949 { } /* end */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02007950};
Kailang Yangccc656c2006-10-17 12:32:26 +02007951
Sasha Alexandrc2592492009-06-16 14:52:54 -04007952static struct snd_kcontrol_new alc883_targa_2ch_mixer[] = {
Kailang Yangccc656c2006-10-17 12:32:26 +02007953 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwaib99dba32009-09-17 18:23:00 +02007954 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Kailang Yangccc656c2006-10-17 12:32:26 +02007955 HDA_CODEC_MUTE("Headphone Playback Switch", 0x14, 0x0, HDA_OUTPUT),
Takashi Iwaib99dba32009-09-17 18:23:00 +02007956 HDA_CODEC_MUTE("Speaker Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
Kailang Yangccc656c2006-10-17 12:32:26 +02007957 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
7958 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
7959 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
Thomas De Schampheleire32360412007-01-24 16:13:35 +01007960 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Kailang Yangccc656c2006-10-17 12:32:26 +02007961 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Jiang zhe4383fae2008-04-14 12:58:57 +02007962 HDA_CODEC_VOLUME("Int Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
7963 HDA_CODEC_VOLUME("Int Mic Boost", 0x19, 0, HDA_INPUT),
7964 HDA_CODEC_MUTE("Int Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Kailang Yangccc656c2006-10-17 12:32:26 +02007965 { } /* end */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02007966};
Kailang Yangccc656c2006-10-17 12:32:26 +02007967
Takashi Iwaib99dba32009-09-17 18:23:00 +02007968static struct snd_kcontrol_new alc883_targa_8ch_mixer[] = {
7969 HDA_CODEC_VOLUME("Side Playback Volume", 0x0f, 0x0, HDA_OUTPUT),
7970 HDA_BIND_MUTE("Side Playback Switch", 0x0f, 2, HDA_INPUT),
7971 HDA_CODEC_VOLUME("Int Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
7972 HDA_CODEC_VOLUME("Int Mic Boost", 0x19, 0, HDA_INPUT),
7973 HDA_CODEC_MUTE("Int Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
7974 { } /* end */
7975};
7976
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007977static struct snd_kcontrol_new alc883_lenovo_101e_2ch_mixer[] = {
7978 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7979 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Takashi Iwai86cd9292008-01-28 18:09:56 +01007980 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
7981 HDA_BIND_MUTE("Speaker Playback Switch", 0x0d, 2, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007982 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
7983 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
7984 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
7985 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007986 { } /* end */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02007987};
Kailang Yangbc9f98a2007-04-12 13:06:07 +02007988
Kailang Yang272a5272007-05-14 11:00:38 +02007989static struct snd_kcontrol_new alc883_lenovo_nb0763_mixer[] = {
7990 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
7991 HDA_BIND_MUTE("Speaker Playback Switch", 0x0c, 2, HDA_INPUT),
7992 HDA_CODEC_MUTE("Headphone Playback Switch", 0x14, 0x0, HDA_OUTPUT),
7993 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
7994 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
7995 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
7996 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
7997 HDA_CODEC_VOLUME("iMic Playback Volume", 0x0b, 0x1, HDA_INPUT),
7998 HDA_CODEC_MUTE("iMic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Kailang Yang272a5272007-05-14 11:00:38 +02007999 { } /* end */
8000};
8001
8002static struct snd_kcontrol_new alc883_medion_md2_mixer[] = {
8003 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
8004 HDA_CODEC_MUTE("Headphone Playback Switch", 0x14, 0x0, HDA_OUTPUT),
8005 HDA_CODEC_MUTE("Front Playback Switch", 0x15, 0x0, HDA_OUTPUT),
8006 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
8007 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
8008 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
8009 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
8010 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
8011 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
Kailang Yang272a5272007-05-14 11:00:38 +02008012 { } /* end */
Kailang Yangea1fb292008-08-26 12:58:38 +02008013};
Kailang Yang272a5272007-05-14 11:00:38 +02008014
Tobin Davis2880a862007-08-07 11:50:26 +02008015static struct snd_kcontrol_new alc883_acer_aspire_mixer[] = {
Kailang Yangd1a991a2007-08-15 16:21:59 +02008016 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
8017 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
Tobin Davis2880a862007-08-07 11:50:26 +02008018 HDA_CODEC_MUTE("Headphone Playback Switch", 0x14, 0x0, HDA_OUTPUT),
Tobin Davis2880a862007-08-07 11:50:26 +02008019 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
8020 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
Kailang Yangd1a991a2007-08-15 16:21:59 +02008021 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
8022 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
8023 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Tobin Davis2880a862007-08-07 11:50:26 +02008024 { } /* end */
Kailang Yangd1a991a2007-08-15 16:21:59 +02008025};
Tobin Davis2880a862007-08-07 11:50:26 +02008026
Tony Vroond2fd4b02009-06-21 00:40:10 +01008027static struct snd_kcontrol_new alc888_acer_aspire_6530_mixer[] = {
8028 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
8029 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
8030 HDA_CODEC_VOLUME("LFE Playback Volume", 0x0f, 0x0, HDA_OUTPUT),
8031 HDA_BIND_MUTE("LFE Playback Switch", 0x0f, 2, HDA_INPUT),
Tony Vroon684a8842009-06-26 09:27:50 +01008032 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
8033 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
Tony Vroond2fd4b02009-06-21 00:40:10 +01008034 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
8035 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
8036 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
8037 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
8038 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
8039 { } /* end */
8040};
8041
Kailang Yange2757d52008-08-26 13:17:46 +02008042static struct snd_kcontrol_new alc888_lenovo_sky_mixer[] = {
8043 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
8044 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
8045 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0e, 0x0, HDA_OUTPUT),
8046 HDA_BIND_MUTE("Surround Playback Switch", 0x0e, 2, HDA_INPUT),
8047 HDA_CODEC_VOLUME_MONO("Center Playback Volume",
8048 0x0d, 1, 0x0, HDA_OUTPUT),
8049 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0d, 2, 0x0, HDA_OUTPUT),
8050 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0d, 1, 2, HDA_INPUT),
8051 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0d, 2, 2, HDA_INPUT),
8052 HDA_CODEC_VOLUME("Side Playback Volume", 0x0f, 0x0, HDA_OUTPUT),
8053 HDA_BIND_MUTE("Side Playback Switch", 0x0f, 2, HDA_INPUT),
Kailang Yange2757d52008-08-26 13:17:46 +02008054 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
8055 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
8056 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
8057 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
8058 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
8059 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
8060 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
8061 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
8062 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
8063 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Kailang Yange2757d52008-08-26 13:17:46 +02008064 { } /* end */
8065};
8066
Torben Schulzeb4c41d2009-05-18 15:02:35 +02008067static struct snd_kcontrol_new alc889A_mb31_mixer[] = {
8068 /* Output mixers */
8069 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x00, HDA_OUTPUT),
8070 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 0x02, HDA_INPUT),
8071 HDA_CODEC_VOLUME("Surround Playback Volume", 0x0d, 0x00, HDA_OUTPUT),
8072 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 0x02, HDA_INPUT),
8073 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x0e, 1, 0x00,
8074 HDA_OUTPUT),
8075 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 0x02, HDA_INPUT),
8076 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x0e, 2, 0x00, HDA_OUTPUT),
8077 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 0x02, HDA_INPUT),
8078 /* Output switches */
8079 HDA_CODEC_MUTE("Enable Speaker", 0x14, 0x00, HDA_OUTPUT),
8080 HDA_CODEC_MUTE("Enable Headphones", 0x15, 0x00, HDA_OUTPUT),
8081 HDA_CODEC_MUTE_MONO("Enable LFE", 0x16, 2, 0x00, HDA_OUTPUT),
8082 /* Boost mixers */
8083 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0x00, HDA_INPUT),
8084 HDA_CODEC_VOLUME("Line Boost", 0x1a, 0x00, HDA_INPUT),
8085 /* Input mixers */
8086 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x00, HDA_INPUT),
8087 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x00, HDA_INPUT),
8088 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
8089 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
8090 { } /* end */
8091};
8092
Guido Günther3e1647c2009-06-05 00:47:26 +02008093static struct snd_kcontrol_new alc883_vaiott_mixer[] = {
8094 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
8095 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
8096 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
8097 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
8098 HDA_CODEC_VOLUME("Mic Boost", 0x19, 0, HDA_INPUT),
8099 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
8100 { } /* end */
8101};
8102
Kailang Yange2757d52008-08-26 13:17:46 +02008103static struct hda_bind_ctls alc883_bind_cap_vol = {
8104 .ops = &snd_hda_bind_vol,
8105 .values = {
8106 HDA_COMPOSE_AMP_VAL(0x08, 3, 0, HDA_INPUT),
8107 HDA_COMPOSE_AMP_VAL(0x09, 3, 0, HDA_INPUT),
8108 0
8109 },
8110};
8111
8112static struct hda_bind_ctls alc883_bind_cap_switch = {
8113 .ops = &snd_hda_bind_sw,
8114 .values = {
8115 HDA_COMPOSE_AMP_VAL(0x08, 3, 0, HDA_INPUT),
8116 HDA_COMPOSE_AMP_VAL(0x09, 3, 0, HDA_INPUT),
8117 0
8118 },
8119};
8120
8121static struct snd_kcontrol_new alc883_asus_eee1601_mixer[] = {
8122 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
8123 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
8124 HDA_CODEC_MUTE("Headphone Playback Switch", 0x14, 0x0, HDA_OUTPUT),
8125 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
8126 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
8127 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
8128 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
8129 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Takashi Iwaif9e336f2008-10-31 16:37:07 +01008130 { } /* end */
8131};
8132
8133static struct snd_kcontrol_new alc883_asus_eee1601_cap_mixer[] = {
Kailang Yange2757d52008-08-26 13:17:46 +02008134 HDA_BIND_VOL("Capture Volume", &alc883_bind_cap_vol),
8135 HDA_BIND_SW("Capture Switch", &alc883_bind_cap_switch),
8136 {
8137 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
8138 /* .name = "Capture Source", */
8139 .name = "Input Source",
8140 .count = 1,
Takashi Iwai54cbc9a2008-10-31 15:24:04 +01008141 .info = alc_mux_enum_info,
8142 .get = alc_mux_enum_get,
8143 .put = alc_mux_enum_put,
Kailang Yange2757d52008-08-26 13:17:46 +02008144 },
8145 { } /* end */
8146};
8147
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008148static struct snd_kcontrol_new alc883_chmode_mixer[] = {
8149 {
8150 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
8151 .name = "Channel Mode",
8152 .info = alc_ch_mode_info,
8153 .get = alc_ch_mode_get,
8154 .put = alc_ch_mode_put,
8155 },
8156 { } /* end */
8157};
8158
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +01008159/* toggle speaker-output according to the hp-jack state */
Takashi Iwai4f5d17062009-08-11 18:17:46 +02008160static void alc883_mitac_setup(struct hda_codec *codec)
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +01008161{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008162 struct alc_spec *spec = codec->spec;
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +01008163
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008164 spec->autocfg.hp_pins[0] = 0x15;
8165 spec->autocfg.speaker_pins[0] = 0x14;
8166 spec->autocfg.speaker_pins[1] = 0x17;
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +01008167}
8168
8169/* auto-toggle front mic */
8170/*
8171static void alc883_mitac_mic_automute(struct hda_codec *codec)
8172{
Wu Fengguang864f92b2009-11-18 12:38:02 +08008173 unsigned char bits = snd_hda_jack_detect(codec, 0x18) ? HDA_AMP_MUTE : 0;
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +01008174
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +01008175 snd_hda_codec_amp_stereo(codec, 0x0b, HDA_INPUT, 1, HDA_AMP_MUTE, bits);
8176}
8177*/
8178
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +01008179static struct hda_verb alc883_mitac_verbs[] = {
8180 /* HP */
8181 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
8182 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
8183 /* Subwoofer */
8184 {0x17, AC_VERB_SET_CONNECT_SEL, 0x02},
8185 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
8186
8187 /* enable unsolicited event */
8188 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
8189 /* {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_MIC_EVENT | AC_USRSP_EN}, */
8190
8191 { } /* end */
8192};
8193
ddiaz@cenditel.gob.vea65cc602009-09-05 16:28:06 -04308194static struct hda_verb alc883_clevo_m540r_verbs[] = {
8195 /* HP */
8196 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
8197 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
8198 /* Int speaker */
8199 /*{0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},*/
8200
8201 /* enable unsolicited event */
8202 /*
8203 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
8204 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_MIC_EVENT | AC_USRSP_EN},
8205 */
8206
8207 { } /* end */
8208};
8209
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01008210static struct hda_verb alc883_clevo_m720_verbs[] = {
Jiang zhe368c7a92008-03-04 11:20:33 +01008211 /* HP */
8212 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
8213 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
8214 /* Int speaker */
8215 {0x14, AC_VERB_SET_CONNECT_SEL, 0x01},
8216 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
8217
8218 /* enable unsolicited event */
8219 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01008220 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_MIC_EVENT | AC_USRSP_EN},
Jiang zhe368c7a92008-03-04 11:20:33 +01008221
8222 { } /* end */
8223};
8224
Jiang zhefb97dc62008-03-06 11:07:11 +01008225static struct hda_verb alc883_2ch_fujitsu_pi2515_verbs[] = {
8226 /* HP */
8227 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
8228 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
8229 /* Subwoofer */
8230 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01},
8231 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
8232
8233 /* enable unsolicited event */
8234 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
8235
8236 { } /* end */
8237};
8238
Sasha Alexandrc2592492009-06-16 14:52:54 -04008239static struct hda_verb alc883_targa_verbs[] = {
Kailang Yangccc656c2006-10-17 12:32:26 +02008240 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
8241 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
8242
8243 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
8244 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Kailang Yangea1fb292008-08-26 12:58:38 +02008245
David Heidelberger64a8be72009-06-08 16:15:18 +02008246/* Connect Line-Out side jack (SPDIF) to Side */
8247 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
8248 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
8249 {0x17, AC_VERB_SET_CONNECT_SEL, 0x03},
8250/* Connect Mic jack to CLFE */
8251 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
8252 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
8253 {0x18, AC_VERB_SET_CONNECT_SEL, 0x02},
8254/* Connect Line-in jack to Surround */
8255 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
8256 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
8257 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x01},
8258/* Connect HP out jack to Front */
8259 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
8260 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
8261 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
Kailang Yangccc656c2006-10-17 12:32:26 +02008262
8263 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
Kailang Yangccc656c2006-10-17 12:32:26 +02008264
8265 { } /* end */
8266};
8267
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008268static struct hda_verb alc883_lenovo_101e_verbs[] = {
8269 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
8270 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_FRONT_EVENT|AC_USRSP_EN},
8271 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT|AC_USRSP_EN},
8272 { } /* end */
8273};
8274
Kailang Yang272a5272007-05-14 11:00:38 +02008275static struct hda_verb alc883_lenovo_nb0763_verbs[] = {
8276 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
8277 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
8278 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
8279 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
8280 { } /* end */
8281};
8282
8283static struct hda_verb alc888_lenovo_ms7195_verbs[] = {
8284 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
8285 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
8286 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
8287 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_FRONT_EVENT | AC_USRSP_EN},
8288 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
8289 { } /* end */
8290};
8291
Kailang Yang189609a2007-08-20 11:31:23 +02008292static struct hda_verb alc883_haier_w66_verbs[] = {
8293 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
8294 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
8295
8296 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
8297
8298 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
8299 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
8300 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
8301 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
8302 { } /* end */
8303};
8304
Kailang Yange2757d52008-08-26 13:17:46 +02008305static struct hda_verb alc888_lenovo_sky_verbs[] = {
8306 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
8307 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
8308 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
8309 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
8310 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
8311 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
8312 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x00},
8313 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
8314 { } /* end */
8315};
8316
Herton Ronaldo Krzesinski8718b702009-03-04 14:22:51 -03008317static struct hda_verb alc888_6st_dell_verbs[] = {
8318 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
8319 { }
8320};
8321
Guido Günther3e1647c2009-06-05 00:47:26 +02008322static struct hda_verb alc883_vaiott_verbs[] = {
8323 /* HP */
8324 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
8325 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
8326
8327 /* enable unsolicited event */
8328 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
8329
8330 { } /* end */
8331};
8332
Takashi Iwai4f5d17062009-08-11 18:17:46 +02008333static void alc888_3st_hp_setup(struct hda_codec *codec)
Herton Ronaldo Krzesinski8718b702009-03-04 14:22:51 -03008334{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008335 struct alc_spec *spec = codec->spec;
Herton Ronaldo Krzesinski8718b702009-03-04 14:22:51 -03008336
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008337 spec->autocfg.hp_pins[0] = 0x1b;
8338 spec->autocfg.speaker_pins[0] = 0x14;
8339 spec->autocfg.speaker_pins[1] = 0x16;
8340 spec->autocfg.speaker_pins[2] = 0x18;
Herton Ronaldo Krzesinski8718b702009-03-04 14:22:51 -03008341}
8342
Claudio Matsuoka4723c022007-07-13 14:36:19 +02008343static struct hda_verb alc888_3st_hp_verbs[] = {
Claudio Matsuoka8341de62007-07-06 12:10:45 +02008344 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00}, /* Front: output 0 (0x0c) */
Herton Ronaldo Krzesinskif32a19e2008-03-18 09:27:08 +01008345 {0x16, AC_VERB_SET_CONNECT_SEL, 0x01}, /* Rear : output 1 (0x0d) */
8346 {0x18, AC_VERB_SET_CONNECT_SEL, 0x02}, /* CLFE : output 2 (0x0e) */
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01008347 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
Herton Ronaldo Krzesinski8718b702009-03-04 14:22:51 -03008348 { } /* end */
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01008349};
8350
Herton Ronaldo Krzesinski3ea0d7c2009-03-04 14:22:50 -03008351/*
8352 * 2ch mode
8353 */
Claudio Matsuoka4723c022007-07-13 14:36:19 +02008354static struct hda_verb alc888_3st_hp_2ch_init[] = {
Claudio Matsuoka8341de62007-07-06 12:10:45 +02008355 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
8356 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
8357 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
8358 { 0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
Herton Ronaldo Krzesinski3ea0d7c2009-03-04 14:22:50 -03008359 { } /* end */
Claudio Matsuoka8341de62007-07-06 12:10:45 +02008360};
8361
Herton Ronaldo Krzesinski3ea0d7c2009-03-04 14:22:50 -03008362/*
8363 * 4ch mode
8364 */
8365static struct hda_verb alc888_3st_hp_4ch_init[] = {
8366 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
8367 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
8368 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
8369 { 0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
8370 { 0x16, AC_VERB_SET_CONNECT_SEL, 0x01 },
8371 { } /* end */
8372};
8373
8374/*
8375 * 6ch mode
8376 */
Claudio Matsuoka4723c022007-07-13 14:36:19 +02008377static struct hda_verb alc888_3st_hp_6ch_init[] = {
Claudio Matsuoka8341de62007-07-06 12:10:45 +02008378 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
8379 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
Herton Ronaldo Krzesinski3ea0d7c2009-03-04 14:22:50 -03008380 { 0x18, AC_VERB_SET_CONNECT_SEL, 0x02 },
Claudio Matsuoka8341de62007-07-06 12:10:45 +02008381 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
8382 { 0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
Herton Ronaldo Krzesinski3ea0d7c2009-03-04 14:22:50 -03008383 { 0x16, AC_VERB_SET_CONNECT_SEL, 0x01 },
8384 { } /* end */
Claudio Matsuoka8341de62007-07-06 12:10:45 +02008385};
8386
Herton Ronaldo Krzesinski3ea0d7c2009-03-04 14:22:50 -03008387static struct hda_channel_mode alc888_3st_hp_modes[3] = {
Claudio Matsuoka4723c022007-07-13 14:36:19 +02008388 { 2, alc888_3st_hp_2ch_init },
Herton Ronaldo Krzesinski3ea0d7c2009-03-04 14:22:50 -03008389 { 4, alc888_3st_hp_4ch_init },
Claudio Matsuoka4723c022007-07-13 14:36:19 +02008390 { 6, alc888_3st_hp_6ch_init },
Claudio Matsuoka8341de62007-07-06 12:10:45 +02008391};
8392
Kailang Yang272a5272007-05-14 11:00:38 +02008393/* toggle front-jack and RCA according to the hp-jack state */
8394static void alc888_lenovo_ms7195_front_automute(struct hda_codec *codec)
8395{
Wu Fengguang864f92b2009-11-18 12:38:02 +08008396 unsigned int present = snd_hda_jack_detect(codec, 0x1b);
Kailang Yangea1fb292008-08-26 12:58:38 +02008397
Takashi Iwai47fd8302007-08-10 17:11:07 +02008398 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
8399 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
8400 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
8401 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
Kailang Yang272a5272007-05-14 11:00:38 +02008402}
8403
8404/* toggle RCA according to the front-jack state */
8405static void alc888_lenovo_ms7195_rca_automute(struct hda_codec *codec)
8406{
Wu Fengguang864f92b2009-11-18 12:38:02 +08008407 unsigned int present = snd_hda_jack_detect(codec, 0x14);
Kailang Yangea1fb292008-08-26 12:58:38 +02008408
Takashi Iwai47fd8302007-08-10 17:11:07 +02008409 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
8410 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
Kailang Yang272a5272007-05-14 11:00:38 +02008411}
Takashi Iwai47fd8302007-08-10 17:11:07 +02008412
Kailang Yang272a5272007-05-14 11:00:38 +02008413static void alc883_lenovo_ms7195_unsol_event(struct hda_codec *codec,
8414 unsigned int res)
8415{
8416 if ((res >> 26) == ALC880_HP_EVENT)
8417 alc888_lenovo_ms7195_front_automute(codec);
8418 if ((res >> 26) == ALC880_FRONT_EVENT)
8419 alc888_lenovo_ms7195_rca_automute(codec);
8420}
8421
8422static struct hda_verb alc883_medion_md2_verbs[] = {
8423 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
8424 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
8425
8426 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
8427
8428 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
8429 { } /* end */
8430};
8431
8432/* toggle speaker-output according to the hp-jack state */
Takashi Iwai4f5d17062009-08-11 18:17:46 +02008433static void alc883_medion_md2_setup(struct hda_codec *codec)
Kailang Yang272a5272007-05-14 11:00:38 +02008434{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008435 struct alc_spec *spec = codec->spec;
Kailang Yangea1fb292008-08-26 12:58:38 +02008436
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008437 spec->autocfg.hp_pins[0] = 0x14;
8438 spec->autocfg.speaker_pins[0] = 0x15;
Kailang Yang272a5272007-05-14 11:00:38 +02008439}
8440
Kailang Yangccc656c2006-10-17 12:32:26 +02008441/* toggle speaker-output according to the hp-jack state */
Sasha Alexandrc2592492009-06-16 14:52:54 -04008442#define alc883_targa_init_hook alc882_targa_init_hook
8443#define alc883_targa_unsol_event alc882_targa_unsol_event
Jiang zhe368c7a92008-03-04 11:20:33 +01008444
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01008445static void alc883_clevo_m720_mic_automute(struct hda_codec *codec)
8446{
8447 unsigned int present;
8448
Takashi Iwaid56757a2009-11-18 08:00:14 +01008449 present = snd_hda_jack_detect(codec, 0x18);
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01008450 snd_hda_codec_amp_stereo(codec, 0x0b, HDA_INPUT, 1,
8451 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
8452}
8453
Takashi Iwai4f5d17062009-08-11 18:17:46 +02008454static void alc883_clevo_m720_setup(struct hda_codec *codec)
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01008455{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008456 struct alc_spec *spec = codec->spec;
8457
8458 spec->autocfg.hp_pins[0] = 0x15;
8459 spec->autocfg.speaker_pins[0] = 0x14;
Takashi Iwai4f5d17062009-08-11 18:17:46 +02008460}
8461
8462static void alc883_clevo_m720_init_hook(struct hda_codec *codec)
8463{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008464 alc_automute_amp(codec);
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01008465 alc883_clevo_m720_mic_automute(codec);
8466}
8467
8468static void alc883_clevo_m720_unsol_event(struct hda_codec *codec,
Jiang zhe368c7a92008-03-04 11:20:33 +01008469 unsigned int res)
8470{
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01008471 switch (res >> 26) {
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01008472 case ALC880_MIC_EVENT:
8473 alc883_clevo_m720_mic_automute(codec);
8474 break;
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008475 default:
8476 alc_automute_amp_unsol_event(codec, res);
8477 break;
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01008478 }
Jiang zhe368c7a92008-03-04 11:20:33 +01008479}
8480
Jiang zhefb97dc62008-03-06 11:07:11 +01008481/* toggle speaker-output according to the hp-jack state */
Takashi Iwai4f5d17062009-08-11 18:17:46 +02008482static void alc883_2ch_fujitsu_pi2515_setup(struct hda_codec *codec)
Jiang zhefb97dc62008-03-06 11:07:11 +01008483{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008484 struct alc_spec *spec = codec->spec;
Jiang zhefb97dc62008-03-06 11:07:11 +01008485
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008486 spec->autocfg.hp_pins[0] = 0x14;
8487 spec->autocfg.speaker_pins[0] = 0x15;
Jiang zhefb97dc62008-03-06 11:07:11 +01008488}
8489
Takashi Iwai4f5d17062009-08-11 18:17:46 +02008490static void alc883_haier_w66_setup(struct hda_codec *codec)
Jiang zhefb97dc62008-03-06 11:07:11 +01008491{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008492 struct alc_spec *spec = codec->spec;
Jiang zhefb97dc62008-03-06 11:07:11 +01008493
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008494 spec->autocfg.hp_pins[0] = 0x1b;
8495 spec->autocfg.speaker_pins[0] = 0x14;
Kailang Yang189609a2007-08-20 11:31:23 +02008496}
8497
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008498static void alc883_lenovo_101e_ispeaker_automute(struct hda_codec *codec)
8499{
Wu Fengguang864f92b2009-11-18 12:38:02 +08008500 int bits = snd_hda_jack_detect(codec, 0x14) ? HDA_AMP_MUTE : 0;
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008501
Takashi Iwai47fd8302007-08-10 17:11:07 +02008502 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
8503 HDA_AMP_MUTE, bits);
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008504}
8505
8506static void alc883_lenovo_101e_all_automute(struct hda_codec *codec)
8507{
Wu Fengguang864f92b2009-11-18 12:38:02 +08008508 int bits = snd_hda_jack_detect(codec, 0x1b) ? HDA_AMP_MUTE : 0;
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008509
Takashi Iwai47fd8302007-08-10 17:11:07 +02008510 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
8511 HDA_AMP_MUTE, bits);
8512 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
8513 HDA_AMP_MUTE, bits);
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008514}
8515
8516static void alc883_lenovo_101e_unsol_event(struct hda_codec *codec,
8517 unsigned int res)
8518{
8519 if ((res >> 26) == ALC880_HP_EVENT)
8520 alc883_lenovo_101e_all_automute(codec);
8521 if ((res >> 26) == ALC880_FRONT_EVENT)
8522 alc883_lenovo_101e_ispeaker_automute(codec);
8523}
8524
Takashi Iwai676a9b52007-08-16 15:23:35 +02008525/* toggle speaker-output according to the hp-jack state */
Takashi Iwai4f5d17062009-08-11 18:17:46 +02008526static void alc883_acer_aspire_setup(struct hda_codec *codec)
Takashi Iwai676a9b52007-08-16 15:23:35 +02008527{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008528 struct alc_spec *spec = codec->spec;
Kailang Yangea1fb292008-08-26 12:58:38 +02008529
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008530 spec->autocfg.hp_pins[0] = 0x14;
8531 spec->autocfg.speaker_pins[0] = 0x15;
8532 spec->autocfg.speaker_pins[1] = 0x16;
Takashi Iwai676a9b52007-08-16 15:23:35 +02008533}
8534
Kailang Yangd1a991a2007-08-15 16:21:59 +02008535static struct hda_verb alc883_acer_eapd_verbs[] = {
8536 /* HP Pin: output 0 (0x0c) */
8537 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
8538 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
8539 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
8540 /* Front Pin: output 0 (0x0c) */
Takashi Iwai676a9b52007-08-16 15:23:35 +02008541 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
8542 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
Kailang Yangd1a991a2007-08-15 16:21:59 +02008543 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Kailang Yangd1a991a2007-08-15 16:21:59 +02008544 {0x16, AC_VERB_SET_CONNECT_SEL, 0x00},
8545 /* eanable EAPD on medion laptop */
8546 {0x20, AC_VERB_SET_COEF_INDEX, 0x07},
8547 {0x20, AC_VERB_SET_PROC_COEF, 0x3050},
Takashi Iwai676a9b52007-08-16 15:23:35 +02008548 /* enable unsolicited event */
8549 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
Kailang Yangd1a991a2007-08-15 16:21:59 +02008550 { }
8551};
8552
Denis Kuplyakovfc86f952009-08-25 18:15:59 +02008553static struct hda_verb alc888_acer_aspire_7730G_verbs[] = {
8554 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
8555 {0x17, AC_VERB_SET_CONNECT_SEL, 0x02},
8556 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
8557 { } /* end */
8558};
8559
Takashi Iwai4f5d17062009-08-11 18:17:46 +02008560static void alc888_6st_dell_setup(struct hda_codec *codec)
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01008561{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008562 struct alc_spec *spec = codec->spec;
Kailang Yangea1fb292008-08-26 12:58:38 +02008563
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008564 spec->autocfg.hp_pins[0] = 0x1b;
8565 spec->autocfg.speaker_pins[0] = 0x14;
8566 spec->autocfg.speaker_pins[1] = 0x15;
8567 spec->autocfg.speaker_pins[2] = 0x16;
8568 spec->autocfg.speaker_pins[3] = 0x17;
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01008569}
8570
Takashi Iwai4f5d17062009-08-11 18:17:46 +02008571static void alc888_lenovo_sky_setup(struct hda_codec *codec)
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01008572{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008573 struct alc_spec *spec = codec->spec;
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01008574
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008575 spec->autocfg.hp_pins[0] = 0x1b;
8576 spec->autocfg.speaker_pins[0] = 0x14;
8577 spec->autocfg.speaker_pins[1] = 0x15;
8578 spec->autocfg.speaker_pins[2] = 0x16;
8579 spec->autocfg.speaker_pins[3] = 0x17;
8580 spec->autocfg.speaker_pins[4] = 0x1a;
Kailang Yange2757d52008-08-26 13:17:46 +02008581}
8582
Takashi Iwai4f5d17062009-08-11 18:17:46 +02008583static void alc883_vaiott_setup(struct hda_codec *codec)
Guido Günther3e1647c2009-06-05 00:47:26 +02008584{
8585 struct alc_spec *spec = codec->spec;
8586
8587 spec->autocfg.hp_pins[0] = 0x15;
8588 spec->autocfg.speaker_pins[0] = 0x14;
8589 spec->autocfg.speaker_pins[1] = 0x17;
Guido Günther3e1647c2009-06-05 00:47:26 +02008590}
8591
Kailang Yange2757d52008-08-26 13:17:46 +02008592static struct hda_verb alc888_asus_m90v_verbs[] = {
8593 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
8594 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
8595 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
8596 /* enable unsolicited event */
8597 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
8598 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_MIC_EVENT | AC_USRSP_EN},
8599 { } /* end */
8600};
8601
Takashi Iwai4f5d17062009-08-11 18:17:46 +02008602static void alc883_mode2_setup(struct hda_codec *codec)
Kailang Yange2757d52008-08-26 13:17:46 +02008603{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008604 struct alc_spec *spec = codec->spec;
Kailang Yange2757d52008-08-26 13:17:46 +02008605
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008606 spec->autocfg.hp_pins[0] = 0x1b;
8607 spec->autocfg.speaker_pins[0] = 0x14;
8608 spec->autocfg.speaker_pins[1] = 0x15;
8609 spec->autocfg.speaker_pins[2] = 0x16;
Takashi Iwai4f5d17062009-08-11 18:17:46 +02008610 spec->ext_mic.pin = 0x18;
8611 spec->int_mic.pin = 0x19;
8612 spec->ext_mic.mux_idx = 0;
8613 spec->int_mic.mux_idx = 1;
8614 spec->auto_mic = 1;
Kailang Yange2757d52008-08-26 13:17:46 +02008615}
8616
8617static struct hda_verb alc888_asus_eee1601_verbs[] = {
8618 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
8619 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
8620 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
8621 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
8622 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
8623 {0x20, AC_VERB_SET_COEF_INDEX, 0x0b},
8624 {0x20, AC_VERB_SET_PROC_COEF, 0x0838},
8625 /* enable unsolicited event */
8626 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
8627 { } /* end */
8628};
8629
Kailang Yange2757d52008-08-26 13:17:46 +02008630static void alc883_eee1601_inithook(struct hda_codec *codec)
8631{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02008632 struct alc_spec *spec = codec->spec;
8633
8634 spec->autocfg.hp_pins[0] = 0x14;
8635 spec->autocfg.speaker_pins[0] = 0x1b;
8636 alc_automute_pin(codec);
Kailang Yange2757d52008-08-26 13:17:46 +02008637}
8638
Torben Schulzeb4c41d2009-05-18 15:02:35 +02008639static struct hda_verb alc889A_mb31_verbs[] = {
8640 /* Init rear pin (used as headphone output) */
8641 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc4}, /* Apple Headphones */
8642 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00}, /* Connect to front */
8643 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
8644 /* Init line pin (used as output in 4ch and 6ch mode) */
8645 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x02}, /* Connect to CLFE */
8646 /* Init line 2 pin (used as headphone out by default) */
8647 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN}, /* Use as input */
8648 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE}, /* Mute output */
8649 { } /* end */
8650};
8651
8652/* Mute speakers according to the headphone jack state */
8653static void alc889A_mb31_automute(struct hda_codec *codec)
8654{
8655 unsigned int present;
8656
8657 /* Mute only in 2ch or 4ch mode */
8658 if (snd_hda_codec_read(codec, 0x15, 0, AC_VERB_GET_CONNECT_SEL, 0)
8659 == 0x00) {
Wu Fengguang864f92b2009-11-18 12:38:02 +08008660 present = snd_hda_jack_detect(codec, 0x15);
Torben Schulzeb4c41d2009-05-18 15:02:35 +02008661 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
8662 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
8663 snd_hda_codec_amp_stereo(codec, 0x16, HDA_OUTPUT, 0,
8664 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
8665 }
8666}
8667
8668static void alc889A_mb31_unsol_event(struct hda_codec *codec, unsigned int res)
8669{
8670 if ((res >> 26) == ALC880_HP_EVENT)
8671 alc889A_mb31_automute(codec);
8672}
8673
Takashi Iwai49535502009-06-30 15:28:30 +02008674
Takashi Iwaicb53c622007-08-10 17:21:45 +02008675#ifdef CONFIG_SND_HDA_POWER_SAVE
Takashi Iwai49535502009-06-30 15:28:30 +02008676#define alc882_loopbacks alc880_loopbacks
Takashi Iwaicb53c622007-08-10 17:21:45 +02008677#endif
8678
Sasha Alexandrdef319f2009-06-16 16:00:15 -04008679/* pcm configuration: identical with ALC880 */
Takashi Iwai49535502009-06-30 15:28:30 +02008680#define alc882_pcm_analog_playback alc880_pcm_analog_playback
8681#define alc882_pcm_analog_capture alc880_pcm_analog_capture
8682#define alc882_pcm_digital_playback alc880_pcm_digital_playback
8683#define alc882_pcm_digital_capture alc880_pcm_digital_capture
8684
8685static hda_nid_t alc883_slave_dig_outs[] = {
8686 ALC1200_DIGOUT_NID, 0,
8687};
8688
8689static hda_nid_t alc1200_slave_dig_outs[] = {
8690 ALC883_DIGOUT_NID, 0,
8691};
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008692
8693/*
8694 * configuration and preset
8695 */
Takashi Iwai49535502009-06-30 15:28:30 +02008696static const char *alc882_models[ALC882_MODEL_LAST] = {
8697 [ALC882_3ST_DIG] = "3stack-dig",
8698 [ALC882_6ST_DIG] = "6stack-dig",
8699 [ALC882_ARIMA] = "arima",
8700 [ALC882_W2JC] = "w2jc",
8701 [ALC882_TARGA] = "targa",
8702 [ALC882_ASUS_A7J] = "asus-a7j",
8703 [ALC882_ASUS_A7M] = "asus-a7m",
8704 [ALC885_MACPRO] = "macpro",
8705 [ALC885_MB5] = "mb5",
8706 [ALC885_MBP3] = "mbp3",
8707 [ALC885_IMAC24] = "imac24",
8708 [ALC883_3ST_2ch_DIG] = "3stack-2ch-dig",
Takashi Iwaif5fcc132006-11-24 17:07:44 +01008709 [ALC883_3ST_6ch_DIG] = "3stack-6ch-dig",
8710 [ALC883_3ST_6ch] = "3stack-6ch",
Takashi Iwai49535502009-06-30 15:28:30 +02008711 [ALC883_6ST_DIG] = "alc883-6stack-dig",
Takashi Iwaif5fcc132006-11-24 17:07:44 +01008712 [ALC883_TARGA_DIG] = "targa-dig",
8713 [ALC883_TARGA_2ch_DIG] = "targa-2ch-dig",
David Heidelberger64a8be72009-06-08 16:15:18 +02008714 [ALC883_TARGA_8ch_DIG] = "targa-8ch-dig",
Takashi Iwaif5fcc132006-11-24 17:07:44 +01008715 [ALC883_ACER] = "acer",
Tobin Davis2880a862007-08-07 11:50:26 +02008716 [ALC883_ACER_ASPIRE] = "acer-aspire",
Vincent Petry5b2d1ec2008-11-18 22:21:57 +08008717 [ALC888_ACER_ASPIRE_4930G] = "acer-aspire-4930g",
Takashi Iwaib1a91462009-06-21 10:56:44 +02008718 [ALC888_ACER_ASPIRE_6530G] = "acer-aspire-6530g",
Hector Martin3b315d72009-06-02 10:54:19 +02008719 [ALC888_ACER_ASPIRE_8930G] = "acer-aspire-8930g",
Denis Kuplyakovfc86f952009-08-25 18:15:59 +02008720 [ALC888_ACER_ASPIRE_7730G] = "acer-aspire-7730g",
Takashi Iwaif5fcc132006-11-24 17:07:44 +01008721 [ALC883_MEDION] = "medion",
Kailang Yang272a5272007-05-14 11:00:38 +02008722 [ALC883_MEDION_MD2] = "medion-md2",
Takashi Iwaif5fcc132006-11-24 17:07:44 +01008723 [ALC883_LAPTOP_EAPD] = "laptop-eapd",
Kailang Yangbc9f98a2007-04-12 13:06:07 +02008724 [ALC883_LENOVO_101E_2ch] = "lenovo-101e",
Kailang Yang272a5272007-05-14 11:00:38 +02008725 [ALC883_LENOVO_NB0763] = "lenovo-nb0763",
8726 [ALC888_LENOVO_MS7195_DIG] = "lenovo-ms7195-dig",
Kailang Yange2757d52008-08-26 13:17:46 +02008727 [ALC888_LENOVO_SKY] = "lenovo-sky",
Kailang Yang189609a2007-08-20 11:31:23 +02008728 [ALC883_HAIER_W66] = "haier-w66",
Claudio Matsuoka4723c022007-07-13 14:36:19 +02008729 [ALC888_3ST_HP] = "3stack-hp",
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01008730 [ALC888_6ST_DELL] = "6stack-dell",
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +01008731 [ALC883_MITAC] = "mitac",
ddiaz@cenditel.gob.vea65cc602009-09-05 16:28:06 -04308732 [ALC883_CLEVO_M540R] = "clevo-m540r",
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01008733 [ALC883_CLEVO_M720] = "clevo-m720",
Jiang zhefb97dc62008-03-06 11:07:11 +01008734 [ALC883_FUJITSU_PI2515] = "fujitsu-pi2515",
Vincent Petryef8ef5f2008-11-23 11:31:41 +08008735 [ALC888_FUJITSU_XA3530] = "fujitsu-xa3530",
Jiang zhe17bba1b2008-06-04 12:11:07 +02008736 [ALC883_3ST_6ch_INTEL] = "3stack-6ch-intel",
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02008737 [ALC889A_INTEL] = "intel-alc889a",
8738 [ALC889_INTEL] = "intel-x58",
Wu Fengguang3ab90932008-11-17 09:51:09 +01008739 [ALC1200_ASUS_P5Q] = "asus-p5q",
Torben Schulzeb4c41d2009-05-18 15:02:35 +02008740 [ALC889A_MB31] = "mb31",
Guido Günther3e1647c2009-06-05 00:47:26 +02008741 [ALC883_SONY_VAIO_TT] = "sony-vaio-tt",
Takashi Iwai49535502009-06-30 15:28:30 +02008742 [ALC882_AUTO] = "auto",
Takashi Iwaif5fcc132006-11-24 17:07:44 +01008743};
8744
Takashi Iwai49535502009-06-30 15:28:30 +02008745static struct snd_pci_quirk alc882_cfg_tbl[] = {
8746 SND_PCI_QUIRK(0x1019, 0x6668, "ECS", ALC882_6ST_DIG),
8747
Takashi Iwaiac3e3742007-12-17 17:14:18 +01008748 SND_PCI_QUIRK(0x1025, 0x006c, "Acer Aspire 9810", ALC883_ACER_ASPIRE),
Takashi Iwai69e50282008-11-03 10:07:43 +01008749 SND_PCI_QUIRK(0x1025, 0x0090, "Acer Aspire", ALC883_ACER_ASPIRE),
Takashi Iwai9b6682f2009-03-23 22:50:52 +01008750 SND_PCI_QUIRK(0x1025, 0x010a, "Acer Ferrari 5000", ALC883_ACER_ASPIRE),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01008751 SND_PCI_QUIRK(0x1025, 0x0110, "Acer Aspire", ALC883_ACER_ASPIRE),
8752 SND_PCI_QUIRK(0x1025, 0x0112, "Acer Aspire 9303", ALC883_ACER_ASPIRE),
Jaroslav Kysela0b18cb12008-07-28 17:07:07 +02008753 SND_PCI_QUIRK(0x1025, 0x0121, "Acer Aspire 5920G", ALC883_ACER_ASPIRE),
Vincent Petry5b2d1ec2008-11-18 22:21:57 +08008754 SND_PCI_QUIRK(0x1025, 0x013e, "Acer Aspire 4930G",
8755 ALC888_ACER_ASPIRE_4930G),
Lukasz Wojnilowicza8e4f9d2009-01-08 12:00:49 +01008756 SND_PCI_QUIRK(0x1025, 0x013f, "Acer Aspire 5930G",
Takashi Iwaif2624792009-11-19 11:48:44 +01008757 ALC888_ACER_ASPIRE_6530G),
Hector Martin3b315d72009-06-02 10:54:19 +02008758 SND_PCI_QUIRK(0x1025, 0x0145, "Acer Aspire 8930G",
8759 ALC888_ACER_ASPIRE_8930G),
Takashi Iwaie46b0c82009-06-13 10:16:43 +02008760 SND_PCI_QUIRK(0x1025, 0x0146, "Acer Aspire 6935G",
8761 ALC888_ACER_ASPIRE_8930G),
Takashi Iwai49535502009-06-30 15:28:30 +02008762 SND_PCI_QUIRK(0x1025, 0x0157, "Acer X3200", ALC882_AUTO),
8763 SND_PCI_QUIRK(0x1025, 0x0158, "Acer AX1700-U3700A", ALC882_AUTO),
Lukasz Wojnilowicza8e4f9d2009-01-08 12:00:49 +01008764 SND_PCI_QUIRK(0x1025, 0x015e, "Acer Aspire 6930G",
Takashi Iwaidde65352009-06-25 08:25:35 +02008765 ALC888_ACER_ASPIRE_6530G),
Juan Jesus Garcia de Soriacc374c42009-02-23 08:11:59 +01008766 SND_PCI_QUIRK(0x1025, 0x0166, "Acer Aspire 6530G",
Tony Vroond2fd4b02009-06-21 00:40:10 +01008767 ALC888_ACER_ASPIRE_6530G),
Denis Kuplyakovfc86f952009-08-25 18:15:59 +02008768 SND_PCI_QUIRK(0x1025, 0x0142, "Acer Aspire 7730G",
8769 ALC888_ACER_ASPIRE_7730G),
Takashi Iwai22b530e2009-05-13 11:32:52 +02008770 /* default Acer -- disabled as it causes more problems.
8771 * model=auto should work fine now
8772 */
8773 /* SND_PCI_QUIRK_VENDOR(0x1025, "Acer laptop", ALC883_ACER), */
Takashi Iwai49535502009-06-30 15:28:30 +02008774
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01008775 SND_PCI_QUIRK(0x1028, 0x020d, "Dell Inspiron 530", ALC888_6ST_DELL),
Takashi Iwai49535502009-06-30 15:28:30 +02008776
Tobin Davisfebe3372007-06-12 11:27:46 +02008777 SND_PCI_QUIRK(0x103c, 0x2a3d, "HP Pavillion", ALC883_6ST_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01008778 SND_PCI_QUIRK(0x103c, 0x2a4f, "HP Samba", ALC888_3ST_HP),
8779 SND_PCI_QUIRK(0x103c, 0x2a60, "HP Lucknow", ALC888_3ST_HP),
Herton Ronaldo Krzesinski5d85f8d2008-03-20 12:13:46 +01008780 SND_PCI_QUIRK(0x103c, 0x2a61, "HP Nettle", ALC883_6ST_DIG),
Chris Bagwell06bf3e12009-01-01 10:32:08 +01008781 SND_PCI_QUIRK(0x103c, 0x2a66, "HP Acacia", ALC888_3ST_HP),
Herton Ronaldo Krzesinski7ec30f02009-03-04 14:22:52 -03008782 SND_PCI_QUIRK(0x103c, 0x2a72, "HP Educ.ar", ALC888_3ST_HP),
Takashi Iwai49535502009-06-30 15:28:30 +02008783
8784 SND_PCI_QUIRK(0x1043, 0x060d, "Asus A7J", ALC882_ASUS_A7J),
8785 SND_PCI_QUIRK(0x1043, 0x1243, "Asus A7J", ALC882_ASUS_A7J),
8786 SND_PCI_QUIRK(0x1043, 0x13c2, "Asus A7M", ALC882_ASUS_A7M),
Kailang Yanga01c30c2008-10-15 11:14:58 +02008787 SND_PCI_QUIRK(0x1043, 0x1873, "Asus M90V", ALC888_ASUS_M90V),
Takashi Iwai49535502009-06-30 15:28:30 +02008788 SND_PCI_QUIRK(0x1043, 0x1971, "Asus W2JC", ALC882_W2JC),
8789 SND_PCI_QUIRK(0x1043, 0x817f, "Asus P5LD2", ALC882_6ST_DIG),
8790 SND_PCI_QUIRK(0x1043, 0x81d8, "Asus P5WD", ALC882_6ST_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01008791 SND_PCI_QUIRK(0x1043, 0x8249, "Asus M2A-VM HDMI", ALC883_3ST_6ch_DIG),
Mackenzie Morgan44a678d2009-02-10 17:13:43 +01008792 SND_PCI_QUIRK(0x1043, 0x8284, "Asus Z37E", ALC883_6ST_DIG),
Wu Fengguang3ab90932008-11-17 09:51:09 +01008793 SND_PCI_QUIRK(0x1043, 0x82fe, "Asus P5Q-EM HDMI", ALC1200_ASUS_P5Q),
Kailang Yange2757d52008-08-26 13:17:46 +02008794 SND_PCI_QUIRK(0x1043, 0x835f, "Asus Eee 1601", ALC888_ASUS_EEE1601),
Takashi Iwai49535502009-06-30 15:28:30 +02008795
8796 SND_PCI_QUIRK(0x104d, 0x9047, "Sony Vaio TT", ALC883_SONY_VAIO_TT),
Travis Place97ec7102008-05-23 18:31:46 +02008797 SND_PCI_QUIRK(0x105b, 0x0ce8, "Foxconn P35AX-S", ALC883_6ST_DIG),
Takashi Iwai49535502009-06-30 15:28:30 +02008798 SND_PCI_QUIRK(0x105b, 0x6668, "Foxconn", ALC882_6ST_DIG),
Luke Yelavich2de686d2009-01-16 15:08:02 +11008799 SND_PCI_QUIRK(0x1071, 0x8227, "Mitac 82801H", ALC883_MITAC),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01008800 SND_PCI_QUIRK(0x1071, 0x8253, "Mitac 8252d", ALC883_MITAC),
8801 SND_PCI_QUIRK(0x1071, 0x8258, "Evesham Voyaeger", ALC883_LAPTOP_EAPD),
Kailang Yange2757d52008-08-26 13:17:46 +02008802 SND_PCI_QUIRK(0x10f1, 0x2350, "TYAN-S2350", ALC888_6ST_DELL),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01008803 SND_PCI_QUIRK(0x108e, 0x534d, NULL, ALC883_3ST_6ch),
Takashi Iwai49535502009-06-30 15:28:30 +02008804 SND_PCI_QUIRK(0x1458, 0xa002, "Gigabyte P35 DS3R", ALC882_6ST_DIG),
8805
Takashi Iwaiac3e3742007-12-17 17:14:18 +01008806 SND_PCI_QUIRK(0x1462, 0x0349, "MSI", ALC883_TARGA_2ch_DIG),
8807 SND_PCI_QUIRK(0x1462, 0x040d, "MSI", ALC883_TARGA_2ch_DIG),
8808 SND_PCI_QUIRK(0x1462, 0x0579, "MSI", ALC883_TARGA_2ch_DIG),
Takashi Iwai49535502009-06-30 15:28:30 +02008809 SND_PCI_QUIRK(0x1462, 0x28fb, "Targa T8", ALC882_TARGA), /* MSI-1049 T8 */
Jiang zhe4383fae2008-04-14 12:58:57 +02008810 SND_PCI_QUIRK(0x1462, 0x2fb3, "MSI", ALC883_TARGA_2ch_DIG),
Takashi Iwai49535502009-06-30 15:28:30 +02008811 SND_PCI_QUIRK(0x1462, 0x6668, "MSI", ALC882_6ST_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01008812 SND_PCI_QUIRK(0x1462, 0x3729, "MSI S420", ALC883_TARGA_DIG),
Pascal Terjan82808232008-03-04 11:33:28 +01008813 SND_PCI_QUIRK(0x1462, 0x3783, "NEC S970", ALC883_TARGA_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01008814 SND_PCI_QUIRK(0x1462, 0x3b7f, "MSI", ALC883_TARGA_2ch_DIG),
8815 SND_PCI_QUIRK(0x1462, 0x3ef9, "MSI", ALC883_TARGA_DIG),
8816 SND_PCI_QUIRK(0x1462, 0x3fc1, "MSI", ALC883_TARGA_DIG),
8817 SND_PCI_QUIRK(0x1462, 0x3fc3, "MSI", ALC883_TARGA_DIG),
8818 SND_PCI_QUIRK(0x1462, 0x3fcc, "MSI", ALC883_TARGA_DIG),
8819 SND_PCI_QUIRK(0x1462, 0x3fdf, "MSI", ALC883_TARGA_DIG),
William Westonb1e44222009-07-08 01:10:05 -07008820 SND_PCI_QUIRK(0x1462, 0x42cd, "MSI", ALC883_TARGA_DIG),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01008821 SND_PCI_QUIRK(0x1462, 0x4314, "MSI", ALC883_TARGA_DIG),
8822 SND_PCI_QUIRK(0x1462, 0x4319, "MSI", ALC883_TARGA_DIG),
8823 SND_PCI_QUIRK(0x1462, 0x4324, "MSI", ALC883_TARGA_DIG),
David Heidelberger64a8be72009-06-08 16:15:18 +02008824 SND_PCI_QUIRK(0x1462, 0x6510, "MSI GX620", ALC883_TARGA_8ch_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01008825 SND_PCI_QUIRK(0x1462, 0x6668, "MSI", ALC883_6ST_DIG),
8826 SND_PCI_QUIRK(0x1462, 0x7187, "MSI", ALC883_6ST_DIG),
Tobin Davis799f88a2007-05-29 14:29:08 +02008827 SND_PCI_QUIRK(0x1462, 0x7250, "MSI", ALC883_6ST_DIG),
Takashi Iwaiee095432008-11-25 15:03:38 +01008828 SND_PCI_QUIRK(0x1462, 0x7260, "MSI 7260", ALC883_TARGA_DIG),
Herton Ronaldo Krzesinski86d34b72008-03-18 09:27:59 +01008829 SND_PCI_QUIRK(0x1462, 0x7267, "MSI", ALC883_3ST_6ch_DIG),
Leonard Norrgard2a296cb2007-01-08 11:28:22 +01008830 SND_PCI_QUIRK(0x1462, 0x7280, "MSI", ALC883_6ST_DIG),
Tobin Davis2dcd5222007-07-10 17:04:57 +02008831 SND_PCI_QUIRK(0x1462, 0x7327, "MSI", ALC883_6ST_DIG),
Sasha Alexandrdf01b8a2009-06-16 14:46:17 -04008832 SND_PCI_QUIRK(0x1462, 0x7350, "MSI", ALC883_6ST_DIG),
Takashi Iwaif5fcc132006-11-24 17:07:44 +01008833 SND_PCI_QUIRK(0x1462, 0xa422, "MSI", ALC883_TARGA_2ch_DIG),
William Westonb1e44222009-07-08 01:10:05 -07008834 SND_PCI_QUIRK(0x1462, 0xaa08, "MSI", ALC883_TARGA_2ch_DIG),
Takashi Iwai49535502009-06-30 15:28:30 +02008835
Takashi Iwaiac3e3742007-12-17 17:14:18 +01008836 SND_PCI_QUIRK(0x147b, 0x1083, "Abit IP35-PRO", ALC883_6ST_DIG),
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01008837 SND_PCI_QUIRK(0x1558, 0x0721, "Clevo laptop M720R", ALC883_CLEVO_M720),
8838 SND_PCI_QUIRK(0x1558, 0x0722, "Clevo laptop M720SR", ALC883_CLEVO_M720),
ddiaz@cenditel.gob.vea65cc602009-09-05 16:28:06 -04308839 SND_PCI_QUIRK(0x1558, 0x5409, "Clevo laptop M540R", ALC883_CLEVO_M540R),
Takashi Iwaidea0a502009-02-09 17:14:52 +01008840 SND_PCI_QUIRK_VENDOR(0x1558, "Clevo laptop", ALC883_LAPTOP_EAPD),
Daniel T Chene60623b2007-05-29 03:41:55 -04008841 SND_PCI_QUIRK(0x15d9, 0x8780, "Supermicro PDSBA", ALC883_3ST_6ch),
Takashi Iwai49535502009-06-30 15:28:30 +02008842 /* SND_PCI_QUIRK(0x161f, 0x2054, "Arima W820", ALC882_ARIMA), */
Takashi Iwaif5fcc132006-11-24 17:07:44 +01008843 SND_PCI_QUIRK(0x161f, 0x2054, "Medion laptop", ALC883_MEDION),
Takashi Iwaibfb53032009-04-14 14:51:04 +02008844 SND_PCI_QUIRK_MASK(0x1734, 0xfff0, 0x1100, "FSC AMILO Xi/Pi25xx",
Takashi Iwaif67d8172009-02-04 23:30:19 +01008845 ALC883_FUJITSU_PI2515),
Takashi Iwaibfb53032009-04-14 14:51:04 +02008846 SND_PCI_QUIRK_MASK(0x1734, 0xfff0, 0x1130, "Fujitsu AMILO Xa35xx",
Vincent Petryef8ef5f2008-11-23 11:31:41 +08008847 ALC888_FUJITSU_XA3530),
Kailang Yang272a5272007-05-14 11:00:38 +02008848 SND_PCI_QUIRK(0x17aa, 0x101e, "Lenovo 101e", ALC883_LENOVO_101E_2ch),
Kailang Yang272a5272007-05-14 11:00:38 +02008849 SND_PCI_QUIRK(0x17aa, 0x2085, "Lenovo NB0763", ALC883_LENOVO_NB0763),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01008850 SND_PCI_QUIRK(0x17aa, 0x3bfc, "Lenovo NB0763", ALC883_LENOVO_NB0763),
8851 SND_PCI_QUIRK(0x17aa, 0x3bfd, "Lenovo NB0763", ALC883_LENOVO_NB0763),
Kailang Yange2757d52008-08-26 13:17:46 +02008852 SND_PCI_QUIRK(0x17aa, 0x101d, "Lenovo Sky", ALC888_LENOVO_SKY),
Kailang Yang272a5272007-05-14 11:00:38 +02008853 SND_PCI_QUIRK(0x17c0, 0x4071, "MEDION MD2", ALC883_MEDION_MD2),
Takashi Iwai959973b2008-11-05 11:30:56 +01008854 SND_PCI_QUIRK(0x17c0, 0x4085, "MEDION MD96630", ALC888_LENOVO_MS7195_DIG),
Andrew Paprocki0b167bf2008-02-03 10:15:44 +01008855 SND_PCI_QUIRK(0x17f2, 0x5000, "Albatron KI690-AM2", ALC883_6ST_DIG),
Kailang Yang189609a2007-08-20 11:31:23 +02008856 SND_PCI_QUIRK(0x1991, 0x5625, "Haier W66", ALC883_HAIER_W66),
Takashi Iwai49535502009-06-30 15:28:30 +02008857
Jiang zhe17bba1b2008-06-04 12:11:07 +02008858 SND_PCI_QUIRK(0x8086, 0x0001, "DG33BUC", ALC883_3ST_6ch_INTEL),
8859 SND_PCI_QUIRK(0x8086, 0x0002, "DG33FBC", ALC883_3ST_6ch_INTEL),
Luke Yelavich2de686d2009-01-16 15:08:02 +11008860 SND_PCI_QUIRK(0x8086, 0x2503, "82801H", ALC883_MITAC),
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02008861 SND_PCI_QUIRK(0x8086, 0x0022, "DX58SO", ALC889_INTEL),
8862 SND_PCI_QUIRK(0x8086, 0x0021, "Intel IbexPeak", ALC889A_INTEL),
8863 SND_PCI_QUIRK(0x8086, 0x3b56, "Intel IbexPeak", ALC889A_INTEL),
Takashi Iwaiac3e3742007-12-17 17:14:18 +01008864 SND_PCI_QUIRK(0x8086, 0xd601, "D102GGC", ALC883_3ST_6ch),
Takashi Iwai49535502009-06-30 15:28:30 +02008865
Takashi Iwai9c7f8522006-06-28 15:08:22 +02008866 {}
8867};
8868
Takashi Iwai49535502009-06-30 15:28:30 +02008869/* codec SSID table for Intel Mac */
8870static struct snd_pci_quirk alc882_ssid_cfg_tbl[] = {
8871 SND_PCI_QUIRK(0x106b, 0x00a0, "MacBookPro 3,1", ALC885_MBP3),
8872 SND_PCI_QUIRK(0x106b, 0x00a1, "Macbook", ALC885_MBP3),
8873 SND_PCI_QUIRK(0x106b, 0x00a4, "MacbookPro 4,1", ALC885_MBP3),
8874 SND_PCI_QUIRK(0x106b, 0x0c00, "Mac Pro", ALC885_MACPRO),
8875 SND_PCI_QUIRK(0x106b, 0x1000, "iMac 24", ALC885_IMAC24),
8876 SND_PCI_QUIRK(0x106b, 0x2800, "AppleTV", ALC885_IMAC24),
8877 SND_PCI_QUIRK(0x106b, 0x2c00, "MacbookPro rev3", ALC885_MBP3),
8878 SND_PCI_QUIRK(0x106b, 0x3600, "Macbook 3,1", ALC889A_MB31),
8879 SND_PCI_QUIRK(0x106b, 0x3800, "MacbookPro 4,1", ALC885_MBP3),
8880 SND_PCI_QUIRK(0x106b, 0x3e00, "iMac 24 Aluminum", ALC885_IMAC24),
8881 SND_PCI_QUIRK(0x106b, 0x3f00, "Macbook 5,1", ALC885_MB5),
Daniel T Chen46ef6ec2009-11-11 14:32:10 -05008882 /* FIXME: HP jack sense seems not working for MBP 5,1 or 5,2,
8883 * so apparently no perfect solution yet
Takashi Iwai49535502009-06-30 15:28:30 +02008884 */
8885 SND_PCI_QUIRK(0x106b, 0x4000, "MacbookPro 5,1", ALC885_MB5),
Daniel T Chen46ef6ec2009-11-11 14:32:10 -05008886 SND_PCI_QUIRK(0x106b, 0x4600, "MacbookPro 5,2", ALC885_MB5),
Takashi Iwai49535502009-06-30 15:28:30 +02008887 {} /* terminator */
Wu Fengguangf3cd3f52009-04-02 19:44:18 +08008888};
8889
Takashi Iwai49535502009-06-30 15:28:30 +02008890static struct alc_config_preset alc882_presets[] = {
8891 [ALC882_3ST_DIG] = {
8892 .mixers = { alc882_base_mixer },
Takashi Iwai8ab9e0a2009-07-03 10:58:12 +02008893 .init_verbs = { alc882_base_init_verbs,
8894 alc882_adc1_init_verbs },
Takashi Iwai49535502009-06-30 15:28:30 +02008895 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
8896 .dac_nids = alc882_dac_nids,
8897 .dig_out_nid = ALC882_DIGOUT_NID,
8898 .dig_in_nid = ALC882_DIGIN_NID,
8899 .num_channel_mode = ARRAY_SIZE(alc882_ch_modes),
8900 .channel_mode = alc882_ch_modes,
8901 .need_dac_fix = 1,
8902 .input_mux = &alc882_capture_source,
8903 },
8904 [ALC882_6ST_DIG] = {
8905 .mixers = { alc882_base_mixer, alc882_chmode_mixer },
Takashi Iwai8ab9e0a2009-07-03 10:58:12 +02008906 .init_verbs = { alc882_base_init_verbs,
8907 alc882_adc1_init_verbs },
Takashi Iwai49535502009-06-30 15:28:30 +02008908 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
8909 .dac_nids = alc882_dac_nids,
8910 .dig_out_nid = ALC882_DIGOUT_NID,
8911 .dig_in_nid = ALC882_DIGIN_NID,
8912 .num_channel_mode = ARRAY_SIZE(alc882_sixstack_modes),
8913 .channel_mode = alc882_sixstack_modes,
8914 .input_mux = &alc882_capture_source,
8915 },
8916 [ALC882_ARIMA] = {
8917 .mixers = { alc882_base_mixer, alc882_chmode_mixer },
Takashi Iwai8ab9e0a2009-07-03 10:58:12 +02008918 .init_verbs = { alc882_base_init_verbs, alc882_adc1_init_verbs,
8919 alc882_eapd_verbs },
Takashi Iwai49535502009-06-30 15:28:30 +02008920 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
8921 .dac_nids = alc882_dac_nids,
8922 .num_channel_mode = ARRAY_SIZE(alc882_sixstack_modes),
8923 .channel_mode = alc882_sixstack_modes,
8924 .input_mux = &alc882_capture_source,
8925 },
8926 [ALC882_W2JC] = {
8927 .mixers = { alc882_w2jc_mixer, alc882_chmode_mixer },
Takashi Iwai8ab9e0a2009-07-03 10:58:12 +02008928 .init_verbs = { alc882_base_init_verbs, alc882_adc1_init_verbs,
8929 alc882_eapd_verbs, alc880_gpio1_init_verbs },
Takashi Iwai49535502009-06-30 15:28:30 +02008930 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
8931 .dac_nids = alc882_dac_nids,
8932 .num_channel_mode = ARRAY_SIZE(alc880_threestack_modes),
8933 .channel_mode = alc880_threestack_modes,
8934 .need_dac_fix = 1,
8935 .input_mux = &alc882_capture_source,
8936 .dig_out_nid = ALC882_DIGOUT_NID,
8937 },
8938 [ALC885_MBP3] = {
8939 .mixers = { alc885_mbp3_mixer, alc882_chmode_mixer },
8940 .init_verbs = { alc885_mbp3_init_verbs,
8941 alc880_gpio1_init_verbs },
Takashi Iwaibe0ae922009-08-31 08:27:10 +02008942 .num_dacs = 2,
Takashi Iwai49535502009-06-30 15:28:30 +02008943 .dac_nids = alc882_dac_nids,
Takashi Iwaibe0ae922009-08-31 08:27:10 +02008944 .hp_nid = 0x04,
8945 .channel_mode = alc885_mbp_4ch_modes,
8946 .num_channel_mode = ARRAY_SIZE(alc885_mbp_4ch_modes),
Takashi Iwai49535502009-06-30 15:28:30 +02008947 .input_mux = &alc882_capture_source,
8948 .dig_out_nid = ALC882_DIGOUT_NID,
8949 .dig_in_nid = ALC882_DIGIN_NID,
8950 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02008951 .setup = alc885_mbp3_setup,
8952 .init_hook = alc_automute_amp,
Takashi Iwai49535502009-06-30 15:28:30 +02008953 },
8954 [ALC885_MB5] = {
8955 .mixers = { alc885_mb5_mixer, alc882_chmode_mixer },
8956 .init_verbs = { alc885_mb5_init_verbs,
8957 alc880_gpio1_init_verbs },
8958 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
8959 .dac_nids = alc882_dac_nids,
8960 .channel_mode = alc885_mb5_6ch_modes,
8961 .num_channel_mode = ARRAY_SIZE(alc885_mb5_6ch_modes),
8962 .input_mux = &mb5_capture_source,
8963 .dig_out_nid = ALC882_DIGOUT_NID,
8964 .dig_in_nid = ALC882_DIGIN_NID,
8965 },
8966 [ALC885_MACPRO] = {
8967 .mixers = { alc882_macpro_mixer },
8968 .init_verbs = { alc882_macpro_init_verbs },
8969 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
8970 .dac_nids = alc882_dac_nids,
8971 .dig_out_nid = ALC882_DIGOUT_NID,
8972 .dig_in_nid = ALC882_DIGIN_NID,
8973 .num_channel_mode = ARRAY_SIZE(alc882_ch_modes),
8974 .channel_mode = alc882_ch_modes,
8975 .input_mux = &alc882_capture_source,
8976 .init_hook = alc885_macpro_init_hook,
8977 },
8978 [ALC885_IMAC24] = {
8979 .mixers = { alc885_imac24_mixer },
8980 .init_verbs = { alc885_imac24_init_verbs },
8981 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
8982 .dac_nids = alc882_dac_nids,
8983 .dig_out_nid = ALC882_DIGOUT_NID,
8984 .dig_in_nid = ALC882_DIGIN_NID,
8985 .num_channel_mode = ARRAY_SIZE(alc882_ch_modes),
8986 .channel_mode = alc882_ch_modes,
8987 .input_mux = &alc882_capture_source,
8988 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02008989 .setup = alc885_imac24_setup,
Takashi Iwai49535502009-06-30 15:28:30 +02008990 .init_hook = alc885_imac24_init_hook,
8991 },
8992 [ALC882_TARGA] = {
8993 .mixers = { alc882_targa_mixer, alc882_chmode_mixer },
Takashi Iwai8ab9e0a2009-07-03 10:58:12 +02008994 .init_verbs = { alc882_base_init_verbs, alc882_adc1_init_verbs,
Takashi Iwai31909b82009-07-10 12:33:48 +02008995 alc880_gpio3_init_verbs, alc882_targa_verbs},
Takashi Iwai49535502009-06-30 15:28:30 +02008996 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
8997 .dac_nids = alc882_dac_nids,
8998 .dig_out_nid = ALC882_DIGOUT_NID,
8999 .num_adc_nids = ARRAY_SIZE(alc882_adc_nids),
9000 .adc_nids = alc882_adc_nids,
9001 .capsrc_nids = alc882_capsrc_nids,
9002 .num_channel_mode = ARRAY_SIZE(alc882_3ST_6ch_modes),
9003 .channel_mode = alc882_3ST_6ch_modes,
9004 .need_dac_fix = 1,
9005 .input_mux = &alc882_capture_source,
9006 .unsol_event = alc882_targa_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009007 .setup = alc882_targa_setup,
9008 .init_hook = alc882_targa_automute,
Takashi Iwai49535502009-06-30 15:28:30 +02009009 },
9010 [ALC882_ASUS_A7J] = {
9011 .mixers = { alc882_asus_a7j_mixer, alc882_chmode_mixer },
Takashi Iwai8ab9e0a2009-07-03 10:58:12 +02009012 .init_verbs = { alc882_base_init_verbs, alc882_adc1_init_verbs,
9013 alc882_asus_a7j_verbs},
Takashi Iwai49535502009-06-30 15:28:30 +02009014 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
9015 .dac_nids = alc882_dac_nids,
9016 .dig_out_nid = ALC882_DIGOUT_NID,
9017 .num_adc_nids = ARRAY_SIZE(alc882_adc_nids),
9018 .adc_nids = alc882_adc_nids,
9019 .capsrc_nids = alc882_capsrc_nids,
9020 .num_channel_mode = ARRAY_SIZE(alc882_3ST_6ch_modes),
9021 .channel_mode = alc882_3ST_6ch_modes,
9022 .need_dac_fix = 1,
9023 .input_mux = &alc882_capture_source,
9024 },
9025 [ALC882_ASUS_A7M] = {
9026 .mixers = { alc882_asus_a7m_mixer, alc882_chmode_mixer },
Takashi Iwai8ab9e0a2009-07-03 10:58:12 +02009027 .init_verbs = { alc882_base_init_verbs, alc882_adc1_init_verbs,
9028 alc882_eapd_verbs, alc880_gpio1_init_verbs,
Takashi Iwai49535502009-06-30 15:28:30 +02009029 alc882_asus_a7m_verbs },
9030 .num_dacs = ARRAY_SIZE(alc882_dac_nids),
9031 .dac_nids = alc882_dac_nids,
9032 .dig_out_nid = ALC882_DIGOUT_NID,
9033 .num_channel_mode = ARRAY_SIZE(alc880_threestack_modes),
9034 .channel_mode = alc880_threestack_modes,
9035 .need_dac_fix = 1,
9036 .input_mux = &alc882_capture_source,
9037 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009038 [ALC883_3ST_2ch_DIG] = {
9039 .mixers = { alc883_3ST_2ch_mixer },
9040 .init_verbs = { alc883_init_verbs },
9041 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9042 .dac_nids = alc883_dac_nids,
9043 .dig_out_nid = ALC883_DIGOUT_NID,
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009044 .dig_in_nid = ALC883_DIGIN_NID,
9045 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
9046 .channel_mode = alc883_3ST_2ch_modes,
9047 .input_mux = &alc883_capture_source,
9048 },
9049 [ALC883_3ST_6ch_DIG] = {
9050 .mixers = { alc883_3ST_6ch_mixer, alc883_chmode_mixer },
9051 .init_verbs = { alc883_init_verbs },
9052 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9053 .dac_nids = alc883_dac_nids,
9054 .dig_out_nid = ALC883_DIGOUT_NID,
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009055 .dig_in_nid = ALC883_DIGIN_NID,
9056 .num_channel_mode = ARRAY_SIZE(alc883_3ST_6ch_modes),
9057 .channel_mode = alc883_3ST_6ch_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02009058 .need_dac_fix = 1,
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009059 .input_mux = &alc883_capture_source,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009060 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009061 [ALC883_3ST_6ch] = {
9062 .mixers = { alc883_3ST_6ch_mixer, alc883_chmode_mixer },
9063 .init_verbs = { alc883_init_verbs },
9064 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9065 .dac_nids = alc883_dac_nids,
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009066 .num_channel_mode = ARRAY_SIZE(alc883_3ST_6ch_modes),
9067 .channel_mode = alc883_3ST_6ch_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +02009068 .need_dac_fix = 1,
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009069 .input_mux = &alc883_capture_source,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009070 },
Jiang zhe17bba1b2008-06-04 12:11:07 +02009071 [ALC883_3ST_6ch_INTEL] = {
9072 .mixers = { alc883_3ST_6ch_intel_mixer, alc883_chmode_mixer },
9073 .init_verbs = { alc883_init_verbs },
9074 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9075 .dac_nids = alc883_dac_nids,
9076 .dig_out_nid = ALC883_DIGOUT_NID,
9077 .dig_in_nid = ALC883_DIGIN_NID,
Wu Fengguangf3cd3f52009-04-02 19:44:18 +08009078 .slave_dig_outs = alc883_slave_dig_outs,
Jiang zhe17bba1b2008-06-04 12:11:07 +02009079 .num_channel_mode = ARRAY_SIZE(alc883_3ST_6ch_intel_modes),
9080 .channel_mode = alc883_3ST_6ch_intel_modes,
9081 .need_dac_fix = 1,
9082 .input_mux = &alc883_3stack_6ch_intel,
9083 },
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02009084 [ALC889A_INTEL] = {
9085 .mixers = { alc885_8ch_intel_mixer, alc883_chmode_mixer },
Wu Fengguang6732bd02009-07-30 09:19:14 +02009086 .init_verbs = { alc885_init_verbs, alc885_init_input_verbs,
9087 alc_hp15_unsol_verbs },
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02009088 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9089 .dac_nids = alc883_dac_nids,
9090 .num_adc_nids = ARRAY_SIZE(alc889_adc_nids),
9091 .adc_nids = alc889_adc_nids,
9092 .dig_out_nid = ALC883_DIGOUT_NID,
9093 .dig_in_nid = ALC883_DIGIN_NID,
9094 .slave_dig_outs = alc883_slave_dig_outs,
9095 .num_channel_mode = ARRAY_SIZE(alc889_8ch_intel_modes),
9096 .channel_mode = alc889_8ch_intel_modes,
9097 .capsrc_nids = alc889_capsrc_nids,
9098 .input_mux = &alc889_capture_source,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009099 .setup = alc889_automute_setup,
9100 .init_hook = alc_automute_amp,
Wu Fengguang6732bd02009-07-30 09:19:14 +02009101 .unsol_event = alc_automute_amp_unsol_event,
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02009102 .need_dac_fix = 1,
9103 },
9104 [ALC889_INTEL] = {
9105 .mixers = { alc885_8ch_intel_mixer, alc883_chmode_mixer },
9106 .init_verbs = { alc885_init_verbs, alc889_init_input_verbs,
Wu Fengguang6732bd02009-07-30 09:19:14 +02009107 alc889_eapd_verbs, alc_hp15_unsol_verbs},
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02009108 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9109 .dac_nids = alc883_dac_nids,
9110 .num_adc_nids = ARRAY_SIZE(alc889_adc_nids),
9111 .adc_nids = alc889_adc_nids,
9112 .dig_out_nid = ALC883_DIGOUT_NID,
9113 .dig_in_nid = ALC883_DIGIN_NID,
9114 .slave_dig_outs = alc883_slave_dig_outs,
9115 .num_channel_mode = ARRAY_SIZE(alc889_8ch_intel_modes),
9116 .channel_mode = alc889_8ch_intel_modes,
9117 .capsrc_nids = alc889_capsrc_nids,
9118 .input_mux = &alc889_capture_source,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009119 .setup = alc889_automute_setup,
Wu Fengguang6732bd02009-07-30 09:19:14 +02009120 .init_hook = alc889_intel_init_hook,
9121 .unsol_event = alc_automute_amp_unsol_event,
Jaroslav Kysela87a8c372009-07-23 10:58:29 +02009122 .need_dac_fix = 1,
9123 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009124 [ALC883_6ST_DIG] = {
9125 .mixers = { alc883_base_mixer, alc883_chmode_mixer },
9126 .init_verbs = { alc883_init_verbs },
9127 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9128 .dac_nids = alc883_dac_nids,
9129 .dig_out_nid = ALC883_DIGOUT_NID,
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009130 .dig_in_nid = ALC883_DIGIN_NID,
9131 .num_channel_mode = ARRAY_SIZE(alc883_sixstack_modes),
9132 .channel_mode = alc883_sixstack_modes,
9133 .input_mux = &alc883_capture_source,
9134 },
Kailang Yangccc656c2006-10-17 12:32:26 +02009135 [ALC883_TARGA_DIG] = {
Sasha Alexandrc2592492009-06-16 14:52:54 -04009136 .mixers = { alc883_targa_mixer, alc883_chmode_mixer },
David Heidelberger005b1072009-07-09 18:45:46 +02009137 .init_verbs = { alc883_init_verbs, alc880_gpio3_init_verbs,
9138 alc883_targa_verbs},
Kailang Yangccc656c2006-10-17 12:32:26 +02009139 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9140 .dac_nids = alc883_dac_nids,
9141 .dig_out_nid = ALC883_DIGOUT_NID,
Kailang Yangccc656c2006-10-17 12:32:26 +02009142 .num_channel_mode = ARRAY_SIZE(alc883_3ST_6ch_modes),
9143 .channel_mode = alc883_3ST_6ch_modes,
9144 .need_dac_fix = 1,
9145 .input_mux = &alc883_capture_source,
Sasha Alexandrc2592492009-06-16 14:52:54 -04009146 .unsol_event = alc883_targa_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009147 .setup = alc882_targa_setup,
9148 .init_hook = alc882_targa_automute,
Kailang Yangccc656c2006-10-17 12:32:26 +02009149 },
9150 [ALC883_TARGA_2ch_DIG] = {
Sasha Alexandrc2592492009-06-16 14:52:54 -04009151 .mixers = { alc883_targa_2ch_mixer},
David Heidelberger005b1072009-07-09 18:45:46 +02009152 .init_verbs = { alc883_init_verbs, alc880_gpio3_init_verbs,
9153 alc883_targa_verbs},
Kailang Yangccc656c2006-10-17 12:32:26 +02009154 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9155 .dac_nids = alc883_dac_nids,
Takashi Iwaif9e336f2008-10-31 16:37:07 +01009156 .adc_nids = alc883_adc_nids_alt,
9157 .num_adc_nids = ARRAY_SIZE(alc883_adc_nids_alt),
Kailang Yangccc656c2006-10-17 12:32:26 +02009158 .dig_out_nid = ALC883_DIGOUT_NID,
Kailang Yangccc656c2006-10-17 12:32:26 +02009159 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
9160 .channel_mode = alc883_3ST_2ch_modes,
9161 .input_mux = &alc883_capture_source,
Sasha Alexandrc2592492009-06-16 14:52:54 -04009162 .unsol_event = alc883_targa_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009163 .setup = alc882_targa_setup,
9164 .init_hook = alc882_targa_automute,
Kailang Yangccc656c2006-10-17 12:32:26 +02009165 },
David Heidelberger64a8be72009-06-08 16:15:18 +02009166 [ALC883_TARGA_8ch_DIG] = {
Takashi Iwaib99dba32009-09-17 18:23:00 +02009167 .mixers = { alc883_targa_mixer, alc883_targa_8ch_mixer,
9168 alc883_chmode_mixer },
David Heidelberger64a8be72009-06-08 16:15:18 +02009169 .init_verbs = { alc883_init_verbs, alc880_gpio3_init_verbs,
Sasha Alexandrc2592492009-06-16 14:52:54 -04009170 alc883_targa_verbs },
David Heidelberger64a8be72009-06-08 16:15:18 +02009171 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9172 .dac_nids = alc883_dac_nids,
9173 .num_adc_nids = ARRAY_SIZE(alc883_adc_nids_rev),
9174 .adc_nids = alc883_adc_nids_rev,
9175 .capsrc_nids = alc883_capsrc_nids_rev,
9176 .dig_out_nid = ALC883_DIGOUT_NID,
9177 .dig_in_nid = ALC883_DIGIN_NID,
9178 .num_channel_mode = ARRAY_SIZE(alc883_4ST_8ch_modes),
9179 .channel_mode = alc883_4ST_8ch_modes,
9180 .need_dac_fix = 1,
9181 .input_mux = &alc883_capture_source,
Sasha Alexandrc2592492009-06-16 14:52:54 -04009182 .unsol_event = alc883_targa_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009183 .setup = alc882_targa_setup,
9184 .init_hook = alc882_targa_automute,
David Heidelberger64a8be72009-06-08 16:15:18 +02009185 },
Vladimir Avdoninbab282b2006-08-22 13:31:58 +02009186 [ALC883_ACER] = {
Takashi Iwai676a9b52007-08-16 15:23:35 +02009187 .mixers = { alc883_base_mixer },
Vladimir Avdoninbab282b2006-08-22 13:31:58 +02009188 /* On TravelMate laptops, GPIO 0 enables the internal speaker
9189 * and the headphone jack. Turn this on and rely on the
9190 * standard mute methods whenever the user wants to turn
9191 * these outputs off.
9192 */
9193 .init_verbs = { alc883_init_verbs, alc880_gpio1_init_verbs },
9194 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9195 .dac_nids = alc883_dac_nids,
Vladimir Avdoninbab282b2006-08-22 13:31:58 +02009196 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
9197 .channel_mode = alc883_3ST_2ch_modes,
9198 .input_mux = &alc883_capture_source,
9199 },
Tobin Davis2880a862007-08-07 11:50:26 +02009200 [ALC883_ACER_ASPIRE] = {
Takashi Iwai676a9b52007-08-16 15:23:35 +02009201 .mixers = { alc883_acer_aspire_mixer },
Kailang Yangd1a991a2007-08-15 16:21:59 +02009202 .init_verbs = { alc883_init_verbs, alc883_acer_eapd_verbs },
Tobin Davis2880a862007-08-07 11:50:26 +02009203 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9204 .dac_nids = alc883_dac_nids,
9205 .dig_out_nid = ALC883_DIGOUT_NID,
Tobin Davis2880a862007-08-07 11:50:26 +02009206 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
9207 .channel_mode = alc883_3ST_2ch_modes,
9208 .input_mux = &alc883_capture_source,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02009209 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009210 .setup = alc883_acer_aspire_setup,
9211 .init_hook = alc_automute_amp,
Kailang Yangd1a991a2007-08-15 16:21:59 +02009212 },
Vincent Petry5b2d1ec2008-11-18 22:21:57 +08009213 [ALC888_ACER_ASPIRE_4930G] = {
Vincent Petryef8ef5f2008-11-23 11:31:41 +08009214 .mixers = { alc888_base_mixer,
Vincent Petry5b2d1ec2008-11-18 22:21:57 +08009215 alc883_chmode_mixer },
9216 .init_verbs = { alc883_init_verbs, alc880_gpio1_init_verbs,
9217 alc888_acer_aspire_4930g_verbs },
9218 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9219 .dac_nids = alc883_dac_nids,
9220 .num_adc_nids = ARRAY_SIZE(alc883_adc_nids_rev),
9221 .adc_nids = alc883_adc_nids_rev,
9222 .capsrc_nids = alc883_capsrc_nids_rev,
9223 .dig_out_nid = ALC883_DIGOUT_NID,
9224 .num_channel_mode = ARRAY_SIZE(alc883_3ST_6ch_modes),
9225 .channel_mode = alc883_3ST_6ch_modes,
9226 .need_dac_fix = 1,
9227 .num_mux_defs =
Vincent Petryef8ef5f2008-11-23 11:31:41 +08009228 ARRAY_SIZE(alc888_2_capture_sources),
9229 .input_mux = alc888_2_capture_sources,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02009230 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009231 .setup = alc888_acer_aspire_4930g_setup,
9232 .init_hook = alc_automute_amp,
Vincent Petry5b2d1ec2008-11-18 22:21:57 +08009233 },
Tony Vroond2fd4b02009-06-21 00:40:10 +01009234 [ALC888_ACER_ASPIRE_6530G] = {
9235 .mixers = { alc888_acer_aspire_6530_mixer },
9236 .init_verbs = { alc883_init_verbs, alc880_gpio1_init_verbs,
9237 alc888_acer_aspire_6530g_verbs },
9238 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9239 .dac_nids = alc883_dac_nids,
9240 .num_adc_nids = ARRAY_SIZE(alc883_adc_nids_rev),
9241 .adc_nids = alc883_adc_nids_rev,
9242 .capsrc_nids = alc883_capsrc_nids_rev,
9243 .dig_out_nid = ALC883_DIGOUT_NID,
9244 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
9245 .channel_mode = alc883_3ST_2ch_modes,
9246 .num_mux_defs =
9247 ARRAY_SIZE(alc888_2_capture_sources),
9248 .input_mux = alc888_acer_aspire_6530_sources,
9249 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009250 .setup = alc888_acer_aspire_6530g_setup,
9251 .init_hook = alc_automute_amp,
Tony Vroond2fd4b02009-06-21 00:40:10 +01009252 },
Hector Martin3b315d72009-06-02 10:54:19 +02009253 [ALC888_ACER_ASPIRE_8930G] = {
9254 .mixers = { alc888_base_mixer,
9255 alc883_chmode_mixer },
9256 .init_verbs = { alc883_init_verbs, alc880_gpio1_init_verbs,
Hector Martin018df412009-06-04 00:13:40 +02009257 alc889_acer_aspire_8930g_verbs },
Hector Martin3b315d72009-06-02 10:54:19 +02009258 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9259 .dac_nids = alc883_dac_nids,
Hector Martin018df412009-06-04 00:13:40 +02009260 .num_adc_nids = ARRAY_SIZE(alc889_adc_nids),
9261 .adc_nids = alc889_adc_nids,
9262 .capsrc_nids = alc889_capsrc_nids,
Hector Martin3b315d72009-06-02 10:54:19 +02009263 .dig_out_nid = ALC883_DIGOUT_NID,
9264 .num_channel_mode = ARRAY_SIZE(alc883_3ST_6ch_modes),
9265 .channel_mode = alc883_3ST_6ch_modes,
9266 .need_dac_fix = 1,
9267 .const_channel_count = 6,
9268 .num_mux_defs =
Hector Martin018df412009-06-04 00:13:40 +02009269 ARRAY_SIZE(alc889_capture_sources),
9270 .input_mux = alc889_capture_sources,
Hector Martin3b315d72009-06-02 10:54:19 +02009271 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009272 .setup = alc889_acer_aspire_8930g_setup,
9273 .init_hook = alc_automute_amp,
Hector Martin3b315d72009-06-02 10:54:19 +02009274 },
Denis Kuplyakovfc86f952009-08-25 18:15:59 +02009275 [ALC888_ACER_ASPIRE_7730G] = {
9276 .mixers = { alc883_3ST_6ch_mixer,
9277 alc883_chmode_mixer },
9278 .init_verbs = { alc883_init_verbs, alc880_gpio1_init_verbs,
9279 alc888_acer_aspire_7730G_verbs },
9280 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9281 .dac_nids = alc883_dac_nids,
9282 .num_adc_nids = ARRAY_SIZE(alc883_adc_nids_rev),
9283 .adc_nids = alc883_adc_nids_rev,
9284 .capsrc_nids = alc883_capsrc_nids_rev,
9285 .dig_out_nid = ALC883_DIGOUT_NID,
9286 .num_channel_mode = ARRAY_SIZE(alc883_3ST_6ch_modes),
9287 .channel_mode = alc883_3ST_6ch_modes,
9288 .need_dac_fix = 1,
9289 .const_channel_count = 6,
9290 .input_mux = &alc883_capture_source,
9291 .unsol_event = alc_automute_amp_unsol_event,
9292 .setup = alc888_acer_aspire_6530g_setup,
9293 .init_hook = alc_automute_amp,
9294 },
Tobin Davisc07584c2006-10-13 12:32:16 +02009295 [ALC883_MEDION] = {
9296 .mixers = { alc883_fivestack_mixer,
9297 alc883_chmode_mixer },
9298 .init_verbs = { alc883_init_verbs,
Andrew L. Neporadab373bde2006-11-07 11:37:08 +01009299 alc883_medion_eapd_verbs },
Tobin Davisc07584c2006-10-13 12:32:16 +02009300 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9301 .dac_nids = alc883_dac_nids,
Takashi Iwaif9e336f2008-10-31 16:37:07 +01009302 .adc_nids = alc883_adc_nids_alt,
9303 .num_adc_nids = ARRAY_SIZE(alc883_adc_nids_alt),
Tobin Davisc07584c2006-10-13 12:32:16 +02009304 .num_channel_mode = ARRAY_SIZE(alc883_sixstack_modes),
9305 .channel_mode = alc883_sixstack_modes,
9306 .input_mux = &alc883_capture_source,
Andrew L. Neporadab373bde2006-11-07 11:37:08 +01009307 },
Kailang Yang272a5272007-05-14 11:00:38 +02009308 [ALC883_MEDION_MD2] = {
9309 .mixers = { alc883_medion_md2_mixer},
9310 .init_verbs = { alc883_init_verbs, alc883_medion_md2_verbs},
9311 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9312 .dac_nids = alc883_dac_nids,
9313 .dig_out_nid = ALC883_DIGOUT_NID,
Kailang Yang272a5272007-05-14 11:00:38 +02009314 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
9315 .channel_mode = alc883_3ST_2ch_modes,
9316 .input_mux = &alc883_capture_source,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02009317 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009318 .setup = alc883_medion_md2_setup,
9319 .init_hook = alc_automute_amp,
Kailang Yangea1fb292008-08-26 12:58:38 +02009320 },
Andrew L. Neporadab373bde2006-11-07 11:37:08 +01009321 [ALC883_LAPTOP_EAPD] = {
Takashi Iwai676a9b52007-08-16 15:23:35 +02009322 .mixers = { alc883_base_mixer },
Andrew L. Neporadab373bde2006-11-07 11:37:08 +01009323 .init_verbs = { alc883_init_verbs, alc882_eapd_verbs },
9324 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9325 .dac_nids = alc883_dac_nids,
Andrew L. Neporadab373bde2006-11-07 11:37:08 +01009326 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
9327 .channel_mode = alc883_3ST_2ch_modes,
9328 .input_mux = &alc883_capture_source,
9329 },
ddiaz@cenditel.gob.vea65cc602009-09-05 16:28:06 -04309330 [ALC883_CLEVO_M540R] = {
9331 .mixers = { alc883_3ST_6ch_mixer, alc883_chmode_mixer },
9332 .init_verbs = { alc883_init_verbs, alc883_clevo_m540r_verbs },
9333 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9334 .dac_nids = alc883_dac_nids,
9335 .dig_out_nid = ALC883_DIGOUT_NID,
9336 .dig_in_nid = ALC883_DIGIN_NID,
9337 .num_channel_mode = ARRAY_SIZE(alc883_3ST_6ch_clevo_modes),
9338 .channel_mode = alc883_3ST_6ch_clevo_modes,
9339 .need_dac_fix = 1,
9340 .input_mux = &alc883_capture_source,
9341 /* This machine has the hardware HP auto-muting, thus
9342 * we need no software mute via unsol event
9343 */
9344 },
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01009345 [ALC883_CLEVO_M720] = {
9346 .mixers = { alc883_clevo_m720_mixer },
9347 .init_verbs = { alc883_init_verbs, alc883_clevo_m720_verbs },
Jiang zhe368c7a92008-03-04 11:20:33 +01009348 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9349 .dac_nids = alc883_dac_nids,
9350 .dig_out_nid = ALC883_DIGOUT_NID,
9351 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
9352 .channel_mode = alc883_3ST_2ch_modes,
9353 .input_mux = &alc883_capture_source,
Herton Ronaldo Krzesinski0c4cc442008-03-22 10:26:05 +01009354 .unsol_event = alc883_clevo_m720_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009355 .setup = alc883_clevo_m720_setup,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02009356 .init_hook = alc883_clevo_m720_init_hook,
Jiang zhe368c7a92008-03-04 11:20:33 +01009357 },
Kailang Yangbc9f98a2007-04-12 13:06:07 +02009358 [ALC883_LENOVO_101E_2ch] = {
9359 .mixers = { alc883_lenovo_101e_2ch_mixer},
9360 .init_verbs = { alc883_init_verbs, alc883_lenovo_101e_verbs},
9361 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9362 .dac_nids = alc883_dac_nids,
Takashi Iwaif9e336f2008-10-31 16:37:07 +01009363 .adc_nids = alc883_adc_nids_alt,
9364 .num_adc_nids = ARRAY_SIZE(alc883_adc_nids_alt),
Kailang Yangbc9f98a2007-04-12 13:06:07 +02009365 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
9366 .channel_mode = alc883_3ST_2ch_modes,
9367 .input_mux = &alc883_lenovo_101e_capture_source,
9368 .unsol_event = alc883_lenovo_101e_unsol_event,
9369 .init_hook = alc883_lenovo_101e_all_automute,
9370 },
Kailang Yang272a5272007-05-14 11:00:38 +02009371 [ALC883_LENOVO_NB0763] = {
9372 .mixers = { alc883_lenovo_nb0763_mixer },
9373 .init_verbs = { alc883_init_verbs, alc883_lenovo_nb0763_verbs},
9374 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9375 .dac_nids = alc883_dac_nids,
Kailang Yang272a5272007-05-14 11:00:38 +02009376 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
9377 .channel_mode = alc883_3ST_2ch_modes,
9378 .need_dac_fix = 1,
9379 .input_mux = &alc883_lenovo_nb0763_capture_source,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02009380 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009381 .setup = alc883_medion_md2_setup,
9382 .init_hook = alc_automute_amp,
Kailang Yang272a5272007-05-14 11:00:38 +02009383 },
9384 [ALC888_LENOVO_MS7195_DIG] = {
9385 .mixers = { alc883_3ST_6ch_mixer, alc883_chmode_mixer },
9386 .init_verbs = { alc883_init_verbs, alc888_lenovo_ms7195_verbs},
9387 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9388 .dac_nids = alc883_dac_nids,
9389 .dig_out_nid = ALC883_DIGOUT_NID,
Kailang Yang272a5272007-05-14 11:00:38 +02009390 .num_channel_mode = ARRAY_SIZE(alc883_3ST_6ch_modes),
9391 .channel_mode = alc883_3ST_6ch_modes,
9392 .need_dac_fix = 1,
9393 .input_mux = &alc883_capture_source,
9394 .unsol_event = alc883_lenovo_ms7195_unsol_event,
9395 .init_hook = alc888_lenovo_ms7195_front_automute,
Kailang Yang189609a2007-08-20 11:31:23 +02009396 },
9397 [ALC883_HAIER_W66] = {
Sasha Alexandrc2592492009-06-16 14:52:54 -04009398 .mixers = { alc883_targa_2ch_mixer},
Kailang Yang189609a2007-08-20 11:31:23 +02009399 .init_verbs = { alc883_init_verbs, alc883_haier_w66_verbs},
9400 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9401 .dac_nids = alc883_dac_nids,
9402 .dig_out_nid = ALC883_DIGOUT_NID,
Kailang Yang189609a2007-08-20 11:31:23 +02009403 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
9404 .channel_mode = alc883_3ST_2ch_modes,
9405 .input_mux = &alc883_capture_source,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02009406 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009407 .setup = alc883_haier_w66_setup,
9408 .init_hook = alc_automute_amp,
Herton Ronaldo Krzesinskieea64192008-03-20 12:14:59 +01009409 },
Claudio Matsuoka4723c022007-07-13 14:36:19 +02009410 [ALC888_3ST_HP] = {
Herton Ronaldo Krzesinskieea64192008-03-20 12:14:59 +01009411 .mixers = { alc883_3ST_6ch_mixer, alc883_chmode_mixer },
Claudio Matsuoka4723c022007-07-13 14:36:19 +02009412 .init_verbs = { alc883_init_verbs, alc888_3st_hp_verbs },
Claudio Matsuoka8341de62007-07-06 12:10:45 +02009413 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9414 .dac_nids = alc883_dac_nids,
Claudio Matsuoka4723c022007-07-13 14:36:19 +02009415 .num_channel_mode = ARRAY_SIZE(alc888_3st_hp_modes),
9416 .channel_mode = alc888_3st_hp_modes,
Claudio Matsuoka8341de62007-07-06 12:10:45 +02009417 .need_dac_fix = 1,
9418 .input_mux = &alc883_capture_source,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02009419 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009420 .setup = alc888_3st_hp_setup,
9421 .init_hook = alc_automute_amp,
Claudio Matsuoka8341de62007-07-06 12:10:45 +02009422 },
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01009423 [ALC888_6ST_DELL] = {
Herton Ronaldo Krzesinskif24dbdc2008-03-20 12:14:28 +01009424 .mixers = { alc883_base_mixer, alc883_chmode_mixer },
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01009425 .init_verbs = { alc883_init_verbs, alc888_6st_dell_verbs },
9426 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9427 .dac_nids = alc883_dac_nids,
9428 .dig_out_nid = ALC883_DIGOUT_NID,
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01009429 .dig_in_nid = ALC883_DIGIN_NID,
9430 .num_channel_mode = ARRAY_SIZE(alc883_sixstack_modes),
9431 .channel_mode = alc883_sixstack_modes,
9432 .input_mux = &alc883_capture_source,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02009433 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009434 .setup = alc888_6st_dell_setup,
9435 .init_hook = alc_automute_amp,
Claudio Matsuoka5795b9e2008-01-13 11:58:27 +01009436 },
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +01009437 [ALC883_MITAC] = {
9438 .mixers = { alc883_mitac_mixer },
9439 .init_verbs = { alc883_init_verbs, alc883_mitac_verbs },
9440 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9441 .dac_nids = alc883_dac_nids,
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +01009442 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
9443 .channel_mode = alc883_3ST_2ch_modes,
9444 .input_mux = &alc883_capture_source,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02009445 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009446 .setup = alc883_mitac_setup,
9447 .init_hook = alc_automute_amp,
Andy Shevchenkoa8848bd2007-12-13 17:32:26 +01009448 },
Jiang zhefb97dc62008-03-06 11:07:11 +01009449 [ALC883_FUJITSU_PI2515] = {
9450 .mixers = { alc883_2ch_fujitsu_pi2515_mixer },
9451 .init_verbs = { alc883_init_verbs,
9452 alc883_2ch_fujitsu_pi2515_verbs},
9453 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9454 .dac_nids = alc883_dac_nids,
9455 .dig_out_nid = ALC883_DIGOUT_NID,
9456 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
9457 .channel_mode = alc883_3ST_2ch_modes,
9458 .input_mux = &alc883_fujitsu_pi2515_capture_source,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02009459 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009460 .setup = alc883_2ch_fujitsu_pi2515_setup,
9461 .init_hook = alc_automute_amp,
Jiang zhefb97dc62008-03-06 11:07:11 +01009462 },
Vincent Petryef8ef5f2008-11-23 11:31:41 +08009463 [ALC888_FUJITSU_XA3530] = {
9464 .mixers = { alc888_base_mixer, alc883_chmode_mixer },
9465 .init_verbs = { alc883_init_verbs,
9466 alc888_fujitsu_xa3530_verbs },
9467 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9468 .dac_nids = alc883_dac_nids,
9469 .num_adc_nids = ARRAY_SIZE(alc883_adc_nids_rev),
9470 .adc_nids = alc883_adc_nids_rev,
9471 .capsrc_nids = alc883_capsrc_nids_rev,
9472 .dig_out_nid = ALC883_DIGOUT_NID,
9473 .num_channel_mode = ARRAY_SIZE(alc888_4ST_8ch_intel_modes),
9474 .channel_mode = alc888_4ST_8ch_intel_modes,
9475 .num_mux_defs =
9476 ARRAY_SIZE(alc888_2_capture_sources),
9477 .input_mux = alc888_2_capture_sources,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02009478 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009479 .setup = alc888_fujitsu_xa3530_setup,
9480 .init_hook = alc_automute_amp,
Vincent Petryef8ef5f2008-11-23 11:31:41 +08009481 },
Kailang Yange2757d52008-08-26 13:17:46 +02009482 [ALC888_LENOVO_SKY] = {
9483 .mixers = { alc888_lenovo_sky_mixer, alc883_chmode_mixer },
9484 .init_verbs = { alc883_init_verbs, alc888_lenovo_sky_verbs},
9485 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9486 .dac_nids = alc883_dac_nids,
9487 .dig_out_nid = ALC883_DIGOUT_NID,
Kailang Yange2757d52008-08-26 13:17:46 +02009488 .num_channel_mode = ARRAY_SIZE(alc883_sixstack_modes),
9489 .channel_mode = alc883_sixstack_modes,
9490 .need_dac_fix = 1,
9491 .input_mux = &alc883_lenovo_sky_capture_source,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02009492 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009493 .setup = alc888_lenovo_sky_setup,
9494 .init_hook = alc_automute_amp,
Kailang Yange2757d52008-08-26 13:17:46 +02009495 },
9496 [ALC888_ASUS_M90V] = {
9497 .mixers = { alc883_3ST_6ch_mixer, alc883_chmode_mixer },
9498 .init_verbs = { alc883_init_verbs, alc888_asus_m90v_verbs },
9499 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9500 .dac_nids = alc883_dac_nids,
9501 .dig_out_nid = ALC883_DIGOUT_NID,
9502 .dig_in_nid = ALC883_DIGIN_NID,
9503 .num_channel_mode = ARRAY_SIZE(alc883_3ST_6ch_modes),
9504 .channel_mode = alc883_3ST_6ch_modes,
9505 .need_dac_fix = 1,
9506 .input_mux = &alc883_fujitsu_pi2515_capture_source,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009507 .unsol_event = alc_sku_unsol_event,
9508 .setup = alc883_mode2_setup,
9509 .init_hook = alc_inithook,
Kailang Yange2757d52008-08-26 13:17:46 +02009510 },
9511 [ALC888_ASUS_EEE1601] = {
9512 .mixers = { alc883_asus_eee1601_mixer },
Takashi Iwaif9e336f2008-10-31 16:37:07 +01009513 .cap_mixer = alc883_asus_eee1601_cap_mixer,
Kailang Yange2757d52008-08-26 13:17:46 +02009514 .init_verbs = { alc883_init_verbs, alc888_asus_eee1601_verbs },
9515 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9516 .dac_nids = alc883_dac_nids,
9517 .dig_out_nid = ALC883_DIGOUT_NID,
9518 .dig_in_nid = ALC883_DIGIN_NID,
9519 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
9520 .channel_mode = alc883_3ST_2ch_modes,
9521 .need_dac_fix = 1,
9522 .input_mux = &alc883_asus_eee1601_capture_source,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +02009523 .unsol_event = alc_sku_unsol_event,
Kailang Yange2757d52008-08-26 13:17:46 +02009524 .init_hook = alc883_eee1601_inithook,
9525 },
Wu Fengguang3ab90932008-11-17 09:51:09 +01009526 [ALC1200_ASUS_P5Q] = {
9527 .mixers = { alc883_base_mixer, alc883_chmode_mixer },
9528 .init_verbs = { alc883_init_verbs },
9529 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9530 .dac_nids = alc883_dac_nids,
9531 .dig_out_nid = ALC1200_DIGOUT_NID,
9532 .dig_in_nid = ALC883_DIGIN_NID,
Wu Fengguangb25c9da2009-02-06 15:02:27 +08009533 .slave_dig_outs = alc1200_slave_dig_outs,
Wu Fengguang3ab90932008-11-17 09:51:09 +01009534 .num_channel_mode = ARRAY_SIZE(alc883_sixstack_modes),
9535 .channel_mode = alc883_sixstack_modes,
9536 .input_mux = &alc883_capture_source,
9537 },
Torben Schulzeb4c41d2009-05-18 15:02:35 +02009538 [ALC889A_MB31] = {
9539 .mixers = { alc889A_mb31_mixer, alc883_chmode_mixer},
9540 .init_verbs = { alc883_init_verbs, alc889A_mb31_verbs,
9541 alc880_gpio1_init_verbs },
9542 .adc_nids = alc883_adc_nids,
9543 .num_adc_nids = ARRAY_SIZE(alc883_adc_nids),
9544 .dac_nids = alc883_dac_nids,
9545 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9546 .channel_mode = alc889A_mb31_6ch_modes,
9547 .num_channel_mode = ARRAY_SIZE(alc889A_mb31_6ch_modes),
9548 .input_mux = &alc889A_mb31_capture_source,
9549 .dig_out_nid = ALC883_DIGOUT_NID,
9550 .unsol_event = alc889A_mb31_unsol_event,
9551 .init_hook = alc889A_mb31_automute,
9552 },
Guido Günther3e1647c2009-06-05 00:47:26 +02009553 [ALC883_SONY_VAIO_TT] = {
9554 .mixers = { alc883_vaiott_mixer },
9555 .init_verbs = { alc883_init_verbs, alc883_vaiott_verbs },
9556 .num_dacs = ARRAY_SIZE(alc883_dac_nids),
9557 .dac_nids = alc883_dac_nids,
9558 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
9559 .channel_mode = alc883_3ST_2ch_modes,
9560 .input_mux = &alc883_capture_source,
9561 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +02009562 .setup = alc883_vaiott_setup,
9563 .init_hook = alc_automute_amp,
Guido Günther3e1647c2009-06-05 00:47:26 +02009564 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009565};
9566
9567
9568/*
Takashi Iwai49535502009-06-30 15:28:30 +02009569 * Pin config fixes
9570 */
9571enum {
9572 PINFIX_ABIT_AW9D_MAX
9573};
9574
9575static struct alc_pincfg alc882_abit_aw9d_pinfix[] = {
9576 { 0x15, 0x01080104 }, /* side */
9577 { 0x16, 0x01011012 }, /* rear */
9578 { 0x17, 0x01016011 }, /* clfe */
9579 { }
9580};
9581
Takashi Iwaif8f25ba2009-10-06 08:31:29 +02009582static const struct alc_fixup alc882_fixups[] = {
9583 [PINFIX_ABIT_AW9D_MAX] = {
9584 .pins = alc882_abit_aw9d_pinfix
9585 },
Takashi Iwai49535502009-06-30 15:28:30 +02009586};
9587
Takashi Iwaif8f25ba2009-10-06 08:31:29 +02009588static struct snd_pci_quirk alc882_fixup_tbl[] = {
Takashi Iwai49535502009-06-30 15:28:30 +02009589 SND_PCI_QUIRK(0x147b, 0x107a, "Abit AW9D-MAX", PINFIX_ABIT_AW9D_MAX),
9590 {}
9591};
9592
9593/*
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009594 * BIOS auto configuration
9595 */
Takashi Iwai05f5f472009-08-25 13:10:18 +02009596static int alc882_auto_create_input_ctls(struct hda_codec *codec,
9597 const struct auto_pin_cfg *cfg)
9598{
9599 return alc_auto_create_input_ctls(codec, cfg, 0x0b, 0x23, 0x22);
9600}
9601
Takashi Iwai49535502009-06-30 15:28:30 +02009602static void alc882_auto_set_output_and_unmute(struct hda_codec *codec,
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009603 hda_nid_t nid, int pin_type,
9604 int dac_idx)
9605{
9606 /* set as output */
9607 struct alc_spec *spec = codec->spec;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009608 int idx;
9609
Takashi Iwaif6c7e542008-02-12 18:32:23 +01009610 alc_set_pin_output(codec, nid, pin_type);
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009611 if (spec->multiout.dac_nids[dac_idx] == 0x25)
9612 idx = 4;
9613 else
9614 idx = spec->multiout.dac_nids[dac_idx] - 2;
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009615 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_CONNECT_SEL, idx);
9616
9617}
9618
Takashi Iwai49535502009-06-30 15:28:30 +02009619static void alc882_auto_init_multi_out(struct hda_codec *codec)
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009620{
9621 struct alc_spec *spec = codec->spec;
9622 int i;
9623
9624 for (i = 0; i <= HDA_SIDE; i++) {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009625 hda_nid_t nid = spec->autocfg.line_out_pins[i];
Takashi Iwaibaba8ee2007-04-23 17:17:48 +02009626 int pin_type = get_pin_type(spec->autocfg.line_out_type);
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009627 if (nid)
Takashi Iwai49535502009-06-30 15:28:30 +02009628 alc882_auto_set_output_and_unmute(codec, nid, pin_type,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009629 i);
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009630 }
9631}
9632
Takashi Iwai49535502009-06-30 15:28:30 +02009633static void alc882_auto_init_hp_out(struct hda_codec *codec)
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009634{
9635 struct alc_spec *spec = codec->spec;
9636 hda_nid_t pin;
9637
Takashi Iwaieb06ed82006-09-20 17:10:27 +02009638 pin = spec->autocfg.hp_pins[0];
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009639 if (pin) /* connect to front */
9640 /* use dac 0 */
Takashi Iwai49535502009-06-30 15:28:30 +02009641 alc882_auto_set_output_and_unmute(codec, pin, PIN_HP, 0);
Takashi Iwaif6c7e542008-02-12 18:32:23 +01009642 pin = spec->autocfg.speaker_pins[0];
9643 if (pin)
Takashi Iwai49535502009-06-30 15:28:30 +02009644 alc882_auto_set_output_and_unmute(codec, pin, PIN_OUT, 0);
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009645}
9646
Takashi Iwai49535502009-06-30 15:28:30 +02009647static void alc882_auto_init_analog_input(struct hda_codec *codec)
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009648{
9649 struct alc_spec *spec = codec->spec;
9650 int i;
9651
9652 for (i = 0; i < AUTO_PIN_LAST; i++) {
9653 hda_nid_t nid = spec->autocfg.input_pins[i];
Takashi Iwai49535502009-06-30 15:28:30 +02009654 if (!nid)
9655 continue;
Takashi Iwai0d971c92009-06-30 16:11:11 +02009656 alc_set_input_pin(codec, nid, i);
Takashi Iwai49535502009-06-30 15:28:30 +02009657 if (get_wcaps(codec, nid) & AC_WCAP_OUT_AMP)
9658 snd_hda_codec_write(codec, nid, 0,
9659 AC_VERB_SET_AMP_GAIN_MUTE,
9660 AMP_OUT_MUTE);
9661 }
9662}
9663
9664static void alc882_auto_init_input_src(struct hda_codec *codec)
9665{
9666 struct alc_spec *spec = codec->spec;
9667 int c;
9668
9669 for (c = 0; c < spec->num_adc_nids; c++) {
9670 hda_nid_t conn_list[HDA_MAX_NUM_INPUTS];
9671 hda_nid_t nid = spec->capsrc_nids[c];
9672 unsigned int mux_idx;
9673 const struct hda_input_mux *imux;
9674 int conns, mute, idx, item;
9675
9676 conns = snd_hda_get_connections(codec, nid, conn_list,
9677 ARRAY_SIZE(conn_list));
9678 if (conns < 0)
9679 continue;
9680 mux_idx = c >= spec->num_mux_defs ? 0 : c;
9681 imux = &spec->input_mux[mux_idx];
9682 for (idx = 0; idx < conns; idx++) {
9683 /* if the current connection is the selected one,
9684 * unmute it as default - otherwise mute it
9685 */
9686 mute = AMP_IN_MUTE(idx);
9687 for (item = 0; item < imux->num_items; item++) {
9688 if (imux->items[item].index == idx) {
9689 if (spec->cur_mux[c] == item)
9690 mute = AMP_IN_UNMUTE(idx);
9691 break;
9692 }
9693 }
9694 /* check if we have a selector or mixer
9695 * we could check for the widget type instead, but
9696 * just check for Amp-In presence (in case of mixer
9697 * without amp-in there is something wrong, this
9698 * function shouldn't be used or capsrc nid is wrong)
9699 */
9700 if (get_wcaps(codec, nid) & AC_WCAP_IN_AMP)
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009701 snd_hda_codec_write(codec, nid, 0,
9702 AC_VERB_SET_AMP_GAIN_MUTE,
Takashi Iwai49535502009-06-30 15:28:30 +02009703 mute);
9704 else if (mute != AMP_IN_MUTE(idx))
9705 snd_hda_codec_write(codec, nid, 0,
9706 AC_VERB_SET_CONNECT_SEL,
9707 idx);
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009708 }
9709 }
9710}
9711
Takashi Iwai49535502009-06-30 15:28:30 +02009712/* add mic boosts if needed */
9713static int alc_auto_add_mic_boost(struct hda_codec *codec)
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009714{
9715 struct alc_spec *spec = codec->spec;
Takashi Iwai49535502009-06-30 15:28:30 +02009716 int err;
9717 hda_nid_t nid;
9718
9719 nid = spec->autocfg.input_pins[AUTO_PIN_MIC];
9720 if (nid && (get_wcaps(codec, nid) & AC_WCAP_IN_AMP)) {
9721 err = add_control(spec, ALC_CTL_WIDGET_VOL,
9722 "Mic Boost",
9723 HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_INPUT));
9724 if (err < 0)
9725 return err;
9726 }
9727 nid = spec->autocfg.input_pins[AUTO_PIN_FRONT_MIC];
9728 if (nid && (get_wcaps(codec, nid) & AC_WCAP_IN_AMP)) {
9729 err = add_control(spec, ALC_CTL_WIDGET_VOL,
9730 "Front Mic Boost",
9731 HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_INPUT));
9732 if (err < 0)
9733 return err;
9734 }
9735 return 0;
9736}
9737
9738/* almost identical with ALC880 parser... */
9739static int alc882_parse_auto_config(struct hda_codec *codec)
9740{
9741 struct alc_spec *spec = codec->spec;
Takashi Iwai05f5f472009-08-25 13:10:18 +02009742 static hda_nid_t alc882_ignore[] = { 0x1d, 0 };
9743 int i, err;
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009744
Takashi Iwai05f5f472009-08-25 13:10:18 +02009745 err = snd_hda_parse_pin_def_config(codec, &spec->autocfg,
9746 alc882_ignore);
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009747 if (err < 0)
9748 return err;
Takashi Iwai05f5f472009-08-25 13:10:18 +02009749 if (!spec->autocfg.line_outs)
9750 return 0; /* can't find valid BIOS pin config */
9751
9752 err = alc880_auto_fill_dac_nids(spec, &spec->autocfg);
9753 if (err < 0)
9754 return err;
9755 err = alc880_auto_create_multi_out_ctls(spec, &spec->autocfg);
9756 if (err < 0)
9757 return err;
9758 err = alc880_auto_create_extra_out(spec,
9759 spec->autocfg.speaker_pins[0],
9760 "Speaker");
9761 if (err < 0)
9762 return err;
9763 err = alc880_auto_create_extra_out(spec, spec->autocfg.hp_pins[0],
9764 "Headphone");
9765 if (err < 0)
9766 return err;
9767 err = alc882_auto_create_input_ctls(codec, &spec->autocfg);
9768 if (err < 0)
9769 return err;
9770
9771 spec->multiout.max_channels = spec->multiout.num_dacs * 2;
9772
9773 /* check multiple SPDIF-out (for recent codecs) */
9774 for (i = 0; i < spec->autocfg.dig_outs; i++) {
9775 hda_nid_t dig_nid;
9776 err = snd_hda_get_connections(codec,
9777 spec->autocfg.dig_out_pins[i],
9778 &dig_nid, 1);
9779 if (err < 0)
9780 continue;
9781 if (!i)
9782 spec->multiout.dig_out_nid = dig_nid;
9783 else {
9784 spec->multiout.slave_dig_outs = spec->slave_dig_outs;
Roel Kluin71121d9f2009-11-10 20:11:55 +01009785 if (i >= ARRAY_SIZE(spec->slave_dig_outs) - 1)
Takashi Iwai05f5f472009-08-25 13:10:18 +02009786 break;
Roel Kluin71121d9f2009-11-10 20:11:55 +01009787 spec->slave_dig_outs[i - 1] = dig_nid;
Takashi Iwai05f5f472009-08-25 13:10:18 +02009788 }
9789 }
9790 if (spec->autocfg.dig_in_pin)
9791 spec->dig_in_nid = ALC880_DIGIN_NID;
9792
9793 if (spec->kctls.list)
9794 add_mixer(spec, spec->kctls.list);
9795
9796 add_verb(spec, alc883_auto_init_verbs);
9797 /* if ADC 0x07 is available, initialize it, too */
9798 if (get_wcaps_type(get_wcaps(codec, 0x07)) == AC_WID_AUD_IN)
9799 add_verb(spec, alc882_adc1_init_verbs);
9800
9801 spec->num_mux_defs = 1;
9802 spec->input_mux = &spec->private_imux[0];
9803
9804 alc_ssid_check(codec, 0x15, 0x1b, 0x14);
Takashi Iwai776e1842007-08-29 15:07:11 +02009805
9806 err = alc_auto_add_mic_boost(codec);
9807 if (err < 0)
9808 return err;
9809
Takashi Iwai776e1842007-08-29 15:07:11 +02009810 return 1; /* config found */
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009811}
9812
9813/* additional initialization for auto-configuration model */
Takashi Iwai49535502009-06-30 15:28:30 +02009814static void alc882_auto_init(struct hda_codec *codec)
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009815{
Takashi Iwaif6c7e542008-02-12 18:32:23 +01009816 struct alc_spec *spec = codec->spec;
Takashi Iwai49535502009-06-30 15:28:30 +02009817 alc882_auto_init_multi_out(codec);
9818 alc882_auto_init_hp_out(codec);
9819 alc882_auto_init_analog_input(codec);
9820 alc882_auto_init_input_src(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +01009821 if (spec->unsol_event)
Kailang Yang7fb0d782008-10-15 11:12:35 +02009822 alc_inithook(codec);
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009823}
9824
Takashi Iwai49535502009-06-30 15:28:30 +02009825static int patch_alc882(struct hda_codec *codec)
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009826{
9827 struct alc_spec *spec;
9828 int err, board_config;
9829
9830 spec = kzalloc(sizeof(*spec), GFP_KERNEL);
9831 if (spec == NULL)
9832 return -ENOMEM;
9833
9834 codec->spec = spec;
9835
Takashi Iwai49535502009-06-30 15:28:30 +02009836 switch (codec->vendor_id) {
9837 case 0x10ec0882:
9838 case 0x10ec0885:
9839 break;
9840 default:
9841 /* ALC883 and variants */
9842 alc_fix_pll_init(codec, 0x20, 0x0a, 10);
9843 break;
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009844 }
9845
Takashi Iwai49535502009-06-30 15:28:30 +02009846 board_config = snd_hda_check_board_config(codec, ALC882_MODEL_LAST,
9847 alc882_models,
9848 alc882_cfg_tbl);
9849
9850 if (board_config < 0 || board_config >= ALC882_MODEL_LAST)
9851 board_config = snd_hda_check_board_codec_sid_config(codec,
9852 ALC882_MODEL_LAST, alc882_models, alc882_ssid_cfg_tbl);
9853
9854 if (board_config < 0 || board_config >= ALC882_MODEL_LAST) {
Takashi Iwai9a11f1a2009-07-28 16:01:20 +02009855 printk(KERN_INFO "hda_codec: %s: BIOS auto-probing.\n",
Takashi Iwai49535502009-06-30 15:28:30 +02009856 codec->chip_name);
9857 board_config = ALC882_AUTO;
9858 }
9859
Takashi Iwaif8f25ba2009-10-06 08:31:29 +02009860 alc_pick_fixup(codec, alc882_fixup_tbl, alc882_fixups);
Takashi Iwai49535502009-06-30 15:28:30 +02009861
9862 if (board_config == ALC882_AUTO) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009863 /* automatic parse from the BIOS config */
Takashi Iwai49535502009-06-30 15:28:30 +02009864 err = alc882_parse_auto_config(codec);
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009865 if (err < 0) {
9866 alc_free(codec);
9867 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +02009868 } else if (!err) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009869 printk(KERN_INFO
9870 "hda_codec: Cannot set up configuration "
9871 "from BIOS. Using base mode...\n");
Takashi Iwai49535502009-06-30 15:28:30 +02009872 board_config = ALC882_3ST_DIG;
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009873 }
9874 }
9875
Kusanagi Kouichi680cd532009-02-05 00:00:58 +09009876 err = snd_hda_attach_beep_device(codec, 0x1);
9877 if (err < 0) {
9878 alc_free(codec);
9879 return err;
9880 }
9881
Takashi Iwai49535502009-06-30 15:28:30 +02009882 if (board_config != ALC882_AUTO)
Takashi Iwaie9c364c2009-08-11 17:16:13 +02009883 setup_preset(codec, &alc882_presets[board_config]);
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009884
Takashi Iwai49535502009-06-30 15:28:30 +02009885 spec->stream_analog_playback = &alc882_pcm_analog_playback;
9886 spec->stream_analog_capture = &alc882_pcm_analog_capture;
9887 /* FIXME: setup DAC5 */
9888 /*spec->stream_analog_alt_playback = &alc880_pcm_analog_alt_playback;*/
9889 spec->stream_analog_alt_capture = &alc880_pcm_analog_alt_capture;
9890
9891 spec->stream_digital_playback = &alc882_pcm_digital_playback;
9892 spec->stream_digital_capture = &alc882_pcm_digital_capture;
9893
9894 if (codec->vendor_id == 0x10ec0888)
Takashi Iwai4a79ba32009-04-22 16:31:35 +02009895 spec->init_amp = ALC_INIT_DEFAULT; /* always initialize */
Takashi Iwai49535502009-06-30 15:28:30 +02009896
9897 if (!spec->adc_nids && spec->input_mux) {
9898 int i;
9899 spec->num_adc_nids = 0;
9900 for (i = 0; i < ARRAY_SIZE(alc882_adc_nids); i++) {
9901 hda_nid_t cap;
9902 hda_nid_t nid = alc882_adc_nids[i];
9903 unsigned int wcap = get_wcaps(codec, nid);
9904 /* get type */
Takashi Iwaia22d5432009-07-27 12:54:26 +02009905 wcap = get_wcaps_type(wcap);
Takashi Iwai49535502009-06-30 15:28:30 +02009906 if (wcap != AC_WID_AUD_IN)
9907 continue;
9908 spec->private_adc_nids[spec->num_adc_nids] = nid;
9909 err = snd_hda_get_connections(codec, nid, &cap, 1);
9910 if (err < 0)
9911 continue;
9912 spec->private_capsrc_nids[spec->num_adc_nids] = cap;
9913 spec->num_adc_nids++;
Herton Ronaldo Krzesinski61b9b9b2009-01-28 09:16:33 -02009914 }
Takashi Iwai49535502009-06-30 15:28:30 +02009915 spec->adc_nids = spec->private_adc_nids;
9916 spec->capsrc_nids = spec->private_capsrc_nids;
Kailang Yang2f893282008-05-27 12:14:47 +02009917 }
9918
Takashi Iwaib59bdf32009-08-11 09:47:30 +02009919 set_capture_mixer(codec);
Takashi Iwai45bdd1c2009-02-06 16:11:25 +01009920 set_beep_amp(spec, 0x0b, 0x05, HDA_INPUT);
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009921
Takashi Iwai2134ea42008-01-10 16:53:55 +01009922 spec->vmaster_nid = 0x0c;
9923
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009924 codec->patch_ops = alc_patch_ops;
Takashi Iwai49535502009-06-30 15:28:30 +02009925 if (board_config == ALC882_AUTO)
9926 spec->init_hook = alc882_auto_init;
Takashi Iwaicb53c622007-08-10 17:21:45 +02009927#ifdef CONFIG_SND_HDA_POWER_SAVE
9928 if (!spec->loopback.amplist)
Takashi Iwai49535502009-06-30 15:28:30 +02009929 spec->loopback.amplist = alc882_loopbacks;
Takashi Iwaicb53c622007-08-10 17:21:45 +02009930#endif
Takashi Iwaidaead532008-11-28 12:55:36 +01009931 codec->proc_widget_hook = print_realtek_coef;
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009932
9933 return 0;
9934}
9935
Takashi Iwai49535502009-06-30 15:28:30 +02009936
Takashi Iwai9c7f8522006-06-28 15:08:22 +02009937/*
Kailang Yangdf694da2005-12-05 19:42:22 +01009938 * ALC262 support
9939 */
9940
9941#define ALC262_DIGOUT_NID ALC880_DIGOUT_NID
9942#define ALC262_DIGIN_NID ALC880_DIGIN_NID
9943
9944#define alc262_dac_nids alc260_dac_nids
9945#define alc262_adc_nids alc882_adc_nids
9946#define alc262_adc_nids_alt alc882_adc_nids_alt
Takashi Iwai88c71a92008-02-14 17:27:17 +01009947#define alc262_capsrc_nids alc882_capsrc_nids
9948#define alc262_capsrc_nids_alt alc882_capsrc_nids_alt
Kailang Yangdf694da2005-12-05 19:42:22 +01009949
9950#define alc262_modes alc260_modes
Takashi Iwaic5f2ea02005-12-06 18:54:31 +01009951#define alc262_capture_source alc882_capture_source
Kailang Yangdf694da2005-12-05 19:42:22 +01009952
Kailang Yang4e555fe2008-08-26 13:05:55 +02009953static hda_nid_t alc262_dmic_adc_nids[1] = {
9954 /* ADC0 */
9955 0x09
9956};
9957
9958static hda_nid_t alc262_dmic_capsrc_nids[1] = { 0x22 };
9959
Kailang Yangdf694da2005-12-05 19:42:22 +01009960static struct snd_kcontrol_new alc262_base_mixer[] = {
9961 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
9962 HDA_CODEC_MUTE("Front Playback Switch", 0x14, 0x0, HDA_OUTPUT),
9963 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
9964 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
9965 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
9966 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
9967 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
9968 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Takashi Iwaicc69d122007-02-15 19:29:26 +01009969 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Kailang Yangdf694da2005-12-05 19:42:22 +01009970 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
9971 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
Takashi Iwaicc69d122007-02-15 19:29:26 +01009972 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
Kailang Yangdf694da2005-12-05 19:42:22 +01009973 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0D, 0x0, HDA_OUTPUT),
9974 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
9975 HDA_CODEC_VOLUME_MONO("Mono Playback Volume", 0x0e, 2, 0x0, HDA_OUTPUT),
9976 HDA_CODEC_MUTE_MONO("Mono Playback Switch", 0x16, 2, 0x0, HDA_OUTPUT),
Kailang Yangdf694da2005-12-05 19:42:22 +01009977 { } /* end */
Takashi Iwai834be882006-03-01 14:16:17 +01009978};
9979
Takashi Iwaice875f02008-01-28 18:17:43 +01009980/* update HP, line and mono-out pins according to the master switch */
9981static void alc262_hp_master_update(struct hda_codec *codec)
9982{
9983 struct alc_spec *spec = codec->spec;
9984 int val = spec->master_sw;
9985
9986 /* HP & line-out */
9987 snd_hda_codec_write_cache(codec, 0x1b, 0,
9988 AC_VERB_SET_PIN_WIDGET_CONTROL,
9989 val ? PIN_HP : 0);
9990 snd_hda_codec_write_cache(codec, 0x15, 0,
9991 AC_VERB_SET_PIN_WIDGET_CONTROL,
9992 val ? PIN_HP : 0);
9993 /* mono (speaker) depending on the HP jack sense */
9994 val = val && !spec->jack_present;
9995 snd_hda_codec_write_cache(codec, 0x16, 0,
9996 AC_VERB_SET_PIN_WIDGET_CONTROL,
9997 val ? PIN_OUT : 0);
9998}
9999
10000static void alc262_hp_bpc_automute(struct hda_codec *codec)
10001{
10002 struct alc_spec *spec = codec->spec;
Wu Fengguang864f92b2009-11-18 12:38:02 +080010003
10004 spec->jack_present = snd_hda_jack_detect(codec, 0x1b);
Takashi Iwaice875f02008-01-28 18:17:43 +010010005 alc262_hp_master_update(codec);
10006}
10007
10008static void alc262_hp_bpc_unsol_event(struct hda_codec *codec, unsigned int res)
10009{
10010 if ((res >> 26) != ALC880_HP_EVENT)
10011 return;
10012 alc262_hp_bpc_automute(codec);
10013}
10014
10015static void alc262_hp_wildwest_automute(struct hda_codec *codec)
10016{
10017 struct alc_spec *spec = codec->spec;
Wu Fengguang864f92b2009-11-18 12:38:02 +080010018
10019 spec->jack_present = snd_hda_jack_detect(codec, 0x15);
Takashi Iwaice875f02008-01-28 18:17:43 +010010020 alc262_hp_master_update(codec);
10021}
10022
10023static void alc262_hp_wildwest_unsol_event(struct hda_codec *codec,
10024 unsigned int res)
10025{
10026 if ((res >> 26) != ALC880_HP_EVENT)
10027 return;
10028 alc262_hp_wildwest_automute(codec);
10029}
10030
Takashi Iwaib72519b2009-05-08 14:31:55 +020010031#define alc262_hp_master_sw_get alc260_hp_master_sw_get
Takashi Iwaice875f02008-01-28 18:17:43 +010010032
10033static int alc262_hp_master_sw_put(struct snd_kcontrol *kcontrol,
10034 struct snd_ctl_elem_value *ucontrol)
10035{
10036 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
10037 struct alc_spec *spec = codec->spec;
10038 int val = !!*ucontrol->value.integer.value;
10039
10040 if (val == spec->master_sw)
10041 return 0;
10042 spec->master_sw = val;
10043 alc262_hp_master_update(codec);
10044 return 1;
10045}
10046
Takashi Iwaib72519b2009-05-08 14:31:55 +020010047#define ALC262_HP_MASTER_SWITCH \
10048 { \
10049 .iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
10050 .name = "Master Playback Switch", \
10051 .info = snd_ctl_boolean_mono_info, \
10052 .get = alc262_hp_master_sw_get, \
10053 .put = alc262_hp_master_sw_put, \
10054 }
10055
Takashi Iwai9c7f8522006-06-28 15:08:22 +020010056static struct snd_kcontrol_new alc262_HP_BPC_mixer[] = {
Takashi Iwaib72519b2009-05-08 14:31:55 +020010057 ALC262_HP_MASTER_SWITCH,
Takashi Iwai9c7f8522006-06-28 15:08:22 +020010058 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
10059 HDA_CODEC_MUTE("Front Playback Switch", 0x15, 0x0, HDA_OUTPUT),
10060 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
Takashi Iwaice875f02008-01-28 18:17:43 +010010061 HDA_CODEC_VOLUME_MONO("Speaker Playback Volume", 0x0e, 2, 0x0,
10062 HDA_OUTPUT),
10063 HDA_CODEC_MUTE_MONO("Speaker Playback Switch", 0x16, 2, 0x0,
10064 HDA_OUTPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +020010065 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
10066 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Takashi Iwaicc69d122007-02-15 19:29:26 +010010067 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +020010068 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
10069 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
Takashi Iwaicc69d122007-02-15 19:29:26 +010010070 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +020010071 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
10072 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
10073 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
10074 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
Takashi Iwai9c7f8522006-06-28 15:08:22 +020010075 HDA_CODEC_VOLUME("AUX IN Playback Volume", 0x0b, 0x06, HDA_INPUT),
10076 HDA_CODEC_MUTE("AUX IN Playback Switch", 0x0b, 0x06, HDA_INPUT),
10077 { } /* end */
10078};
10079
Kailang Yangcd7509a2007-01-26 18:33:17 +010010080static struct snd_kcontrol_new alc262_HP_BPC_WildWest_mixer[] = {
Takashi Iwaib72519b2009-05-08 14:31:55 +020010081 ALC262_HP_MASTER_SWITCH,
Kailang Yangcd7509a2007-01-26 18:33:17 +010010082 HDA_CODEC_VOLUME("Front Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
10083 HDA_CODEC_MUTE("Front Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
10084 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
10085 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
Takashi Iwaice875f02008-01-28 18:17:43 +010010086 HDA_CODEC_VOLUME_MONO("Speaker Playback Volume", 0x0e, 2, 0x0,
10087 HDA_OUTPUT),
10088 HDA_CODEC_MUTE_MONO("Speaker Playback Switch", 0x16, 2, 0x0,
10089 HDA_OUTPUT),
Kailang Yangcd7509a2007-01-26 18:33:17 +010010090 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x02, HDA_INPUT),
10091 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x02, HDA_INPUT),
Takashi Iwaicc69d122007-02-15 19:29:26 +010010092 HDA_CODEC_VOLUME("Front Mic Boost", 0x1a, 0, HDA_INPUT),
Kailang Yangcd7509a2007-01-26 18:33:17 +010010093 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x01, HDA_INPUT),
10094 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x01, HDA_INPUT),
10095 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
10096 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
Kailang Yangcd7509a2007-01-26 18:33:17 +010010097 { } /* end */
10098};
10099
10100static struct snd_kcontrol_new alc262_HP_BPC_WildWest_option_mixer[] = {
10101 HDA_CODEC_VOLUME("Rear Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
10102 HDA_CODEC_MUTE("Rear Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Takashi Iwaicc69d122007-02-15 19:29:26 +010010103 HDA_CODEC_VOLUME("Rear Mic Boost", 0x18, 0, HDA_INPUT),
Kailang Yangcd7509a2007-01-26 18:33:17 +010010104 { } /* end */
10105};
10106
Kailang Yang66d2a9d2007-11-14 12:06:21 +010010107/* mute/unmute internal speaker according to the hp jack and mute state */
Takashi Iwai4f5d17062009-08-11 18:17:46 +020010108static void alc262_hp_t5735_setup(struct hda_codec *codec)
Kailang Yang66d2a9d2007-11-14 12:06:21 +010010109{
10110 struct alc_spec *spec = codec->spec;
Kailang Yang66d2a9d2007-11-14 12:06:21 +010010111
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020010112 spec->autocfg.hp_pins[0] = 0x15;
10113 spec->autocfg.speaker_pins[0] = 0x0c; /* HACK: not actually a pin */
Kailang Yang66d2a9d2007-11-14 12:06:21 +010010114}
10115
Kailang Yang66d2a9d2007-11-14 12:06:21 +010010116static struct snd_kcontrol_new alc262_hp_t5735_mixer[] = {
Takashi Iwai86cd9292008-01-28 18:09:56 +010010117 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
10118 HDA_CODEC_MUTE("Speaker Playback Switch", 0x14, 0x0, HDA_OUTPUT),
Kailang Yang66d2a9d2007-11-14 12:06:21 +010010119 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
10120 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
10121 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
10122 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
10123 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
10124 { } /* end */
10125};
10126
10127static struct hda_verb alc262_hp_t5735_verbs[] = {
10128 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
10129 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
10130
10131 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
10132 { }
10133};
10134
Kailang Yang8c427222008-01-10 13:03:59 +010010135static struct snd_kcontrol_new alc262_hp_rp5700_mixer[] = {
Takashi Iwaif2f48e12008-01-28 18:14:43 +010010136 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
10137 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
Takashi Iwai86cd9292008-01-28 18:09:56 +010010138 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0e, 0x0, HDA_OUTPUT),
10139 HDA_CODEC_MUTE("Speaker Playback Switch", 0x16, 0x0, HDA_OUTPUT),
Kailang Yang8c427222008-01-10 13:03:59 +010010140 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x01, HDA_INPUT),
10141 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x01, HDA_INPUT),
10142 { } /* end */
10143};
10144
10145static struct hda_verb alc262_hp_rp5700_verbs[] = {
10146 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
10147 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
10148 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
10149 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
10150 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
10151 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
10152 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
10153 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
10154 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x00 << 8))},
10155 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x00 << 8))},
10156 {}
10157};
10158
10159static struct hda_input_mux alc262_hp_rp5700_capture_source = {
10160 .num_items = 1,
10161 .items = {
10162 { "Line", 0x1 },
10163 },
10164};
10165
Takashi Iwai42171c12009-05-08 14:11:43 +020010166/* bind hp and internal speaker mute (with plug check) as master switch */
10167static void alc262_hippo_master_update(struct hda_codec *codec)
10168{
10169 struct alc_spec *spec = codec->spec;
10170 hda_nid_t hp_nid = spec->autocfg.hp_pins[0];
10171 hda_nid_t line_nid = spec->autocfg.line_out_pins[0];
10172 hda_nid_t speaker_nid = spec->autocfg.speaker_pins[0];
10173 unsigned int mute;
10174
10175 /* HP */
10176 mute = spec->master_sw ? 0 : HDA_AMP_MUTE;
10177 snd_hda_codec_amp_stereo(codec, hp_nid, HDA_OUTPUT, 0,
10178 HDA_AMP_MUTE, mute);
10179 /* mute internal speaker per jack sense */
10180 if (spec->jack_present)
10181 mute = HDA_AMP_MUTE;
10182 if (line_nid)
10183 snd_hda_codec_amp_stereo(codec, line_nid, HDA_OUTPUT, 0,
10184 HDA_AMP_MUTE, mute);
10185 if (speaker_nid && speaker_nid != line_nid)
10186 snd_hda_codec_amp_stereo(codec, speaker_nid, HDA_OUTPUT, 0,
10187 HDA_AMP_MUTE, mute);
10188}
10189
10190#define alc262_hippo_master_sw_get alc262_hp_master_sw_get
10191
10192static int alc262_hippo_master_sw_put(struct snd_kcontrol *kcontrol,
10193 struct snd_ctl_elem_value *ucontrol)
Takashi Iwai0724ea22007-08-23 00:31:43 +020010194{
10195 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
Takashi Iwai42171c12009-05-08 14:11:43 +020010196 struct alc_spec *spec = codec->spec;
10197 int val = !!*ucontrol->value.integer.value;
Takashi Iwai0724ea22007-08-23 00:31:43 +020010198
Takashi Iwai42171c12009-05-08 14:11:43 +020010199 if (val == spec->master_sw)
10200 return 0;
10201 spec->master_sw = val;
10202 alc262_hippo_master_update(codec);
10203 return 1;
Takashi Iwai0724ea22007-08-23 00:31:43 +020010204}
Takashi Iwai5b319542007-07-26 11:49:22 +020010205
Takashi Iwai42171c12009-05-08 14:11:43 +020010206#define ALC262_HIPPO_MASTER_SWITCH \
10207 { \
10208 .iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
10209 .name = "Master Playback Switch", \
10210 .info = snd_ctl_boolean_mono_info, \
10211 .get = alc262_hippo_master_sw_get, \
10212 .put = alc262_hippo_master_sw_put, \
10213 }
10214
10215static struct snd_kcontrol_new alc262_hippo_mixer[] = {
10216 ALC262_HIPPO_MASTER_SWITCH,
10217 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
10218 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
10219 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
10220 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
10221 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
10222 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
10223 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
10224 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
10225 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
10226 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
10227 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
10228 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
10229 { } /* end */
10230};
10231
10232static struct snd_kcontrol_new alc262_hippo1_mixer[] = {
10233 HDA_CODEC_VOLUME("Master Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
10234 ALC262_HIPPO_MASTER_SWITCH,
10235 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
10236 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
10237 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
10238 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
10239 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
10240 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
10241 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
10242 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
10243 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
10244 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
10245 { } /* end */
10246};
10247
10248/* mute/unmute internal speaker according to the hp jack and mute state */
10249static void alc262_hippo_automute(struct hda_codec *codec)
10250{
10251 struct alc_spec *spec = codec->spec;
10252 hda_nid_t hp_nid = spec->autocfg.hp_pins[0];
Takashi Iwai42171c12009-05-08 14:11:43 +020010253
Wu Fengguang864f92b2009-11-18 12:38:02 +080010254 spec->jack_present = snd_hda_jack_detect(codec, hp_nid);
Takashi Iwai42171c12009-05-08 14:11:43 +020010255 alc262_hippo_master_update(codec);
10256}
10257
10258static void alc262_hippo_unsol_event(struct hda_codec *codec, unsigned int res)
10259{
10260 if ((res >> 26) != ALC880_HP_EVENT)
10261 return;
10262 alc262_hippo_automute(codec);
10263}
10264
Takashi Iwai4f5d17062009-08-11 18:17:46 +020010265static void alc262_hippo_setup(struct hda_codec *codec)
Takashi Iwai42171c12009-05-08 14:11:43 +020010266{
10267 struct alc_spec *spec = codec->spec;
10268
10269 spec->autocfg.hp_pins[0] = 0x15;
10270 spec->autocfg.speaker_pins[0] = 0x14;
Takashi Iwai42171c12009-05-08 14:11:43 +020010271}
10272
Takashi Iwai4f5d17062009-08-11 18:17:46 +020010273static void alc262_hippo1_setup(struct hda_codec *codec)
Takashi Iwai42171c12009-05-08 14:11:43 +020010274{
10275 struct alc_spec *spec = codec->spec;
10276
10277 spec->autocfg.hp_pins[0] = 0x1b;
10278 spec->autocfg.speaker_pins[0] = 0x14;
Takashi Iwai42171c12009-05-08 14:11:43 +020010279}
10280
10281
Kailang Yang272a5272007-05-14 11:00:38 +020010282static struct snd_kcontrol_new alc262_sony_mixer[] = {
Takashi Iwai0724ea22007-08-23 00:31:43 +020010283 HDA_CODEC_VOLUME("Master Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
Takashi Iwai42171c12009-05-08 14:11:43 +020010284 ALC262_HIPPO_MASTER_SWITCH,
Kailang Yang272a5272007-05-14 11:00:38 +020010285 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
10286 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
10287 HDA_CODEC_VOLUME("ATAPI Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
10288 HDA_CODEC_MUTE("ATAPI Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
10289 { } /* end */
10290};
10291
Kailang Yang83c34212007-07-05 11:43:05 +020010292static struct snd_kcontrol_new alc262_benq_t31_mixer[] = {
Takashi Iwai42171c12009-05-08 14:11:43 +020010293 HDA_CODEC_VOLUME("Master Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
10294 ALC262_HIPPO_MASTER_SWITCH,
Kailang Yang83c34212007-07-05 11:43:05 +020010295 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
10296 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
10297 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
10298 HDA_CODEC_VOLUME("ATAPI Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
10299 HDA_CODEC_MUTE("ATAPI Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
10300 { } /* end */
10301};
Kailang Yang272a5272007-05-14 11:00:38 +020010302
Tony Vroonba340e82009-02-02 19:01:30 +000010303static struct snd_kcontrol_new alc262_tyan_mixer[] = {
10304 HDA_CODEC_VOLUME("Master Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
10305 HDA_BIND_MUTE("Master Playback Switch", 0x0c, 2, HDA_INPUT),
10306 HDA_CODEC_VOLUME("Aux Playback Volume", 0x0b, 0x06, HDA_INPUT),
10307 HDA_CODEC_MUTE("Aux Playback Switch", 0x0b, 0x06, HDA_INPUT),
10308 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
10309 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
10310 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
10311 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
10312 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
10313 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
10314 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
10315 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
10316 { } /* end */
10317};
10318
10319static struct hda_verb alc262_tyan_verbs[] = {
10320 /* Headphone automute */
10321 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
10322 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
10323 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
10324
10325 /* P11 AUX_IN, white 4-pin connector */
10326 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
10327 {0x14, AC_VERB_SET_CONFIG_DEFAULT_BYTES_1, 0xe1},
10328 {0x14, AC_VERB_SET_CONFIG_DEFAULT_BYTES_2, 0x93},
10329 {0x14, AC_VERB_SET_CONFIG_DEFAULT_BYTES_3, 0x19},
10330
10331 {}
10332};
10333
10334/* unsolicited event for HP jack sensing */
Takashi Iwai4f5d17062009-08-11 18:17:46 +020010335static void alc262_tyan_setup(struct hda_codec *codec)
Tony Vroonba340e82009-02-02 19:01:30 +000010336{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020010337 struct alc_spec *spec = codec->spec;
Tony Vroonba340e82009-02-02 19:01:30 +000010338
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020010339 spec->autocfg.hp_pins[0] = 0x1b;
10340 spec->autocfg.speaker_pins[0] = 0x15;
Tony Vroonba340e82009-02-02 19:01:30 +000010341}
10342
Tony Vroonba340e82009-02-02 19:01:30 +000010343
Kailang Yangdf694da2005-12-05 19:42:22 +010010344#define alc262_capture_mixer alc882_capture_mixer
10345#define alc262_capture_alt_mixer alc882_capture_alt_mixer
10346
10347/*
10348 * generic initialization of ADC, input mixers and output mixers
10349 */
10350static struct hda_verb alc262_init_verbs[] = {
10351 /*
10352 * Unmute ADC0-2 and set the default input to mic-in
10353 */
10354 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
10355 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10356 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
10357 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10358 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
10359 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10360
Takashi Iwaicb53c622007-08-10 17:21:45 +020010361 /* Mute input amps (CD, Line In, Mic 1 & Mic 2) of the analog-loopback
Kailang Yangdf694da2005-12-05 19:42:22 +010010362 * mixer widget
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020010363 * Note: PASD motherboards uses the Line In 2 as the input for
10364 * front panel mic (mic 2)
Kailang Yangdf694da2005-12-05 19:42:22 +010010365 */
10366 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +020010367 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
10368 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
10369 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
10370 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
10371 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Kailang Yangdf694da2005-12-05 19:42:22 +010010372
10373 /*
10374 * Set up output mixers (0x0c - 0x0e)
10375 */
10376 /* set vol=0 to output mixers */
10377 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
10378 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
10379 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
10380 /* set up input amps for analog loopback */
10381 /* Amp Indices: DAC = 0, mixer = 1 */
10382 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10383 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
10384 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10385 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
10386 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10387 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
10388
10389 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40},
10390 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc0},
10391 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40},
10392 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
10393 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
10394 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
10395
10396 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
10397 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
10398 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
10399 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
10400 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
Kailang Yangea1fb292008-08-26 12:58:38 +020010401
Kailang Yangdf694da2005-12-05 19:42:22 +010010402 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
10403 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01},
Kailang Yangea1fb292008-08-26 12:58:38 +020010404
Kailang Yangdf694da2005-12-05 19:42:22 +010010405 /* FIXME: use matrix-type input source selection */
10406 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
10407 /* Input mixer1: unmute Mic, F-Mic, Line, CD inputs */
10408 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
10409 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
10410 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
10411 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
10412 /* Input mixer2 */
10413 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
10414 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
10415 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
10416 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
10417 /* Input mixer3 */
10418 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
10419 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
10420 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020010421 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
Kailang Yangdf694da2005-12-05 19:42:22 +010010422
10423 { }
10424};
10425
Kailang Yang4e555fe2008-08-26 13:05:55 +020010426static struct hda_verb alc262_eapd_verbs[] = {
10427 {0x14, AC_VERB_SET_EAPD_BTLENABLE, 2},
10428 {0x15, AC_VERB_SET_EAPD_BTLENABLE, 2},
10429 { }
10430};
10431
Kailang Yangccc656c2006-10-17 12:32:26 +020010432static struct hda_verb alc262_hippo1_unsol_verbs[] = {
10433 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc0},
10434 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
10435 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, 0x0000},
10436
10437 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
10438 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
10439 {}
10440};
10441
Kailang Yang272a5272007-05-14 11:00:38 +020010442static struct hda_verb alc262_sony_unsol_verbs[] = {
10443 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc0},
10444 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
10445 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24}, // Front Mic
10446
10447 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
10448 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Akio Idehara7b1e8792008-06-09 22:46:07 +090010449 {}
Kailang Yang272a5272007-05-14 11:00:38 +020010450};
10451
Kailang Yang4e555fe2008-08-26 13:05:55 +020010452static struct snd_kcontrol_new alc262_toshiba_s06_mixer[] = {
10453 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
10454 HDA_CODEC_MUTE("Speaker Playback Switch", 0x14, 0x0, HDA_OUTPUT),
10455 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
10456 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
10457 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Kailang Yang4e555fe2008-08-26 13:05:55 +020010458 { } /* end */
10459};
10460
10461static struct hda_verb alc262_toshiba_s06_verbs[] = {
10462 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
10463 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
10464 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
10465 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
10466 {0x22, AC_VERB_SET_CONNECT_SEL, 0x09},
10467 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
10468 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
10469 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
10470 {}
10471};
10472
Takashi Iwai4f5d17062009-08-11 18:17:46 +020010473static void alc262_toshiba_s06_setup(struct hda_codec *codec)
Kailang Yang4e555fe2008-08-26 13:05:55 +020010474{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020010475 struct alc_spec *spec = codec->spec;
10476
10477 spec->autocfg.hp_pins[0] = 0x15;
10478 spec->autocfg.speaker_pins[0] = 0x14;
Takashi Iwai4f5d17062009-08-11 18:17:46 +020010479 spec->ext_mic.pin = 0x18;
10480 spec->ext_mic.mux_idx = 0;
10481 spec->int_mic.pin = 0x12;
10482 spec->int_mic.mux_idx = 9;
10483 spec->auto_mic = 1;
Kailang Yang4e555fe2008-08-26 13:05:55 +020010484}
10485
Takashi Iwai834be882006-03-01 14:16:17 +010010486/*
Pascal Terjane8f9ae22008-08-04 14:36:05 +020010487 * nec model
10488 * 0x15 = headphone
10489 * 0x16 = internal speaker
10490 * 0x18 = external mic
10491 */
10492
10493static struct snd_kcontrol_new alc262_nec_mixer[] = {
10494 HDA_CODEC_VOLUME_MONO("Speaker Playback Volume", 0x0e, 1, 0x0, HDA_OUTPUT),
10495 HDA_CODEC_MUTE_MONO("Speaker Playback Switch", 0x16, 0, 0x0, HDA_OUTPUT),
10496
10497 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
10498 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
10499 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
10500
10501 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x0d, 0x0, HDA_OUTPUT),
10502 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
10503 { } /* end */
10504};
10505
10506static struct hda_verb alc262_nec_verbs[] = {
10507 /* Unmute Speaker */
10508 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
10509
10510 /* Headphone */
10511 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
10512 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
10513
10514 /* External mic to headphone */
10515 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
10516 /* External mic to speaker */
10517 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
10518 {}
10519};
10520
10521/*
Takashi Iwai834be882006-03-01 14:16:17 +010010522 * fujitsu model
Tony Vroon5d9fab22008-03-14 17:09:18 +010010523 * 0x14 = headphone/spdif-out, 0x15 = internal speaker,
10524 * 0x1b = port replicator headphone out
Takashi Iwai834be882006-03-01 14:16:17 +010010525 */
10526
10527#define ALC_HP_EVENT 0x37
10528
10529static struct hda_verb alc262_fujitsu_unsol_verbs[] = {
10530 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC_HP_EVENT},
10531 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Tony Vroon5d9fab22008-03-14 17:09:18 +010010532 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC_HP_EVENT},
10533 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Takashi Iwai834be882006-03-01 14:16:17 +010010534 {}
10535};
10536
Jiang zhe0e31daf2008-03-20 12:12:39 +010010537static struct hda_verb alc262_lenovo_3000_unsol_verbs[] = {
10538 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC_HP_EVENT},
10539 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
10540 {}
10541};
10542
Takashi Iwai834be882006-03-01 14:16:17 +010010543static struct hda_input_mux alc262_fujitsu_capture_source = {
Takashi Iwai39d3ed32007-10-12 15:03:48 +020010544 .num_items = 3,
Takashi Iwai834be882006-03-01 14:16:17 +010010545 .items = {
10546 { "Mic", 0x0 },
Takashi Iwai39d3ed32007-10-12 15:03:48 +020010547 { "Int Mic", 0x1 },
Takashi Iwai834be882006-03-01 14:16:17 +010010548 { "CD", 0x4 },
10549 },
10550};
10551
Takashi Iwai9c7f8522006-06-28 15:08:22 +020010552static struct hda_input_mux alc262_HP_capture_source = {
10553 .num_items = 5,
10554 .items = {
10555 { "Mic", 0x0 },
zhejiangaccbe492007-08-31 12:36:05 +020010556 { "Front Mic", 0x1 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +020010557 { "Line", 0x2 },
10558 { "CD", 0x4 },
10559 { "AUX IN", 0x6 },
10560 },
10561};
10562
zhejiangaccbe492007-08-31 12:36:05 +020010563static struct hda_input_mux alc262_HP_D7000_capture_source = {
10564 .num_items = 4,
10565 .items = {
10566 { "Mic", 0x0 },
10567 { "Front Mic", 0x2 },
10568 { "Line", 0x1 },
10569 { "CD", 0x4 },
10570 },
10571};
10572
Takashi Iwaiebc7a402008-05-20 09:23:05 +020010573/* mute/unmute internal speaker according to the hp jacks and mute state */
Takashi Iwai834be882006-03-01 14:16:17 +010010574static void alc262_fujitsu_automute(struct hda_codec *codec, int force)
10575{
10576 struct alc_spec *spec = codec->spec;
10577 unsigned int mute;
10578
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020010579 if (force || !spec->sense_updated) {
Wu Fengguang864f92b2009-11-18 12:38:02 +080010580 spec->jack_present = snd_hda_jack_detect(codec, 0x14) ||
10581 snd_hda_jack_detect(codec, 0x1b);
Takashi Iwai834be882006-03-01 14:16:17 +010010582 spec->sense_updated = 1;
10583 }
Takashi Iwaiebc7a402008-05-20 09:23:05 +020010584 /* unmute internal speaker only if both HPs are unplugged and
10585 * master switch is on
10586 */
10587 if (spec->jack_present)
10588 mute = HDA_AMP_MUTE;
10589 else
Takashi Iwai834be882006-03-01 14:16:17 +010010590 mute = snd_hda_codec_amp_read(codec, 0x14, 0, HDA_OUTPUT, 0);
Takashi Iwaiebc7a402008-05-20 09:23:05 +020010591 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
10592 HDA_AMP_MUTE, mute);
Takashi Iwai834be882006-03-01 14:16:17 +010010593}
10594
10595/* unsolicited event for HP jack sensing */
10596static void alc262_fujitsu_unsol_event(struct hda_codec *codec,
10597 unsigned int res)
10598{
10599 if ((res >> 26) != ALC_HP_EVENT)
10600 return;
10601 alc262_fujitsu_automute(codec, 1);
10602}
10603
Takashi Iwaiebc7a402008-05-20 09:23:05 +020010604static void alc262_fujitsu_init_hook(struct hda_codec *codec)
10605{
10606 alc262_fujitsu_automute(codec, 1);
10607}
10608
Takashi Iwai834be882006-03-01 14:16:17 +010010609/* bind volumes of both NID 0x0c and 0x0d */
Takashi Iwaicca3b372007-08-10 17:12:15 +020010610static struct hda_bind_ctls alc262_fujitsu_bind_master_vol = {
10611 .ops = &snd_hda_bind_vol,
10612 .values = {
10613 HDA_COMPOSE_AMP_VAL(0x0c, 3, 0, HDA_OUTPUT),
10614 HDA_COMPOSE_AMP_VAL(0x0d, 3, 0, HDA_OUTPUT),
10615 0
10616 },
10617};
Takashi Iwai834be882006-03-01 14:16:17 +010010618
Jiang zhe0e31daf2008-03-20 12:12:39 +010010619/* mute/unmute internal speaker according to the hp jack and mute state */
10620static void alc262_lenovo_3000_automute(struct hda_codec *codec, int force)
10621{
10622 struct alc_spec *spec = codec->spec;
10623 unsigned int mute;
10624
10625 if (force || !spec->sense_updated) {
Wu Fengguang864f92b2009-11-18 12:38:02 +080010626 spec->jack_present = snd_hda_jack_detect(codec, 0x1b);
Jiang zhe0e31daf2008-03-20 12:12:39 +010010627 spec->sense_updated = 1;
10628 }
10629 if (spec->jack_present) {
10630 /* mute internal speaker */
10631 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
10632 HDA_AMP_MUTE, HDA_AMP_MUTE);
10633 snd_hda_codec_amp_stereo(codec, 0x16, HDA_OUTPUT, 0,
10634 HDA_AMP_MUTE, HDA_AMP_MUTE);
10635 } else {
10636 /* unmute internal speaker if necessary */
10637 mute = snd_hda_codec_amp_read(codec, 0x1b, 0, HDA_OUTPUT, 0);
10638 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
10639 HDA_AMP_MUTE, mute);
10640 snd_hda_codec_amp_stereo(codec, 0x16, HDA_OUTPUT, 0,
10641 HDA_AMP_MUTE, mute);
10642 }
10643}
10644
10645/* unsolicited event for HP jack sensing */
10646static void alc262_lenovo_3000_unsol_event(struct hda_codec *codec,
10647 unsigned int res)
10648{
10649 if ((res >> 26) != ALC_HP_EVENT)
10650 return;
10651 alc262_lenovo_3000_automute(codec, 1);
10652}
10653
Takashi Iwai8de56b72009-07-24 16:51:47 +020010654static int amp_stereo_mute_update(struct hda_codec *codec, hda_nid_t nid,
10655 int dir, int idx, long *valp)
10656{
10657 int i, change = 0;
10658
10659 for (i = 0; i < 2; i++, valp++)
10660 change |= snd_hda_codec_amp_update(codec, nid, i, dir, idx,
10661 HDA_AMP_MUTE,
10662 *valp ? 0 : HDA_AMP_MUTE);
10663 return change;
10664}
10665
Takashi Iwai834be882006-03-01 14:16:17 +010010666/* bind hp and internal speaker mute (with plug check) */
10667static int alc262_fujitsu_master_sw_put(struct snd_kcontrol *kcontrol,
10668 struct snd_ctl_elem_value *ucontrol)
10669{
10670 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
10671 long *valp = ucontrol->value.integer.value;
10672 int change;
10673
Takashi Iwai8de56b72009-07-24 16:51:47 +020010674 change = amp_stereo_mute_update(codec, 0x14, HDA_OUTPUT, 0, valp);
10675 change |= amp_stereo_mute_update(codec, 0x1b, HDA_OUTPUT, 0, valp);
Takashi Iwai82beb8f2007-08-10 17:09:26 +020010676 if (change)
10677 alc262_fujitsu_automute(codec, 0);
Takashi Iwai834be882006-03-01 14:16:17 +010010678 return change;
10679}
10680
10681static struct snd_kcontrol_new alc262_fujitsu_mixer[] = {
Takashi Iwaicca3b372007-08-10 17:12:15 +020010682 HDA_BIND_VOL("Master Playback Volume", &alc262_fujitsu_bind_master_vol),
Takashi Iwai834be882006-03-01 14:16:17 +010010683 {
10684 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
10685 .name = "Master Playback Switch",
10686 .info = snd_hda_mixer_amp_switch_info,
10687 .get = snd_hda_mixer_amp_switch_get,
10688 .put = alc262_fujitsu_master_sw_put,
10689 .private_value = HDA_COMPOSE_AMP_VAL(0x14, 3, 0, HDA_OUTPUT),
10690 },
10691 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
10692 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
10693 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
10694 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
10695 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Takashi Iwai39d3ed32007-10-12 15:03:48 +020010696 HDA_CODEC_VOLUME("Int Mic Boost", 0x19, 0, HDA_INPUT),
10697 HDA_CODEC_VOLUME("Int Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
10698 HDA_CODEC_MUTE("Int Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Takashi Iwai834be882006-03-01 14:16:17 +010010699 { } /* end */
10700};
10701
Jiang zhe0e31daf2008-03-20 12:12:39 +010010702/* bind hp and internal speaker mute (with plug check) */
10703static int alc262_lenovo_3000_master_sw_put(struct snd_kcontrol *kcontrol,
10704 struct snd_ctl_elem_value *ucontrol)
10705{
10706 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
10707 long *valp = ucontrol->value.integer.value;
10708 int change;
10709
Takashi Iwai8de56b72009-07-24 16:51:47 +020010710 change = amp_stereo_mute_update(codec, 0x1b, HDA_OUTPUT, 0, valp);
Jiang zhe0e31daf2008-03-20 12:12:39 +010010711 if (change)
10712 alc262_lenovo_3000_automute(codec, 0);
10713 return change;
10714}
10715
10716static struct snd_kcontrol_new alc262_lenovo_3000_mixer[] = {
10717 HDA_BIND_VOL("Master Playback Volume", &alc262_fujitsu_bind_master_vol),
10718 {
10719 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
10720 .name = "Master Playback Switch",
10721 .info = snd_hda_mixer_amp_switch_info,
10722 .get = snd_hda_mixer_amp_switch_get,
10723 .put = alc262_lenovo_3000_master_sw_put,
10724 .private_value = HDA_COMPOSE_AMP_VAL(0x1b, 3, 0, HDA_OUTPUT),
10725 },
10726 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
10727 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
10728 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
10729 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
10730 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
10731 HDA_CODEC_VOLUME("Int Mic Boost", 0x19, 0, HDA_INPUT),
10732 HDA_CODEC_VOLUME("Int Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
10733 HDA_CODEC_MUTE("Int Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
10734 { } /* end */
10735};
10736
Hiroshi Miura9f99a632008-08-28 16:09:06 +020010737static struct snd_kcontrol_new alc262_toshiba_rx1_mixer[] = {
10738 HDA_BIND_VOL("Master Playback Volume", &alc262_fujitsu_bind_master_vol),
Takashi Iwai42171c12009-05-08 14:11:43 +020010739 ALC262_HIPPO_MASTER_SWITCH,
Hiroshi Miura9f99a632008-08-28 16:09:06 +020010740 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
10741 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
10742 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
10743 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
10744 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
10745 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
10746 { } /* end */
10747};
10748
Takashi Iwai304dcaa2006-07-25 14:51:16 +020010749/* additional init verbs for Benq laptops */
10750static struct hda_verb alc262_EAPD_verbs[] = {
10751 {0x20, AC_VERB_SET_COEF_INDEX, 0x07},
10752 {0x20, AC_VERB_SET_PROC_COEF, 0x3070},
10753 {}
10754};
10755
Kailang Yang83c34212007-07-05 11:43:05 +020010756static struct hda_verb alc262_benq_t31_EAPD_verbs[] = {
10757 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
10758 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
10759
10760 {0x20, AC_VERB_SET_COEF_INDEX, 0x07},
10761 {0x20, AC_VERB_SET_PROC_COEF, 0x3050},
10762 {}
10763};
10764
Tobin Davisf651b502007-10-26 12:40:47 +020010765/* Samsung Q1 Ultra Vista model setup */
10766static struct snd_kcontrol_new alc262_ultra_mixer[] = {
Takashi Iwaibb9f76c2008-03-12 12:51:09 +010010767 HDA_CODEC_VOLUME("Master Playback Volume", 0x0c, 0x0, HDA_OUTPUT),
10768 HDA_BIND_MUTE("Master Playback Switch", 0x0c, 2, HDA_INPUT),
Tobin Davisf651b502007-10-26 12:40:47 +020010769 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
10770 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
10771 HDA_CODEC_VOLUME("Mic Boost", 0x19, 0, HDA_INPUT),
Takashi Iwaibb9f76c2008-03-12 12:51:09 +010010772 HDA_CODEC_VOLUME("Headphone Mic Boost", 0x15, 0, HDA_INPUT),
Tobin Davisf651b502007-10-26 12:40:47 +020010773 { } /* end */
10774};
10775
10776static struct hda_verb alc262_ultra_verbs[] = {
Takashi Iwaibb9f76c2008-03-12 12:51:09 +010010777 /* output mixer */
10778 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
10779 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
10780 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
10781 /* speaker */
10782 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
10783 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
10784 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10785 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
10786 /* HP */
Tobin Davisf651b502007-10-26 12:40:47 +020010787 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Takashi Iwaibb9f76c2008-03-12 12:51:09 +010010788 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
10789 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10790 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
10791 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
10792 /* internal mic */
10793 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
10794 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
10795 /* ADC, choose mic */
10796 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
10797 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
10798 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
10799 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
10800 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
10801 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
10802 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(5)},
10803 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(6)},
10804 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(7)},
10805 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(8)},
Tobin Davisf651b502007-10-26 12:40:47 +020010806 {}
10807};
10808
Tobin Davisf651b502007-10-26 12:40:47 +020010809/* mute/unmute internal speaker according to the hp jack and mute state */
10810static void alc262_ultra_automute(struct hda_codec *codec)
10811{
10812 struct alc_spec *spec = codec->spec;
10813 unsigned int mute;
Tobin Davisf651b502007-10-26 12:40:47 +020010814
Takashi Iwaibb9f76c2008-03-12 12:51:09 +010010815 mute = 0;
10816 /* auto-mute only when HP is used as HP */
10817 if (!spec->cur_mux[0]) {
Wu Fengguang864f92b2009-11-18 12:38:02 +080010818 spec->jack_present = snd_hda_jack_detect(codec, 0x15);
Takashi Iwaibb9f76c2008-03-12 12:51:09 +010010819 if (spec->jack_present)
10820 mute = HDA_AMP_MUTE;
Tobin Davisf651b502007-10-26 12:40:47 +020010821 }
Takashi Iwaibb9f76c2008-03-12 12:51:09 +010010822 /* mute/unmute internal speaker */
10823 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
10824 HDA_AMP_MUTE, mute);
10825 /* mute/unmute HP */
10826 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
10827 HDA_AMP_MUTE, mute ? 0 : HDA_AMP_MUTE);
Tobin Davisf651b502007-10-26 12:40:47 +020010828}
10829
10830/* unsolicited event for HP jack sensing */
10831static void alc262_ultra_unsol_event(struct hda_codec *codec,
10832 unsigned int res)
10833{
10834 if ((res >> 26) != ALC880_HP_EVENT)
10835 return;
10836 alc262_ultra_automute(codec);
10837}
10838
Takashi Iwaibb9f76c2008-03-12 12:51:09 +010010839static struct hda_input_mux alc262_ultra_capture_source = {
10840 .num_items = 2,
10841 .items = {
10842 { "Mic", 0x1 },
10843 { "Headphone", 0x7 },
10844 },
10845};
10846
10847static int alc262_ultra_mux_enum_put(struct snd_kcontrol *kcontrol,
10848 struct snd_ctl_elem_value *ucontrol)
10849{
10850 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
10851 struct alc_spec *spec = codec->spec;
10852 int ret;
10853
Takashi Iwai54cbc9a2008-10-31 15:24:04 +010010854 ret = alc_mux_enum_put(kcontrol, ucontrol);
Takashi Iwaibb9f76c2008-03-12 12:51:09 +010010855 if (!ret)
10856 return 0;
10857 /* reprogram the HP pin as mic or HP according to the input source */
10858 snd_hda_codec_write_cache(codec, 0x15, 0,
10859 AC_VERB_SET_PIN_WIDGET_CONTROL,
10860 spec->cur_mux[0] ? PIN_VREF80 : PIN_HP);
10861 alc262_ultra_automute(codec); /* mute/unmute HP */
10862 return ret;
10863}
10864
10865static struct snd_kcontrol_new alc262_ultra_capture_mixer[] = {
10866 HDA_CODEC_VOLUME("Capture Volume", 0x07, 0x0, HDA_INPUT),
10867 HDA_CODEC_MUTE("Capture Switch", 0x07, 0x0, HDA_INPUT),
10868 {
10869 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
10870 .name = "Capture Source",
Takashi Iwai54cbc9a2008-10-31 15:24:04 +010010871 .info = alc_mux_enum_info,
10872 .get = alc_mux_enum_get,
Takashi Iwaibb9f76c2008-03-12 12:51:09 +010010873 .put = alc262_ultra_mux_enum_put,
10874 },
10875 { } /* end */
10876};
10877
Takashi Iwaic3fc1f52009-08-25 16:08:47 +020010878/* We use two mixers depending on the output pin; 0x16 is a mono output
10879 * and thus it's bound with a different mixer.
10880 * This function returns which mixer amp should be used.
10881 */
10882static int alc262_check_volbit(hda_nid_t nid)
10883{
10884 if (!nid)
10885 return 0;
10886 else if (nid == 0x16)
10887 return 2;
10888 else
10889 return 1;
10890}
10891
10892static int alc262_add_out_vol_ctl(struct alc_spec *spec, hda_nid_t nid,
10893 const char *pfx, int *vbits)
10894{
Takashi Iwaic3fc1f52009-08-25 16:08:47 +020010895 unsigned long val;
10896 int vbit;
10897
10898 vbit = alc262_check_volbit(nid);
10899 if (!vbit)
10900 return 0;
10901 if (*vbits & vbit) /* a volume control for this mixer already there */
10902 return 0;
10903 *vbits |= vbit;
Takashi Iwaic3fc1f52009-08-25 16:08:47 +020010904 if (vbit == 2)
10905 val = HDA_COMPOSE_AMP_VAL(0x0e, 2, 0, HDA_OUTPUT);
10906 else
10907 val = HDA_COMPOSE_AMP_VAL(0x0c, 3, 0, HDA_OUTPUT);
Takashi Iwai0afe5f82009-10-02 09:20:00 +020010908 return add_pb_vol_ctrl(spec, ALC_CTL_WIDGET_VOL, pfx, val);
Takashi Iwaic3fc1f52009-08-25 16:08:47 +020010909}
10910
10911static int alc262_add_out_sw_ctl(struct alc_spec *spec, hda_nid_t nid,
10912 const char *pfx)
10913{
Takashi Iwaic3fc1f52009-08-25 16:08:47 +020010914 unsigned long val;
10915
10916 if (!nid)
10917 return 0;
Takashi Iwaic3fc1f52009-08-25 16:08:47 +020010918 if (nid == 0x16)
10919 val = HDA_COMPOSE_AMP_VAL(nid, 2, 0, HDA_OUTPUT);
10920 else
10921 val = HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_OUTPUT);
Takashi Iwai0afe5f82009-10-02 09:20:00 +020010922 return add_pb_sw_ctrl(spec, ALC_CTL_WIDGET_MUTE, pfx, val);
Takashi Iwaic3fc1f52009-08-25 16:08:47 +020010923}
10924
Kailang Yangdf694da2005-12-05 19:42:22 +010010925/* add playback controls from the parsed DAC table */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020010926static int alc262_auto_create_multi_out_ctls(struct alc_spec *spec,
10927 const struct auto_pin_cfg *cfg)
Kailang Yangdf694da2005-12-05 19:42:22 +010010928{
Takashi Iwaic3fc1f52009-08-25 16:08:47 +020010929 const char *pfx;
10930 int vbits;
Kailang Yangdf694da2005-12-05 19:42:22 +010010931 int err;
10932
10933 spec->multiout.num_dacs = 1; /* only use one dac */
10934 spec->multiout.dac_nids = spec->private_dac_nids;
10935 spec->multiout.dac_nids[0] = 2;
10936
Takashi Iwaic3fc1f52009-08-25 16:08:47 +020010937 if (!cfg->speaker_pins[0] && !cfg->hp_pins[0])
10938 pfx = "Master";
10939 else if (cfg->line_out_type == AUTO_PIN_SPEAKER_OUT)
10940 pfx = "Speaker";
10941 else
10942 pfx = "Front";
10943 err = alc262_add_out_sw_ctl(spec, cfg->line_out_pins[0], pfx);
10944 if (err < 0)
10945 return err;
10946 err = alc262_add_out_sw_ctl(spec, cfg->speaker_pins[0], "Speaker");
10947 if (err < 0)
10948 return err;
10949 err = alc262_add_out_sw_ctl(spec, cfg->hp_pins[0], "Headphone");
10950 if (err < 0)
10951 return err;
Kailang Yangdf694da2005-12-05 19:42:22 +010010952
Takashi Iwaic3fc1f52009-08-25 16:08:47 +020010953 vbits = alc262_check_volbit(cfg->line_out_pins[0]) |
10954 alc262_check_volbit(cfg->speaker_pins[0]) |
10955 alc262_check_volbit(cfg->hp_pins[0]);
10956 if (vbits == 1 || vbits == 2)
10957 pfx = "Master"; /* only one mixer is used */
10958 else if (cfg->line_out_type == AUTO_PIN_SPEAKER_OUT)
10959 pfx = "Speaker";
10960 else
10961 pfx = "Front";
10962 vbits = 0;
10963 err = alc262_add_out_vol_ctl(spec, cfg->line_out_pins[0], pfx, &vbits);
10964 if (err < 0)
10965 return err;
10966 err = alc262_add_out_vol_ctl(spec, cfg->speaker_pins[0], "Speaker",
10967 &vbits);
10968 if (err < 0)
10969 return err;
10970 err = alc262_add_out_vol_ctl(spec, cfg->hp_pins[0], "Headphone",
10971 &vbits);
10972 if (err < 0)
10973 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020010974 return 0;
Kailang Yangdf694da2005-12-05 19:42:22 +010010975}
10976
Takashi Iwai05f5f472009-08-25 13:10:18 +020010977#define alc262_auto_create_input_ctls \
10978 alc880_auto_create_input_ctls
Kailang Yangdf694da2005-12-05 19:42:22 +010010979
10980/*
10981 * generic initialization of ADC, input mixers and output mixers
10982 */
10983static struct hda_verb alc262_volume_init_verbs[] = {
10984 /*
10985 * Unmute ADC0-2 and set the default input to mic-in
10986 */
10987 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
10988 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10989 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
10990 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10991 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
10992 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
10993
Takashi Iwaicb53c622007-08-10 17:21:45 +020010994 /* Mute input amps (CD, Line In, Mic 1 & Mic 2) of the analog-loopback
Kailang Yangdf694da2005-12-05 19:42:22 +010010995 * mixer widget
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020010996 * Note: PASD motherboards uses the Line In 2 as the input for
10997 * front panel mic (mic 2)
Kailang Yangdf694da2005-12-05 19:42:22 +010010998 */
10999 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +020011000 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
11001 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
11002 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
11003 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
11004 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Kailang Yangdf694da2005-12-05 19:42:22 +010011005
11006 /*
11007 * Set up output mixers (0x0c - 0x0f)
11008 */
11009 /* set vol=0 to output mixers */
11010 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
11011 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
11012 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
Kailang Yangea1fb292008-08-26 12:58:38 +020011013
Kailang Yangdf694da2005-12-05 19:42:22 +010011014 /* set up input amps for analog loopback */
11015 /* Amp Indices: DAC = 0, mixer = 1 */
11016 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11017 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11018 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11019 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11020 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11021 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11022
11023 /* FIXME: use matrix-type input source selection */
11024 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
11025 /* Input mixer1: unmute Mic, F-Mic, Line, CD inputs */
11026 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
11027 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
11028 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
11029 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
11030 /* Input mixer2 */
11031 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
11032 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
11033 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
11034 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
11035 /* Input mixer3 */
11036 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
11037 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
11038 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
11039 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
11040
11041 { }
11042};
11043
Takashi Iwai9c7f8522006-06-28 15:08:22 +020011044static struct hda_verb alc262_HP_BPC_init_verbs[] = {
11045 /*
11046 * Unmute ADC0-2 and set the default input to mic-in
11047 */
11048 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
11049 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11050 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
11051 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11052 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
11053 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11054
Takashi Iwaicb53c622007-08-10 17:21:45 +020011055 /* Mute input amps (CD, Line In, Mic 1 & Mic 2) of the analog-loopback
Takashi Iwai9c7f8522006-06-28 15:08:22 +020011056 * mixer widget
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011057 * Note: PASD motherboards uses the Line In 2 as the input for
11058 * front panel mic (mic 2)
Takashi Iwai9c7f8522006-06-28 15:08:22 +020011059 */
11060 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +020011061 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
11062 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
11063 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
11064 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
11065 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
11066 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(5)},
11067 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(6)},
Kailang Yangea1fb292008-08-26 12:58:38 +020011068
Takashi Iwai9c7f8522006-06-28 15:08:22 +020011069 /*
11070 * Set up output mixers (0x0c - 0x0e)
11071 */
11072 /* set vol=0 to output mixers */
11073 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
11074 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
11075 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
11076
11077 /* set up input amps for analog loopback */
11078 /* Amp Indices: DAC = 0, mixer = 1 */
11079 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11080 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11081 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11082 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11083 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11084 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11085
Takashi Iwaice875f02008-01-28 18:17:43 +010011086 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
Takashi Iwai9c7f8522006-06-28 15:08:22 +020011087 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
11088 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
11089
11090 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
11091 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
11092
11093 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
11094 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
11095
11096 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
11097 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
11098 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
11099 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
11100 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
11101
Jaroslav Kysela0e4835c2009-05-25 16:44:20 +020011102 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +020011103 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000 },
11104 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000 },
Jaroslav Kysela0e4835c2009-05-25 16:44:20 +020011105 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +020011106 {0x1c, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000 },
11107 {0x1d, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000 },
11108
11109
11110 /* FIXME: use matrix-type input source selection */
Jaroslav Kysela0e4835c2009-05-25 16:44:20 +020011111 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 0b, 12 */
11112 /* Input mixer1: only unmute Mic */
Takashi Iwai9c7f8522006-06-28 15:08:22 +020011113 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
Jaroslav Kysela0e4835c2009-05-25 16:44:20 +020011114 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x01 << 8))},
11115 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
11116 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
11117 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
11118 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x05 << 8))},
11119 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x06 << 8))},
11120 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x07 << 8))},
11121 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x08 << 8))},
Takashi Iwai9c7f8522006-06-28 15:08:22 +020011122 /* Input mixer2 */
11123 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
Jaroslav Kysela0e4835c2009-05-25 16:44:20 +020011124 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x01 << 8))},
11125 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
11126 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
11127 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
11128 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x05 << 8))},
11129 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x06 << 8))},
11130 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x07 << 8))},
11131 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x08 << 8))},
Takashi Iwai9c7f8522006-06-28 15:08:22 +020011132 /* Input mixer3 */
11133 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
Jaroslav Kysela0e4835c2009-05-25 16:44:20 +020011134 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x01 << 8))},
11135 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8))},
11136 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x03 << 8))},
11137 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x04 << 8))},
11138 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x05 << 8))},
11139 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x06 << 8))},
11140 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x07 << 8))},
11141 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x08 << 8))},
Takashi Iwai9c7f8522006-06-28 15:08:22 +020011142
Takashi Iwaice875f02008-01-28 18:17:43 +010011143 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
11144
Takashi Iwai9c7f8522006-06-28 15:08:22 +020011145 { }
11146};
11147
Kailang Yangcd7509a2007-01-26 18:33:17 +010011148static struct hda_verb alc262_HP_BPC_WildWest_init_verbs[] = {
11149 /*
11150 * Unmute ADC0-2 and set the default input to mic-in
11151 */
11152 {0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
11153 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11154 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
11155 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11156 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
11157 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11158
Takashi Iwaicb53c622007-08-10 17:21:45 +020011159 /* Mute input amps (CD, Line In, Mic 1 & Mic 2) of the analog-loopback
Kailang Yangcd7509a2007-01-26 18:33:17 +010011160 * mixer widget
11161 * Note: PASD motherboards uses the Line In 2 as the input for front
11162 * panel mic (mic 2)
11163 */
11164 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +020011165 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
11166 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
11167 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
11168 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
11169 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
11170 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(5)},
11171 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(6)},
11172 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(7)},
Kailang Yangcd7509a2007-01-26 18:33:17 +010011173 /*
11174 * Set up output mixers (0x0c - 0x0e)
11175 */
11176 /* set vol=0 to output mixers */
11177 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
11178 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
11179 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
11180
11181 /* set up input amps for analog loopback */
11182 /* Amp Indices: DAC = 0, mixer = 1 */
11183 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11184 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11185 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11186 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11187 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
11188 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
11189
11190
11191 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP }, /* HP */
11192 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT }, /* Mono */
11193 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 }, /* rear MIC */
11194 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN }, /* Line in */
11195 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 }, /* Front MIC */
11196 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT }, /* Line out */
11197 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN }, /* CD in */
11198
11199 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
11200 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
11201
11202 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
11203 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01},
11204
11205 /* {0x14, AC_VERB_SET_AMP_GAIN_MUTE, 0x7023 }, */
11206 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000 },
11207 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000 },
11208 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, 0x7023 },
11209 {0x1c, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000 },
11210 {0x1d, AC_VERB_SET_AMP_GAIN_MUTE, 0x7000 },
11211
11212 /* FIXME: use matrix-type input source selection */
11213 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
11214 /* Input mixer1: unmute Mic, F-Mic, Line, CD inputs */
11215 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))}, /*rear MIC*/
11216 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))}, /*Line in*/
11217 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x02 << 8))}, /*F MIC*/
11218 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x03 << 8))}, /*Front*/
11219 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x04 << 8))}, /*CD*/
11220 /* {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x06 << 8))}, */
11221 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x07 << 8))}, /*HP*/
11222 /* Input mixer2 */
11223 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
11224 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
11225 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x02 << 8))},
11226 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x03 << 8))},
11227 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x04 << 8))},
11228 /* {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x06 << 8))}, */
11229 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x07 << 8))},
11230 /* Input mixer3 */
11231 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x00 << 8))},
11232 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8))},
11233 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x02 << 8))},
11234 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x03 << 8))},
11235 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x04 << 8))},
11236 /* {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x06 << 8))}, */
11237 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x07 << 8))},
11238
Takashi Iwaice875f02008-01-28 18:17:43 +010011239 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
11240
Kailang Yangcd7509a2007-01-26 18:33:17 +010011241 { }
11242};
11243
Hiroshi Miura9f99a632008-08-28 16:09:06 +020011244static struct hda_verb alc262_toshiba_rx1_unsol_verbs[] = {
11245
11246 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT }, /* Front Speaker */
11247 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
11248 {0x14, AC_VERB_SET_CONNECT_SEL, 0x01},
11249
11250 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 }, /* MIC jack */
11251 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 }, /* Front MIC */
11252 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0) },
11253 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0) },
11254
11255 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP }, /* HP jack */
11256 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
11257 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
11258 {}
11259};
11260
11261
Takashi Iwaicb53c622007-08-10 17:21:45 +020011262#ifdef CONFIG_SND_HDA_POWER_SAVE
11263#define alc262_loopbacks alc880_loopbacks
11264#endif
11265
Sasha Alexandrdef319f2009-06-16 16:00:15 -040011266/* pcm configuration: identical with ALC880 */
Kailang Yangdf694da2005-12-05 19:42:22 +010011267#define alc262_pcm_analog_playback alc880_pcm_analog_playback
11268#define alc262_pcm_analog_capture alc880_pcm_analog_capture
11269#define alc262_pcm_digital_playback alc880_pcm_digital_playback
11270#define alc262_pcm_digital_capture alc880_pcm_digital_capture
11271
11272/*
11273 * BIOS auto configuration
11274 */
11275static int alc262_parse_auto_config(struct hda_codec *codec)
11276{
11277 struct alc_spec *spec = codec->spec;
11278 int err;
11279 static hda_nid_t alc262_ignore[] = { 0x1d, 0 };
11280
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011281 err = snd_hda_parse_pin_def_config(codec, &spec->autocfg,
11282 alc262_ignore);
11283 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +010011284 return err;
Takashi Iwaie64f14f2009-01-20 18:32:55 +010011285 if (!spec->autocfg.line_outs) {
Takashi Iwai0852d7a2009-02-11 11:35:15 +010011286 if (spec->autocfg.dig_outs || spec->autocfg.dig_in_pin) {
Takashi Iwaie64f14f2009-01-20 18:32:55 +010011287 spec->multiout.max_channels = 2;
11288 spec->no_analog = 1;
11289 goto dig_only;
11290 }
Kailang Yangdf694da2005-12-05 19:42:22 +010011291 return 0; /* can't find valid BIOS pin config */
Takashi Iwaie64f14f2009-01-20 18:32:55 +010011292 }
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011293 err = alc262_auto_create_multi_out_ctls(spec, &spec->autocfg);
11294 if (err < 0)
11295 return err;
Takashi Iwai05f5f472009-08-25 13:10:18 +020011296 err = alc262_auto_create_input_ctls(codec, &spec->autocfg);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011297 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +010011298 return err;
11299
11300 spec->multiout.max_channels = spec->multiout.num_dacs * 2;
11301
Takashi Iwaie64f14f2009-01-20 18:32:55 +010011302 dig_only:
Takashi Iwai0852d7a2009-02-11 11:35:15 +010011303 if (spec->autocfg.dig_outs) {
Kailang Yangdf694da2005-12-05 19:42:22 +010011304 spec->multiout.dig_out_nid = ALC262_DIGOUT_NID;
Takashi Iwai0852d7a2009-02-11 11:35:15 +010011305 spec->dig_out_type = spec->autocfg.dig_out_type[0];
Takashi Iwaie64f14f2009-01-20 18:32:55 +010011306 }
Kailang Yangdf694da2005-12-05 19:42:22 +010011307 if (spec->autocfg.dig_in_pin)
11308 spec->dig_in_nid = ALC262_DIGIN_NID;
11309
Takashi Iwai603c4012008-07-30 15:01:44 +020011310 if (spec->kctls.list)
Takashi Iwaid88897e2008-10-31 15:01:37 +010011311 add_mixer(spec, spec->kctls.list);
Kailang Yangdf694da2005-12-05 19:42:22 +010011312
Takashi Iwaid88897e2008-10-31 15:01:37 +010011313 add_verb(spec, alc262_volume_init_verbs);
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +020011314 spec->num_mux_defs = 1;
Herton Ronaldo Krzesinski61b9b9b2009-01-28 09:16:33 -020011315 spec->input_mux = &spec->private_imux[0];
Kailang Yangdf694da2005-12-05 19:42:22 +010011316
Takashi Iwai776e1842007-08-29 15:07:11 +020011317 err = alc_auto_add_mic_boost(codec);
11318 if (err < 0)
11319 return err;
11320
Takashi Iwai4a79ba32009-04-22 16:31:35 +020011321 alc_ssid_check(codec, 0x15, 0x14, 0x1b);
11322
Kailang Yangdf694da2005-12-05 19:42:22 +010011323 return 1;
11324}
11325
11326#define alc262_auto_init_multi_out alc882_auto_init_multi_out
11327#define alc262_auto_init_hp_out alc882_auto_init_hp_out
11328#define alc262_auto_init_analog_input alc882_auto_init_analog_input
Takashi Iwaif511b012008-08-15 16:46:42 +020011329#define alc262_auto_init_input_src alc882_auto_init_input_src
Kailang Yangdf694da2005-12-05 19:42:22 +010011330
11331
11332/* init callback for auto-configuration model -- overriding the default init */
Takashi Iwaiae6b8132006-03-03 16:47:17 +010011333static void alc262_auto_init(struct hda_codec *codec)
Kailang Yangdf694da2005-12-05 19:42:22 +010011334{
Takashi Iwaif6c7e542008-02-12 18:32:23 +010011335 struct alc_spec *spec = codec->spec;
Kailang Yangdf694da2005-12-05 19:42:22 +010011336 alc262_auto_init_multi_out(codec);
11337 alc262_auto_init_hp_out(codec);
11338 alc262_auto_init_analog_input(codec);
Takashi Iwaif511b012008-08-15 16:46:42 +020011339 alc262_auto_init_input_src(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +010011340 if (spec->unsol_event)
Kailang Yang7fb0d782008-10-15 11:12:35 +020011341 alc_inithook(codec);
Kailang Yangdf694da2005-12-05 19:42:22 +010011342}
11343
11344/*
11345 * configuration and preset
11346 */
Takashi Iwaif5fcc132006-11-24 17:07:44 +010011347static const char *alc262_models[ALC262_MODEL_LAST] = {
11348 [ALC262_BASIC] = "basic",
11349 [ALC262_HIPPO] = "hippo",
11350 [ALC262_HIPPO_1] = "hippo_1",
11351 [ALC262_FUJITSU] = "fujitsu",
11352 [ALC262_HP_BPC] = "hp-bpc",
Kailang Yangcd7509a2007-01-26 18:33:17 +010011353 [ALC262_HP_BPC_D7000_WL]= "hp-bpc-d7000",
Takashi Iwai61dc35d2007-11-14 12:26:44 +010011354 [ALC262_HP_TC_T5735] = "hp-tc-t5735",
Kailang Yang8c427222008-01-10 13:03:59 +010011355 [ALC262_HP_RP5700] = "hp-rp5700",
Takashi Iwaif5fcc132006-11-24 17:07:44 +010011356 [ALC262_BENQ_ED8] = "benq",
Takashi Iwai0f405022007-07-06 12:24:11 +020011357 [ALC262_BENQ_T31] = "benq-t31",
11358 [ALC262_SONY_ASSAMD] = "sony-assamd",
Takashi Iwai2922c9a2008-08-27 18:12:42 +020011359 [ALC262_TOSHIBA_S06] = "toshiba-s06",
Hiroshi Miura9f99a632008-08-28 16:09:06 +020011360 [ALC262_TOSHIBA_RX1] = "toshiba-rx1",
Tobin Davisf651b502007-10-26 12:40:47 +020011361 [ALC262_ULTRA] = "ultra",
Jiang zhe0e31daf2008-03-20 12:12:39 +010011362 [ALC262_LENOVO_3000] = "lenovo-3000",
Pascal Terjane8f9ae22008-08-04 14:36:05 +020011363 [ALC262_NEC] = "nec",
Tony Vroonba340e82009-02-02 19:01:30 +000011364 [ALC262_TYAN] = "tyan",
Takashi Iwaif5fcc132006-11-24 17:07:44 +010011365 [ALC262_AUTO] = "auto",
11366};
11367
11368static struct snd_pci_quirk alc262_cfg_tbl[] = {
11369 SND_PCI_QUIRK(0x1002, 0x437b, "Hippo", ALC262_HIPPO),
Pascal Terjane8f9ae22008-08-04 14:36:05 +020011370 SND_PCI_QUIRK(0x1033, 0x8895, "NEC Versa S9100", ALC262_NEC),
Takashi Iwaidea0a502009-02-09 17:14:52 +010011371 SND_PCI_QUIRK_MASK(0x103c, 0xff00, 0x1200, "HP xw series",
11372 ALC262_HP_BPC),
11373 SND_PCI_QUIRK_MASK(0x103c, 0xff00, 0x1300, "HP xw series",
11374 ALC262_HP_BPC),
Takashi Iwai53eff7e2009-02-27 17:49:44 +010011375 SND_PCI_QUIRK_MASK(0x103c, 0xff00, 0x1700, "HP xw series",
11376 ALC262_HP_BPC),
Kailang Yangcd7509a2007-01-26 18:33:17 +010011377 SND_PCI_QUIRK(0x103c, 0x2800, "HP D7000", ALC262_HP_BPC_D7000_WL),
Kailang Yangcd7509a2007-01-26 18:33:17 +010011378 SND_PCI_QUIRK(0x103c, 0x2801, "HP D7000", ALC262_HP_BPC_D7000_WF),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010011379 SND_PCI_QUIRK(0x103c, 0x2802, "HP D7000", ALC262_HP_BPC_D7000_WL),
Kailang Yangcd7509a2007-01-26 18:33:17 +010011380 SND_PCI_QUIRK(0x103c, 0x2803, "HP D7000", ALC262_HP_BPC_D7000_WF),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010011381 SND_PCI_QUIRK(0x103c, 0x2804, "HP D7000", ALC262_HP_BPC_D7000_WL),
Kailang Yangcd7509a2007-01-26 18:33:17 +010011382 SND_PCI_QUIRK(0x103c, 0x2805, "HP D7000", ALC262_HP_BPC_D7000_WF),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010011383 SND_PCI_QUIRK(0x103c, 0x2806, "HP D7000", ALC262_HP_BPC_D7000_WL),
Kailang Yangcd7509a2007-01-26 18:33:17 +010011384 SND_PCI_QUIRK(0x103c, 0x2807, "HP D7000", ALC262_HP_BPC_D7000_WF),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010011385 SND_PCI_QUIRK(0x103c, 0x280c, "HP xw4400", ALC262_HP_BPC),
11386 SND_PCI_QUIRK(0x103c, 0x3014, "HP xw6400", ALC262_HP_BPC),
11387 SND_PCI_QUIRK(0x103c, 0x3015, "HP xw8400", ALC262_HP_BPC),
Kailang Yang66d2a9d2007-11-14 12:06:21 +010011388 SND_PCI_QUIRK(0x103c, 0x302f, "HP Thin Client T5735",
11389 ALC262_HP_TC_T5735),
Kailang Yang8c427222008-01-10 13:03:59 +010011390 SND_PCI_QUIRK(0x103c, 0x2817, "HP RP5700", ALC262_HP_RP5700),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010011391 SND_PCI_QUIRK(0x104d, 0x1f00, "Sony ASSAMD", ALC262_SONY_ASSAMD),
Takashi Iwaif5fcc132006-11-24 17:07:44 +010011392 SND_PCI_QUIRK(0x104d, 0x8203, "Sony UX-90", ALC262_HIPPO),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010011393 SND_PCI_QUIRK(0x104d, 0x820f, "Sony ASSAMD", ALC262_SONY_ASSAMD),
Takashi Iwaibd6afe32009-03-04 11:30:25 +010011394 SND_PCI_QUIRK(0x104d, 0x9016, "Sony VAIO", ALC262_AUTO), /* dig-only */
Takashi Iwai376b5082009-06-22 11:03:13 +020011395 SND_PCI_QUIRK(0x104d, 0x9025, "Sony VAIO Z21MN", ALC262_TOSHIBA_S06),
Daniel T Chen95491d92009-11-08 19:03:55 -050011396 SND_PCI_QUIRK(0x104d, 0x9035, "Sony VAIO VGN-FW170J", ALC262_AUTO),
Takashi Iwai12929ba2009-11-17 15:58:35 +010011397 SND_PCI_QUIRK(0x104d, 0x9047, "Sony VAIO Type G", ALC262_AUTO),
Takashi Iwaic5b51652009-11-17 16:01:58 +010011398#if 0 /* disable the quirk since model=auto works better in recent versions */
Takashi Iwaif872a912009-02-26 00:57:01 +010011399 SND_PCI_QUIRK_MASK(0x104d, 0xff00, 0x9000, "Sony VAIO",
11400 ALC262_SONY_ASSAMD),
Takashi Iwaic5b51652009-11-17 16:01:58 +010011401#endif
Akio Idehara36ca6e12008-06-09 22:57:40 +090011402 SND_PCI_QUIRK(0x1179, 0x0001, "Toshiba dynabook SS RX1",
Hiroshi Miura9f99a632008-08-28 16:09:06 +020011403 ALC262_TOSHIBA_RX1),
Kailang Yang80ffe862008-10-15 11:23:27 +020011404 SND_PCI_QUIRK(0x1179, 0xff7b, "Toshiba S06", ALC262_TOSHIBA_S06),
Takashi Iwaif5fcc132006-11-24 17:07:44 +010011405 SND_PCI_QUIRK(0x10cf, 0x1397, "Fujitsu", ALC262_FUJITSU),
Tony Vroon3f1eeae2008-02-25 16:44:13 +010011406 SND_PCI_QUIRK(0x10cf, 0x142d, "Fujitsu Lifebook E8410", ALC262_FUJITSU),
Tony Vroonba340e82009-02-02 19:01:30 +000011407 SND_PCI_QUIRK(0x10f1, 0x2915, "Tyan Thunder n6650W", ALC262_TYAN),
Takashi Iwaidea0a502009-02-09 17:14:52 +010011408 SND_PCI_QUIRK_MASK(0x144d, 0xff00, 0xc032, "Samsung Q1",
11409 ALC262_ULTRA),
Luke Yelavich3e420e72008-12-16 12:37:47 +110011410 SND_PCI_QUIRK(0x144d, 0xc510, "Samsung Q45", ALC262_HIPPO),
Jiang zhe0e31daf2008-03-20 12:12:39 +010011411 SND_PCI_QUIRK(0x17aa, 0x384e, "Lenovo 3000 y410", ALC262_LENOVO_3000),
Takashi Iwaif5fcc132006-11-24 17:07:44 +010011412 SND_PCI_QUIRK(0x17ff, 0x0560, "Benq ED8", ALC262_BENQ_ED8),
Kailang Yang83c34212007-07-05 11:43:05 +020011413 SND_PCI_QUIRK(0x17ff, 0x058d, "Benq T31-16", ALC262_BENQ_T31),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010011414 SND_PCI_QUIRK(0x17ff, 0x058f, "Benq Hippo", ALC262_HIPPO_1),
Kailang Yangdf694da2005-12-05 19:42:22 +010011415 {}
11416};
11417
11418static struct alc_config_preset alc262_presets[] = {
11419 [ALC262_BASIC] = {
11420 .mixers = { alc262_base_mixer },
11421 .init_verbs = { alc262_init_verbs },
11422 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
11423 .dac_nids = alc262_dac_nids,
11424 .hp_nid = 0x03,
11425 .num_channel_mode = ARRAY_SIZE(alc262_modes),
11426 .channel_mode = alc262_modes,
Takashi Iwaia3bcba32005-12-06 19:05:29 +010011427 .input_mux = &alc262_capture_source,
Kailang Yangdf694da2005-12-05 19:42:22 +010011428 },
Kailang Yangccc656c2006-10-17 12:32:26 +020011429 [ALC262_HIPPO] = {
Takashi Iwai42171c12009-05-08 14:11:43 +020011430 .mixers = { alc262_hippo_mixer },
Wu Fengguang6732bd02009-07-30 09:19:14 +020011431 .init_verbs = { alc262_init_verbs, alc_hp15_unsol_verbs},
Kailang Yangccc656c2006-10-17 12:32:26 +020011432 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
11433 .dac_nids = alc262_dac_nids,
11434 .hp_nid = 0x03,
11435 .dig_out_nid = ALC262_DIGOUT_NID,
11436 .num_channel_mode = ARRAY_SIZE(alc262_modes),
11437 .channel_mode = alc262_modes,
11438 .input_mux = &alc262_capture_source,
11439 .unsol_event = alc262_hippo_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020011440 .setup = alc262_hippo_setup,
11441 .init_hook = alc262_hippo_automute,
Kailang Yangccc656c2006-10-17 12:32:26 +020011442 },
11443 [ALC262_HIPPO_1] = {
11444 .mixers = { alc262_hippo1_mixer },
11445 .init_verbs = { alc262_init_verbs, alc262_hippo1_unsol_verbs},
11446 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
11447 .dac_nids = alc262_dac_nids,
11448 .hp_nid = 0x02,
11449 .dig_out_nid = ALC262_DIGOUT_NID,
11450 .num_channel_mode = ARRAY_SIZE(alc262_modes),
11451 .channel_mode = alc262_modes,
11452 .input_mux = &alc262_capture_source,
Takashi Iwai42171c12009-05-08 14:11:43 +020011453 .unsol_event = alc262_hippo_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020011454 .setup = alc262_hippo1_setup,
11455 .init_hook = alc262_hippo_automute,
Kailang Yangccc656c2006-10-17 12:32:26 +020011456 },
Takashi Iwai834be882006-03-01 14:16:17 +010011457 [ALC262_FUJITSU] = {
11458 .mixers = { alc262_fujitsu_mixer },
Takashi Iwai39d3ed32007-10-12 15:03:48 +020011459 .init_verbs = { alc262_init_verbs, alc262_EAPD_verbs,
11460 alc262_fujitsu_unsol_verbs },
Takashi Iwai834be882006-03-01 14:16:17 +010011461 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
11462 .dac_nids = alc262_dac_nids,
11463 .hp_nid = 0x03,
11464 .dig_out_nid = ALC262_DIGOUT_NID,
11465 .num_channel_mode = ARRAY_SIZE(alc262_modes),
11466 .channel_mode = alc262_modes,
11467 .input_mux = &alc262_fujitsu_capture_source,
Takashi Iwaiae6b8132006-03-03 16:47:17 +010011468 .unsol_event = alc262_fujitsu_unsol_event,
Takashi Iwaiebc7a402008-05-20 09:23:05 +020011469 .init_hook = alc262_fujitsu_init_hook,
Takashi Iwai834be882006-03-01 14:16:17 +010011470 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +020011471 [ALC262_HP_BPC] = {
11472 .mixers = { alc262_HP_BPC_mixer },
11473 .init_verbs = { alc262_HP_BPC_init_verbs },
11474 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
11475 .dac_nids = alc262_dac_nids,
11476 .hp_nid = 0x03,
11477 .num_channel_mode = ARRAY_SIZE(alc262_modes),
11478 .channel_mode = alc262_modes,
11479 .input_mux = &alc262_HP_capture_source,
Takashi Iwaice875f02008-01-28 18:17:43 +010011480 .unsol_event = alc262_hp_bpc_unsol_event,
11481 .init_hook = alc262_hp_bpc_automute,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011482 },
Kailang Yangcd7509a2007-01-26 18:33:17 +010011483 [ALC262_HP_BPC_D7000_WF] = {
11484 .mixers = { alc262_HP_BPC_WildWest_mixer },
11485 .init_verbs = { alc262_HP_BPC_WildWest_init_verbs },
11486 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
11487 .dac_nids = alc262_dac_nids,
11488 .hp_nid = 0x03,
11489 .num_channel_mode = ARRAY_SIZE(alc262_modes),
11490 .channel_mode = alc262_modes,
zhejiangaccbe492007-08-31 12:36:05 +020011491 .input_mux = &alc262_HP_D7000_capture_source,
Takashi Iwaice875f02008-01-28 18:17:43 +010011492 .unsol_event = alc262_hp_wildwest_unsol_event,
11493 .init_hook = alc262_hp_wildwest_automute,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011494 },
Kailang Yangcd7509a2007-01-26 18:33:17 +010011495 [ALC262_HP_BPC_D7000_WL] = {
11496 .mixers = { alc262_HP_BPC_WildWest_mixer,
11497 alc262_HP_BPC_WildWest_option_mixer },
11498 .init_verbs = { alc262_HP_BPC_WildWest_init_verbs },
11499 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
11500 .dac_nids = alc262_dac_nids,
11501 .hp_nid = 0x03,
11502 .num_channel_mode = ARRAY_SIZE(alc262_modes),
11503 .channel_mode = alc262_modes,
zhejiangaccbe492007-08-31 12:36:05 +020011504 .input_mux = &alc262_HP_D7000_capture_source,
Takashi Iwaice875f02008-01-28 18:17:43 +010011505 .unsol_event = alc262_hp_wildwest_unsol_event,
11506 .init_hook = alc262_hp_wildwest_automute,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011507 },
Kailang Yang66d2a9d2007-11-14 12:06:21 +010011508 [ALC262_HP_TC_T5735] = {
11509 .mixers = { alc262_hp_t5735_mixer },
11510 .init_verbs = { alc262_init_verbs, alc262_hp_t5735_verbs },
11511 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
11512 .dac_nids = alc262_dac_nids,
11513 .hp_nid = 0x03,
11514 .num_channel_mode = ARRAY_SIZE(alc262_modes),
11515 .channel_mode = alc262_modes,
11516 .input_mux = &alc262_capture_source,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020011517 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020011518 .setup = alc262_hp_t5735_setup,
11519 .init_hook = alc_automute_amp,
Kailang Yang8c427222008-01-10 13:03:59 +010011520 },
11521 [ALC262_HP_RP5700] = {
11522 .mixers = { alc262_hp_rp5700_mixer },
11523 .init_verbs = { alc262_init_verbs, alc262_hp_rp5700_verbs },
11524 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
11525 .dac_nids = alc262_dac_nids,
11526 .num_channel_mode = ARRAY_SIZE(alc262_modes),
11527 .channel_mode = alc262_modes,
11528 .input_mux = &alc262_hp_rp5700_capture_source,
Kailang Yang66d2a9d2007-11-14 12:06:21 +010011529 },
Takashi Iwai304dcaa2006-07-25 14:51:16 +020011530 [ALC262_BENQ_ED8] = {
11531 .mixers = { alc262_base_mixer },
11532 .init_verbs = { alc262_init_verbs, alc262_EAPD_verbs },
11533 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
11534 .dac_nids = alc262_dac_nids,
11535 .hp_nid = 0x03,
11536 .num_channel_mode = ARRAY_SIZE(alc262_modes),
11537 .channel_mode = alc262_modes,
11538 .input_mux = &alc262_capture_source,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011539 },
Kailang Yang272a5272007-05-14 11:00:38 +020011540 [ALC262_SONY_ASSAMD] = {
11541 .mixers = { alc262_sony_mixer },
11542 .init_verbs = { alc262_init_verbs, alc262_sony_unsol_verbs},
11543 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
11544 .dac_nids = alc262_dac_nids,
11545 .hp_nid = 0x02,
11546 .num_channel_mode = ARRAY_SIZE(alc262_modes),
11547 .channel_mode = alc262_modes,
11548 .input_mux = &alc262_capture_source,
11549 .unsol_event = alc262_hippo_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020011550 .setup = alc262_hippo_setup,
11551 .init_hook = alc262_hippo_automute,
Kailang Yang83c34212007-07-05 11:43:05 +020011552 },
11553 [ALC262_BENQ_T31] = {
11554 .mixers = { alc262_benq_t31_mixer },
Wu Fengguang6732bd02009-07-30 09:19:14 +020011555 .init_verbs = { alc262_init_verbs, alc262_benq_t31_EAPD_verbs,
11556 alc_hp15_unsol_verbs },
Kailang Yang83c34212007-07-05 11:43:05 +020011557 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
11558 .dac_nids = alc262_dac_nids,
11559 .hp_nid = 0x03,
11560 .num_channel_mode = ARRAY_SIZE(alc262_modes),
11561 .channel_mode = alc262_modes,
11562 .input_mux = &alc262_capture_source,
11563 .unsol_event = alc262_hippo_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020011564 .setup = alc262_hippo_setup,
11565 .init_hook = alc262_hippo_automute,
Kailang Yangea1fb292008-08-26 12:58:38 +020011566 },
Tobin Davisf651b502007-10-26 12:40:47 +020011567 [ALC262_ULTRA] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010011568 .mixers = { alc262_ultra_mixer },
11569 .cap_mixer = alc262_ultra_capture_mixer,
Takashi Iwaibb9f76c2008-03-12 12:51:09 +010011570 .init_verbs = { alc262_ultra_verbs },
Tobin Davisf651b502007-10-26 12:40:47 +020011571 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
11572 .dac_nids = alc262_dac_nids,
Tobin Davisf651b502007-10-26 12:40:47 +020011573 .num_channel_mode = ARRAY_SIZE(alc262_modes),
11574 .channel_mode = alc262_modes,
11575 .input_mux = &alc262_ultra_capture_source,
Takashi Iwaibb9f76c2008-03-12 12:51:09 +010011576 .adc_nids = alc262_adc_nids, /* ADC0 */
11577 .capsrc_nids = alc262_capsrc_nids,
11578 .num_adc_nids = 1, /* single ADC */
Tobin Davisf651b502007-10-26 12:40:47 +020011579 .unsol_event = alc262_ultra_unsol_event,
11580 .init_hook = alc262_ultra_automute,
11581 },
Jiang zhe0e31daf2008-03-20 12:12:39 +010011582 [ALC262_LENOVO_3000] = {
11583 .mixers = { alc262_lenovo_3000_mixer },
11584 .init_verbs = { alc262_init_verbs, alc262_EAPD_verbs,
11585 alc262_lenovo_3000_unsol_verbs },
11586 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
11587 .dac_nids = alc262_dac_nids,
11588 .hp_nid = 0x03,
11589 .dig_out_nid = ALC262_DIGOUT_NID,
11590 .num_channel_mode = ARRAY_SIZE(alc262_modes),
11591 .channel_mode = alc262_modes,
11592 .input_mux = &alc262_fujitsu_capture_source,
11593 .unsol_event = alc262_lenovo_3000_unsol_event,
11594 },
Pascal Terjane8f9ae22008-08-04 14:36:05 +020011595 [ALC262_NEC] = {
11596 .mixers = { alc262_nec_mixer },
11597 .init_verbs = { alc262_nec_verbs },
11598 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
11599 .dac_nids = alc262_dac_nids,
11600 .hp_nid = 0x03,
11601 .num_channel_mode = ARRAY_SIZE(alc262_modes),
11602 .channel_mode = alc262_modes,
11603 .input_mux = &alc262_capture_source,
11604 },
Kailang Yang4e555fe2008-08-26 13:05:55 +020011605 [ALC262_TOSHIBA_S06] = {
11606 .mixers = { alc262_toshiba_s06_mixer },
11607 .init_verbs = { alc262_init_verbs, alc262_toshiba_s06_verbs,
11608 alc262_eapd_verbs },
11609 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
11610 .capsrc_nids = alc262_dmic_capsrc_nids,
11611 .dac_nids = alc262_dac_nids,
11612 .adc_nids = alc262_dmic_adc_nids, /* ADC0 */
Takashi Iwaiae14ef62009-06-22 08:16:56 +020011613 .num_adc_nids = 1, /* single ADC */
Kailang Yang4e555fe2008-08-26 13:05:55 +020011614 .dig_out_nid = ALC262_DIGOUT_NID,
11615 .num_channel_mode = ARRAY_SIZE(alc262_modes),
11616 .channel_mode = alc262_modes,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020011617 .unsol_event = alc_sku_unsol_event,
11618 .setup = alc262_toshiba_s06_setup,
11619 .init_hook = alc_inithook,
Kailang Yang4e555fe2008-08-26 13:05:55 +020011620 },
Hiroshi Miura9f99a632008-08-28 16:09:06 +020011621 [ALC262_TOSHIBA_RX1] = {
11622 .mixers = { alc262_toshiba_rx1_mixer },
11623 .init_verbs = { alc262_init_verbs, alc262_toshiba_rx1_unsol_verbs },
11624 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
11625 .dac_nids = alc262_dac_nids,
11626 .hp_nid = 0x03,
11627 .num_channel_mode = ARRAY_SIZE(alc262_modes),
11628 .channel_mode = alc262_modes,
11629 .input_mux = &alc262_capture_source,
11630 .unsol_event = alc262_hippo_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020011631 .setup = alc262_hippo_setup,
11632 .init_hook = alc262_hippo_automute,
Hiroshi Miura9f99a632008-08-28 16:09:06 +020011633 },
Tony Vroonba340e82009-02-02 19:01:30 +000011634 [ALC262_TYAN] = {
11635 .mixers = { alc262_tyan_mixer },
11636 .init_verbs = { alc262_init_verbs, alc262_tyan_verbs},
11637 .num_dacs = ARRAY_SIZE(alc262_dac_nids),
11638 .dac_nids = alc262_dac_nids,
11639 .hp_nid = 0x02,
11640 .dig_out_nid = ALC262_DIGOUT_NID,
11641 .num_channel_mode = ARRAY_SIZE(alc262_modes),
11642 .channel_mode = alc262_modes,
11643 .input_mux = &alc262_capture_source,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020011644 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020011645 .setup = alc262_tyan_setup,
11646 .init_hook = alc_automute_amp,
Tony Vroonba340e82009-02-02 19:01:30 +000011647 },
Kailang Yangdf694da2005-12-05 19:42:22 +010011648};
11649
11650static int patch_alc262(struct hda_codec *codec)
11651{
11652 struct alc_spec *spec;
11653 int board_config;
11654 int err;
11655
Robert P. J. Daydc041e02006-12-19 14:44:15 +010011656 spec = kzalloc(sizeof(*spec), GFP_KERNEL);
Kailang Yangdf694da2005-12-05 19:42:22 +010011657 if (spec == NULL)
11658 return -ENOMEM;
11659
11660 codec->spec = spec;
11661#if 0
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011662 /* pshou 07/11/05 set a zero PCM sample to DAC when FIFO is
11663 * under-run
11664 */
Kailang Yangdf694da2005-12-05 19:42:22 +010011665 {
11666 int tmp;
11667 snd_hda_codec_write(codec, 0x1a, 0, AC_VERB_SET_COEF_INDEX, 7);
11668 tmp = snd_hda_codec_read(codec, 0x20, 0, AC_VERB_GET_PROC_COEF, 0);
11669 snd_hda_codec_write(codec, 0x1a, 0, AC_VERB_SET_COEF_INDEX, 7);
11670 snd_hda_codec_write(codec, 0x1a, 0, AC_VERB_SET_PROC_COEF, tmp | 0x80);
11671 }
11672#endif
11673
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +020011674 alc_fix_pll_init(codec, 0x20, 0x0a, 10);
11675
Takashi Iwaif5fcc132006-11-24 17:07:44 +010011676 board_config = snd_hda_check_board_config(codec, ALC262_MODEL_LAST,
11677 alc262_models,
11678 alc262_cfg_tbl);
Kailang Yangcd7509a2007-01-26 18:33:17 +010011679
Takashi Iwaif5fcc132006-11-24 17:07:44 +010011680 if (board_config < 0) {
Takashi Iwai9a11f1a2009-07-28 16:01:20 +020011681 printk(KERN_INFO "hda_codec: %s: BIOS auto-probing.\n",
11682 codec->chip_name);
Kailang Yangdf694da2005-12-05 19:42:22 +010011683 board_config = ALC262_AUTO;
11684 }
11685
11686 if (board_config == ALC262_AUTO) {
11687 /* automatic parse from the BIOS config */
11688 err = alc262_parse_auto_config(codec);
11689 if (err < 0) {
11690 alc_free(codec);
11691 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011692 } else if (!err) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +020011693 printk(KERN_INFO
11694 "hda_codec: Cannot set up configuration "
11695 "from BIOS. Using base mode...\n");
Kailang Yangdf694da2005-12-05 19:42:22 +010011696 board_config = ALC262_BASIC;
11697 }
11698 }
11699
Takashi Iwai07eba612009-02-19 08:06:35 +010011700 if (!spec->no_analog) {
11701 err = snd_hda_attach_beep_device(codec, 0x1);
11702 if (err < 0) {
11703 alc_free(codec);
11704 return err;
11705 }
Kusanagi Kouichi680cd532009-02-05 00:00:58 +090011706 }
11707
Kailang Yangdf694da2005-12-05 19:42:22 +010011708 if (board_config != ALC262_AUTO)
Takashi Iwaie9c364c2009-08-11 17:16:13 +020011709 setup_preset(codec, &alc262_presets[board_config]);
Kailang Yangdf694da2005-12-05 19:42:22 +010011710
Kailang Yangdf694da2005-12-05 19:42:22 +010011711 spec->stream_analog_playback = &alc262_pcm_analog_playback;
11712 spec->stream_analog_capture = &alc262_pcm_analog_capture;
Kailang Yangea1fb292008-08-26 12:58:38 +020011713
Kailang Yangdf694da2005-12-05 19:42:22 +010011714 spec->stream_digital_playback = &alc262_pcm_digital_playback;
11715 spec->stream_digital_capture = &alc262_pcm_digital_capture;
11716
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020011717 if (!spec->adc_nids && spec->input_mux) {
Takashi Iwai8c927b42009-06-22 10:56:54 +020011718 int i;
11719 /* check whether the digital-mic has to be supported */
11720 for (i = 0; i < spec->input_mux->num_items; i++) {
11721 if (spec->input_mux->items[i].index >= 9)
11722 break;
11723 }
11724 if (i < spec->input_mux->num_items) {
11725 /* use only ADC0 */
11726 spec->adc_nids = alc262_dmic_adc_nids;
11727 spec->num_adc_nids = 1;
11728 spec->capsrc_nids = alc262_dmic_capsrc_nids;
Kailang Yangdf694da2005-12-05 19:42:22 +010011729 } else {
Takashi Iwai8c927b42009-06-22 10:56:54 +020011730 /* all analog inputs */
11731 /* check whether NID 0x07 is valid */
11732 unsigned int wcap = get_wcaps(codec, 0x07);
11733
11734 /* get type */
Takashi Iwaia22d5432009-07-27 12:54:26 +020011735 wcap = get_wcaps_type(wcap);
Takashi Iwai8c927b42009-06-22 10:56:54 +020011736 if (wcap != AC_WID_AUD_IN) {
11737 spec->adc_nids = alc262_adc_nids_alt;
11738 spec->num_adc_nids =
11739 ARRAY_SIZE(alc262_adc_nids_alt);
11740 spec->capsrc_nids = alc262_capsrc_nids_alt;
11741 } else {
11742 spec->adc_nids = alc262_adc_nids;
11743 spec->num_adc_nids =
11744 ARRAY_SIZE(alc262_adc_nids);
11745 spec->capsrc_nids = alc262_capsrc_nids;
11746 }
Kailang Yangdf694da2005-12-05 19:42:22 +010011747 }
11748 }
Takashi Iwaie64f14f2009-01-20 18:32:55 +010011749 if (!spec->cap_mixer && !spec->no_analog)
Takashi Iwaib59bdf32009-08-11 09:47:30 +020011750 set_capture_mixer(codec);
Takashi Iwai07eba612009-02-19 08:06:35 +010011751 if (!spec->no_analog)
11752 set_beep_amp(spec, 0x0b, 0x05, HDA_INPUT);
Kailang Yangdf694da2005-12-05 19:42:22 +010011753
Takashi Iwai2134ea42008-01-10 16:53:55 +010011754 spec->vmaster_nid = 0x0c;
11755
Kailang Yangdf694da2005-12-05 19:42:22 +010011756 codec->patch_ops = alc_patch_ops;
11757 if (board_config == ALC262_AUTO)
Takashi Iwaiae6b8132006-03-03 16:47:17 +010011758 spec->init_hook = alc262_auto_init;
Takashi Iwaicb53c622007-08-10 17:21:45 +020011759#ifdef CONFIG_SND_HDA_POWER_SAVE
11760 if (!spec->loopback.amplist)
11761 spec->loopback.amplist = alc262_loopbacks;
11762#endif
Takashi Iwaidaead532008-11-28 12:55:36 +010011763 codec->proc_widget_hook = print_realtek_coef;
Kailang Yangea1fb292008-08-26 12:58:38 +020011764
Kailang Yangdf694da2005-12-05 19:42:22 +010011765 return 0;
11766}
11767
Kailang Yangdf694da2005-12-05 19:42:22 +010011768/*
Kailang Yanga361d842007-06-05 12:30:55 +020011769 * ALC268 channel source setting (2 channel)
11770 */
11771#define ALC268_DIGOUT_NID ALC880_DIGOUT_NID
11772#define alc268_modes alc260_modes
Kailang Yangea1fb292008-08-26 12:58:38 +020011773
Kailang Yanga361d842007-06-05 12:30:55 +020011774static hda_nid_t alc268_dac_nids[2] = {
11775 /* front, hp */
11776 0x02, 0x03
11777};
11778
11779static hda_nid_t alc268_adc_nids[2] = {
11780 /* ADC0-1 */
11781 0x08, 0x07
11782};
11783
11784static hda_nid_t alc268_adc_nids_alt[1] = {
11785 /* ADC0 */
11786 0x08
11787};
11788
Takashi Iwaie1406342008-02-11 18:32:32 +010011789static hda_nid_t alc268_capsrc_nids[2] = { 0x23, 0x24 };
11790
Kailang Yanga361d842007-06-05 12:30:55 +020011791static struct snd_kcontrol_new alc268_base_mixer[] = {
11792 /* output mixer control */
11793 HDA_CODEC_VOLUME("Front Playback Volume", 0x2, 0x0, HDA_OUTPUT),
11794 HDA_CODEC_MUTE("Front Playback Switch", 0x14, 0x0, HDA_OUTPUT),
11795 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x3, 0x0, HDA_OUTPUT),
11796 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
Takashi Iwai33bf17a2007-08-21 11:51:42 +020011797 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
11798 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
11799 HDA_CODEC_VOLUME("Line In Boost", 0x1a, 0, HDA_INPUT),
Kailang Yanga361d842007-06-05 12:30:55 +020011800 { }
11801};
11802
Takashi Iwai42171c12009-05-08 14:11:43 +020011803static struct snd_kcontrol_new alc268_toshiba_mixer[] = {
11804 /* output mixer control */
11805 HDA_CODEC_VOLUME("Front Playback Volume", 0x2, 0x0, HDA_OUTPUT),
11806 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x3, 0x0, HDA_OUTPUT),
11807 ALC262_HIPPO_MASTER_SWITCH,
11808 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
11809 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
11810 HDA_CODEC_VOLUME("Line In Boost", 0x1a, 0, HDA_INPUT),
11811 { }
11812};
11813
Takashi Iwaiaef9d312008-02-19 13:16:41 +010011814/* bind Beep switches of both NID 0x0f and 0x10 */
11815static struct hda_bind_ctls alc268_bind_beep_sw = {
11816 .ops = &snd_hda_bind_sw,
11817 .values = {
11818 HDA_COMPOSE_AMP_VAL(0x0f, 3, 1, HDA_INPUT),
11819 HDA_COMPOSE_AMP_VAL(0x10, 3, 1, HDA_INPUT),
11820 0
11821 },
11822};
11823
11824static struct snd_kcontrol_new alc268_beep_mixer[] = {
11825 HDA_CODEC_VOLUME("Beep Playback Volume", 0x1d, 0x0, HDA_INPUT),
11826 HDA_BIND_SW("Beep Playback Switch", &alc268_bind_beep_sw),
11827 { }
11828};
11829
Kailang Yangd1a991a2007-08-15 16:21:59 +020011830static struct hda_verb alc268_eapd_verbs[] = {
11831 {0x14, AC_VERB_SET_EAPD_BTLENABLE, 2},
11832 {0x15, AC_VERB_SET_EAPD_BTLENABLE, 2},
11833 { }
11834};
11835
Takashi Iwaid2738092007-08-16 14:59:45 +020011836/* Toshiba specific */
Takashi Iwaid2738092007-08-16 14:59:45 +020011837static struct hda_verb alc268_toshiba_verbs[] = {
11838 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
11839 { } /* end */
11840};
11841
11842/* Acer specific */
Takashi Iwai889c4392007-08-23 18:56:52 +020011843/* bind volumes of both NID 0x02 and 0x03 */
Takashi Iwai6bc96852007-08-17 09:02:12 +020011844static struct hda_bind_ctls alc268_acer_bind_master_vol = {
11845 .ops = &snd_hda_bind_vol,
11846 .values = {
11847 HDA_COMPOSE_AMP_VAL(0x02, 3, 0, HDA_OUTPUT),
11848 HDA_COMPOSE_AMP_VAL(0x03, 3, 0, HDA_OUTPUT),
11849 0
11850 },
11851};
11852
Takashi Iwai889c4392007-08-23 18:56:52 +020011853/* mute/unmute internal speaker according to the hp jack and mute state */
11854static void alc268_acer_automute(struct hda_codec *codec, int force)
11855{
11856 struct alc_spec *spec = codec->spec;
11857 unsigned int mute;
11858
11859 if (force || !spec->sense_updated) {
Wu Fengguang864f92b2009-11-18 12:38:02 +080011860 spec->jack_present = snd_hda_jack_detect(codec, 0x14);
Takashi Iwai889c4392007-08-23 18:56:52 +020011861 spec->sense_updated = 1;
11862 }
11863 if (spec->jack_present)
11864 mute = HDA_AMP_MUTE; /* mute internal speaker */
11865 else /* unmute internal speaker if necessary */
11866 mute = snd_hda_codec_amp_read(codec, 0x14, 0, HDA_OUTPUT, 0);
11867 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
11868 HDA_AMP_MUTE, mute);
11869}
11870
11871
11872/* bind hp and internal speaker mute (with plug check) */
11873static int alc268_acer_master_sw_put(struct snd_kcontrol *kcontrol,
11874 struct snd_ctl_elem_value *ucontrol)
11875{
11876 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
11877 long *valp = ucontrol->value.integer.value;
11878 int change;
11879
Takashi Iwai8de56b72009-07-24 16:51:47 +020011880 change = amp_stereo_mute_update(codec, 0x14, HDA_OUTPUT, 0, valp);
Takashi Iwai889c4392007-08-23 18:56:52 +020011881 if (change)
11882 alc268_acer_automute(codec, 0);
11883 return change;
11884}
Takashi Iwaid2738092007-08-16 14:59:45 +020011885
Kailang Yang8ef355d2008-08-26 13:10:22 +020011886static struct snd_kcontrol_new alc268_acer_aspire_one_mixer[] = {
11887 /* output mixer control */
11888 HDA_BIND_VOL("Master Playback Volume", &alc268_acer_bind_master_vol),
11889 {
11890 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
11891 .name = "Master Playback Switch",
11892 .info = snd_hda_mixer_amp_switch_info,
11893 .get = snd_hda_mixer_amp_switch_get,
11894 .put = alc268_acer_master_sw_put,
11895 .private_value = HDA_COMPOSE_AMP_VAL(0x14, 3, 0, HDA_OUTPUT),
11896 },
11897 HDA_CODEC_VOLUME("Mic Boost Capture Volume", 0x18, 0, HDA_INPUT),
11898 { }
11899};
11900
Takashi Iwaid2738092007-08-16 14:59:45 +020011901static struct snd_kcontrol_new alc268_acer_mixer[] = {
11902 /* output mixer control */
11903 HDA_BIND_VOL("Master Playback Volume", &alc268_acer_bind_master_vol),
11904 {
11905 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
11906 .name = "Master Playback Switch",
11907 .info = snd_hda_mixer_amp_switch_info,
11908 .get = snd_hda_mixer_amp_switch_get,
11909 .put = alc268_acer_master_sw_put,
11910 .private_value = HDA_COMPOSE_AMP_VAL(0x14, 3, 0, HDA_OUTPUT),
11911 },
Takashi Iwai33bf17a2007-08-21 11:51:42 +020011912 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
11913 HDA_CODEC_VOLUME("Internal Mic Boost", 0x19, 0, HDA_INPUT),
11914 HDA_CODEC_VOLUME("Line In Boost", 0x1a, 0, HDA_INPUT),
Takashi Iwaid2738092007-08-16 14:59:45 +020011915 { }
11916};
11917
Takashi Iwaic238b4f2008-11-05 14:57:20 +010011918static struct snd_kcontrol_new alc268_acer_dmic_mixer[] = {
11919 /* output mixer control */
11920 HDA_BIND_VOL("Master Playback Volume", &alc268_acer_bind_master_vol),
11921 {
11922 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
11923 .name = "Master Playback Switch",
11924 .info = snd_hda_mixer_amp_switch_info,
11925 .get = snd_hda_mixer_amp_switch_get,
11926 .put = alc268_acer_master_sw_put,
11927 .private_value = HDA_COMPOSE_AMP_VAL(0x14, 3, 0, HDA_OUTPUT),
11928 },
11929 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
11930 HDA_CODEC_VOLUME("Line In Boost", 0x1a, 0, HDA_INPUT),
11931 { }
11932};
11933
Kailang Yang8ef355d2008-08-26 13:10:22 +020011934static struct hda_verb alc268_acer_aspire_one_verbs[] = {
11935 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
11936 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
11937 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
11938 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
11939 {0x23, AC_VERB_SET_CONNECT_SEL, 0x06},
11940 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, 0xa017},
11941 { }
11942};
11943
Takashi Iwaid2738092007-08-16 14:59:45 +020011944static struct hda_verb alc268_acer_verbs[] = {
Takashi Iwai0ccb5412008-03-06 16:58:35 +010011945 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN}, /* internal dmic? */
11946 {0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Takashi Iwaid2738092007-08-16 14:59:45 +020011947 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
11948 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
Takashi Iwai0ccb5412008-03-06 16:58:35 +010011949 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
11950 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
Takashi Iwaid2738092007-08-16 14:59:45 +020011951 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
11952 { }
11953};
11954
11955/* unsolicited event for HP jack sensing */
Takashi Iwai42171c12009-05-08 14:11:43 +020011956#define alc268_toshiba_unsol_event alc262_hippo_unsol_event
Takashi Iwai4f5d17062009-08-11 18:17:46 +020011957#define alc268_toshiba_setup alc262_hippo_setup
11958#define alc268_toshiba_automute alc262_hippo_automute
Takashi Iwaid2738092007-08-16 14:59:45 +020011959
11960static void alc268_acer_unsol_event(struct hda_codec *codec,
11961 unsigned int res)
11962{
Takashi Iwai889c4392007-08-23 18:56:52 +020011963 if ((res >> 26) != ALC880_HP_EVENT)
Takashi Iwaid2738092007-08-16 14:59:45 +020011964 return;
11965 alc268_acer_automute(codec, 1);
11966}
11967
Takashi Iwai889c4392007-08-23 18:56:52 +020011968static void alc268_acer_init_hook(struct hda_codec *codec)
11969{
11970 alc268_acer_automute(codec, 1);
11971}
11972
Kailang Yang8ef355d2008-08-26 13:10:22 +020011973/* toggle speaker-output according to the hp-jack state */
11974static void alc268_aspire_one_speaker_automute(struct hda_codec *codec)
11975{
11976 unsigned int present;
11977 unsigned char bits;
11978
Wu Fengguang864f92b2009-11-18 12:38:02 +080011979 present = snd_hda_jack_detect(codec, 0x15);
Kailang Yang8ef355d2008-08-26 13:10:22 +020011980 bits = present ? AMP_IN_MUTE(0) : 0;
11981 snd_hda_codec_amp_stereo(codec, 0x0f, HDA_INPUT, 0,
11982 AMP_IN_MUTE(0), bits);
11983 snd_hda_codec_amp_stereo(codec, 0x0f, HDA_INPUT, 1,
11984 AMP_IN_MUTE(0), bits);
11985}
11986
Kailang Yang8ef355d2008-08-26 13:10:22 +020011987static void alc268_acer_lc_unsol_event(struct hda_codec *codec,
11988 unsigned int res)
11989{
Takashi Iwai4f5d17062009-08-11 18:17:46 +020011990 switch (res >> 26) {
11991 case ALC880_HP_EVENT:
Kailang Yang8ef355d2008-08-26 13:10:22 +020011992 alc268_aspire_one_speaker_automute(codec);
Takashi Iwai4f5d17062009-08-11 18:17:46 +020011993 break;
11994 case ALC880_MIC_EVENT:
11995 alc_mic_automute(codec);
11996 break;
11997 }
11998}
11999
12000static void alc268_acer_lc_setup(struct hda_codec *codec)
12001{
12002 struct alc_spec *spec = codec->spec;
12003 spec->ext_mic.pin = 0x18;
12004 spec->ext_mic.mux_idx = 0;
12005 spec->int_mic.pin = 0x12;
12006 spec->int_mic.mux_idx = 6;
12007 spec->auto_mic = 1;
Kailang Yang8ef355d2008-08-26 13:10:22 +020012008}
12009
12010static void alc268_acer_lc_init_hook(struct hda_codec *codec)
12011{
12012 alc268_aspire_one_speaker_automute(codec);
Takashi Iwai4f5d17062009-08-11 18:17:46 +020012013 alc_mic_automute(codec);
Kailang Yang8ef355d2008-08-26 13:10:22 +020012014}
12015
Takashi Iwai3866f0b2008-01-15 12:37:42 +010012016static struct snd_kcontrol_new alc268_dell_mixer[] = {
12017 /* output mixer control */
12018 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x02, 0x0, HDA_OUTPUT),
12019 HDA_CODEC_MUTE("Speaker Playback Switch", 0x14, 0x0, HDA_OUTPUT),
12020 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x03, 0x0, HDA_OUTPUT),
12021 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
12022 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
12023 HDA_CODEC_VOLUME("Internal Mic Boost", 0x19, 0, HDA_INPUT),
12024 { }
12025};
12026
12027static struct hda_verb alc268_dell_verbs[] = {
12028 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
12029 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
12030 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
Takashi Iwai4f5d17062009-08-11 18:17:46 +020012031 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_MIC_EVENT | AC_USRSP_EN},
Takashi Iwai3866f0b2008-01-15 12:37:42 +010012032 { }
12033};
12034
12035/* mute/unmute internal speaker according to the hp jack and mute state */
Takashi Iwai4f5d17062009-08-11 18:17:46 +020012036static void alc268_dell_setup(struct hda_codec *codec)
Takashi Iwai3866f0b2008-01-15 12:37:42 +010012037{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020012038 struct alc_spec *spec = codec->spec;
Takashi Iwai3866f0b2008-01-15 12:37:42 +010012039
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020012040 spec->autocfg.hp_pins[0] = 0x15;
12041 spec->autocfg.speaker_pins[0] = 0x14;
Takashi Iwai4f5d17062009-08-11 18:17:46 +020012042 spec->ext_mic.pin = 0x18;
12043 spec->ext_mic.mux_idx = 0;
12044 spec->int_mic.pin = 0x19;
12045 spec->int_mic.mux_idx = 1;
12046 spec->auto_mic = 1;
Takashi Iwai3866f0b2008-01-15 12:37:42 +010012047}
12048
Herton Ronaldo Krzesinskieb5a6622008-04-14 13:46:28 +020012049static struct snd_kcontrol_new alc267_quanta_il1_mixer[] = {
12050 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x2, 0x0, HDA_OUTPUT),
12051 HDA_CODEC_MUTE("Speaker Playback Switch", 0x14, 0x0, HDA_OUTPUT),
12052 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x3, 0x0, HDA_OUTPUT),
12053 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
12054 HDA_CODEC_VOLUME("Mic Capture Volume", 0x23, 0x0, HDA_OUTPUT),
12055 HDA_BIND_MUTE("Mic Capture Switch", 0x23, 2, HDA_OUTPUT),
12056 HDA_CODEC_VOLUME("Ext Mic Boost", 0x18, 0, HDA_INPUT),
12057 HDA_CODEC_VOLUME("Int Mic Boost", 0x19, 0, HDA_INPUT),
12058 { }
12059};
12060
12061static struct hda_verb alc267_quanta_il1_verbs[] = {
12062 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
12063 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_MIC_EVENT | AC_USRSP_EN},
12064 { }
12065};
12066
Takashi Iwai4f5d17062009-08-11 18:17:46 +020012067static void alc267_quanta_il1_setup(struct hda_codec *codec)
Herton Ronaldo Krzesinskieb5a6622008-04-14 13:46:28 +020012068{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020012069 struct alc_spec *spec = codec->spec;
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020012070 spec->autocfg.hp_pins[0] = 0x15;
12071 spec->autocfg.speaker_pins[0] = 0x14;
Takashi Iwai4f5d17062009-08-11 18:17:46 +020012072 spec->ext_mic.pin = 0x18;
12073 spec->ext_mic.mux_idx = 0;
12074 spec->int_mic.pin = 0x19;
12075 spec->int_mic.mux_idx = 1;
12076 spec->auto_mic = 1;
Herton Ronaldo Krzesinskieb5a6622008-04-14 13:46:28 +020012077}
12078
Kailang Yanga361d842007-06-05 12:30:55 +020012079/*
12080 * generic initialization of ADC, input mixers and output mixers
12081 */
12082static struct hda_verb alc268_base_init_verbs[] = {
12083 /* Unmute DAC0-1 and set vol = 0 */
12084 {0x02, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
Kailang Yanga361d842007-06-05 12:30:55 +020012085 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
Kailang Yanga361d842007-06-05 12:30:55 +020012086
12087 /*
12088 * Set up output mixers (0x0c - 0x0e)
12089 */
12090 /* set vol=0 to output mixers */
12091 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Kailang Yanga361d842007-06-05 12:30:55 +020012092 {0x0e, AC_VERB_SET_CONNECT_SEL, 0x00},
12093
12094 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
12095 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
12096
12097 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40},
12098 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc0},
12099 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40},
12100 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
12101 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
12102 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
12103 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
12104 {0x1d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
12105
12106 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
12107 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
12108 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
12109 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
Kailang Yanga361d842007-06-05 12:30:55 +020012110 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
Takashi Iwaiaef9d312008-02-19 13:16:41 +010012111
12112 /* set PCBEEP vol = 0, mute connections */
12113 {0x1d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
12114 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
12115 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Kailang Yanga361d842007-06-05 12:30:55 +020012116
Jiang Zhea9b3aa82007-12-20 13:13:13 +010012117 /* Unmute Selector 23h,24h and set the default input to mic-in */
Kailang Yangea1fb292008-08-26 12:58:38 +020012118
Jiang Zhea9b3aa82007-12-20 13:13:13 +010012119 {0x23, AC_VERB_SET_CONNECT_SEL, 0x00},
12120 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
12121 {0x24, AC_VERB_SET_CONNECT_SEL, 0x00},
12122 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Kailang Yanga361d842007-06-05 12:30:55 +020012123
Kailang Yanga361d842007-06-05 12:30:55 +020012124 { }
12125};
12126
12127/*
12128 * generic initialization of ADC, input mixers and output mixers
12129 */
12130static struct hda_verb alc268_volume_init_verbs[] = {
12131 /* set output DAC */
Takashi Iwai4cfb91c2009-01-23 12:53:09 +010012132 {0x02, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
12133 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
Kailang Yanga361d842007-06-05 12:30:55 +020012134
12135 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
12136 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24},
12137 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
12138 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
12139 {0x1d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20},
12140
Kailang Yanga361d842007-06-05 12:30:55 +020012141 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Kailang Yanga361d842007-06-05 12:30:55 +020012142 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
12143 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
12144
12145 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
Kailang Yanga361d842007-06-05 12:30:55 +020012146 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
Kailang Yanga361d842007-06-05 12:30:55 +020012147
Takashi Iwaiaef9d312008-02-19 13:16:41 +010012148 /* set PCBEEP vol = 0, mute connections */
12149 {0x1d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
12150 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
12151 {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Kailang Yanga361d842007-06-05 12:30:55 +020012152
12153 { }
12154};
12155
Takashi Iwaifdbc6622009-08-19 00:18:10 +020012156static struct snd_kcontrol_new alc268_capture_nosrc_mixer[] = {
12157 HDA_CODEC_VOLUME("Capture Volume", 0x23, 0x0, HDA_OUTPUT),
12158 HDA_CODEC_MUTE("Capture Switch", 0x23, 0x0, HDA_OUTPUT),
12159 { } /* end */
12160};
12161
Kailang Yanga361d842007-06-05 12:30:55 +020012162static struct snd_kcontrol_new alc268_capture_alt_mixer[] = {
12163 HDA_CODEC_VOLUME("Capture Volume", 0x23, 0x0, HDA_OUTPUT),
12164 HDA_CODEC_MUTE("Capture Switch", 0x23, 0x0, HDA_OUTPUT),
Takashi Iwaifdbc6622009-08-19 00:18:10 +020012165 _DEFINE_CAPSRC(1),
Kailang Yanga361d842007-06-05 12:30:55 +020012166 { } /* end */
12167};
12168
12169static struct snd_kcontrol_new alc268_capture_mixer[] = {
12170 HDA_CODEC_VOLUME("Capture Volume", 0x23, 0x0, HDA_OUTPUT),
12171 HDA_CODEC_MUTE("Capture Switch", 0x23, 0x0, HDA_OUTPUT),
12172 HDA_CODEC_VOLUME_IDX("Capture Volume", 1, 0x24, 0x0, HDA_OUTPUT),
12173 HDA_CODEC_MUTE_IDX("Capture Switch", 1, 0x24, 0x0, HDA_OUTPUT),
Takashi Iwaifdbc6622009-08-19 00:18:10 +020012174 _DEFINE_CAPSRC(2),
Kailang Yanga361d842007-06-05 12:30:55 +020012175 { } /* end */
12176};
12177
12178static struct hda_input_mux alc268_capture_source = {
12179 .num_items = 4,
12180 .items = {
12181 { "Mic", 0x0 },
12182 { "Front Mic", 0x1 },
12183 { "Line", 0x2 },
12184 { "CD", 0x3 },
12185 },
12186};
12187
Takashi Iwai0ccb5412008-03-06 16:58:35 +010012188static struct hda_input_mux alc268_acer_capture_source = {
12189 .num_items = 3,
12190 .items = {
12191 { "Mic", 0x0 },
Takashi Iwaic238b4f2008-11-05 14:57:20 +010012192 { "Internal Mic", 0x1 },
12193 { "Line", 0x2 },
12194 },
12195};
12196
12197static struct hda_input_mux alc268_acer_dmic_capture_source = {
12198 .num_items = 3,
12199 .items = {
12200 { "Mic", 0x0 },
Takashi Iwai0ccb5412008-03-06 16:58:35 +010012201 { "Internal Mic", 0x6 },
12202 { "Line", 0x2 },
12203 },
12204};
12205
Jonathan Woithe86c53bd2008-01-08 12:33:19 +010012206#ifdef CONFIG_SND_DEBUG
12207static struct snd_kcontrol_new alc268_test_mixer[] = {
Jonathan Woithe86c53bd2008-01-08 12:33:19 +010012208 /* Volume widgets */
12209 HDA_CODEC_VOLUME("LOUT1 Playback Volume", 0x02, 0x0, HDA_OUTPUT),
12210 HDA_CODEC_VOLUME("LOUT2 Playback Volume", 0x03, 0x0, HDA_OUTPUT),
12211 HDA_BIND_MUTE_MONO("Mono sum Playback Switch", 0x0e, 1, 2, HDA_INPUT),
12212 HDA_BIND_MUTE("LINE-OUT sum Playback Switch", 0x0f, 2, HDA_INPUT),
12213 HDA_BIND_MUTE("HP-OUT sum Playback Switch", 0x10, 2, HDA_INPUT),
12214 HDA_BIND_MUTE("LINE-OUT Playback Switch", 0x14, 2, HDA_OUTPUT),
12215 HDA_BIND_MUTE("HP-OUT Playback Switch", 0x15, 2, HDA_OUTPUT),
12216 HDA_BIND_MUTE("Mono Playback Switch", 0x16, 2, HDA_OUTPUT),
12217 HDA_CODEC_VOLUME("MIC1 Capture Volume", 0x18, 0x0, HDA_INPUT),
12218 HDA_BIND_MUTE("MIC1 Capture Switch", 0x18, 2, HDA_OUTPUT),
12219 HDA_CODEC_VOLUME("MIC2 Capture Volume", 0x19, 0x0, HDA_INPUT),
12220 HDA_CODEC_VOLUME("LINE1 Capture Volume", 0x1a, 0x0, HDA_INPUT),
12221 HDA_BIND_MUTE("LINE1 Capture Switch", 0x1a, 2, HDA_OUTPUT),
Takashi Iwaif0747ee2008-01-15 11:41:41 +010012222 /* The below appears problematic on some hardwares */
12223 /*HDA_CODEC_VOLUME("PCBEEP Playback Volume", 0x1d, 0x0, HDA_INPUT),*/
Jonathan Woithe86c53bd2008-01-08 12:33:19 +010012224 HDA_CODEC_VOLUME("PCM-IN1 Capture Volume", 0x23, 0x0, HDA_OUTPUT),
12225 HDA_BIND_MUTE("PCM-IN1 Capture Switch", 0x23, 2, HDA_OUTPUT),
12226 HDA_CODEC_VOLUME("PCM-IN2 Capture Volume", 0x24, 0x0, HDA_OUTPUT),
12227 HDA_BIND_MUTE("PCM-IN2 Capture Switch", 0x24, 2, HDA_OUTPUT),
12228
12229 /* Modes for retasking pin widgets */
12230 ALC_PIN_MODE("LINE-OUT pin mode", 0x14, ALC_PIN_DIR_INOUT),
12231 ALC_PIN_MODE("HP-OUT pin mode", 0x15, ALC_PIN_DIR_INOUT),
12232 ALC_PIN_MODE("MIC1 pin mode", 0x18, ALC_PIN_DIR_INOUT),
12233 ALC_PIN_MODE("LINE1 pin mode", 0x1a, ALC_PIN_DIR_INOUT),
12234
12235 /* Controls for GPIO pins, assuming they are configured as outputs */
12236 ALC_GPIO_DATA_SWITCH("GPIO pin 0", 0x01, 0x01),
12237 ALC_GPIO_DATA_SWITCH("GPIO pin 1", 0x01, 0x02),
12238 ALC_GPIO_DATA_SWITCH("GPIO pin 2", 0x01, 0x04),
12239 ALC_GPIO_DATA_SWITCH("GPIO pin 3", 0x01, 0x08),
12240
12241 /* Switches to allow the digital SPDIF output pin to be enabled.
12242 * The ALC268 does not have an SPDIF input.
12243 */
12244 ALC_SPDIF_CTRL_SWITCH("SPDIF Playback Switch", 0x06, 0x01),
12245
12246 /* A switch allowing EAPD to be enabled. Some laptops seem to use
12247 * this output to turn on an external amplifier.
12248 */
12249 ALC_EAPD_CTRL_SWITCH("LINE-OUT EAPD Enable Switch", 0x0f, 0x02),
12250 ALC_EAPD_CTRL_SWITCH("HP-OUT EAPD Enable Switch", 0x10, 0x02),
12251
12252 { } /* end */
12253};
12254#endif
12255
Kailang Yanga361d842007-06-05 12:30:55 +020012256/* create input playback/capture controls for the given pin */
12257static int alc268_new_analog_output(struct alc_spec *spec, hda_nid_t nid,
12258 const char *ctlname, int idx)
12259{
Takashi Iwai3f3b7c12009-07-17 14:36:59 +020012260 hda_nid_t dac;
Kailang Yanga361d842007-06-05 12:30:55 +020012261 int err;
12262
Takashi Iwai3f3b7c12009-07-17 14:36:59 +020012263 switch (nid) {
12264 case 0x14:
12265 case 0x16:
12266 dac = 0x02;
12267 break;
12268 case 0x15:
12269 dac = 0x03;
12270 break;
12271 default:
12272 return 0;
12273 }
12274 if (spec->multiout.dac_nids[0] != dac &&
12275 spec->multiout.dac_nids[1] != dac) {
Takashi Iwai0afe5f82009-10-02 09:20:00 +020012276 err = add_pb_vol_ctrl(spec, ALC_CTL_WIDGET_VOL, ctlname,
Takashi Iwai3f3b7c12009-07-17 14:36:59 +020012277 HDA_COMPOSE_AMP_VAL(dac, 3, idx,
Kailang Yanga361d842007-06-05 12:30:55 +020012278 HDA_OUTPUT));
12279 if (err < 0)
12280 return err;
Takashi Iwai3f3b7c12009-07-17 14:36:59 +020012281 spec->multiout.dac_nids[spec->multiout.num_dacs++] = dac;
12282 }
12283
Takashi Iwai3f3b7c12009-07-17 14:36:59 +020012284 if (nid != 0x16)
Takashi Iwai0afe5f82009-10-02 09:20:00 +020012285 err = add_pb_sw_ctrl(spec, ALC_CTL_WIDGET_MUTE, ctlname,
Kailang Yanga361d842007-06-05 12:30:55 +020012286 HDA_COMPOSE_AMP_VAL(nid, 3, idx, HDA_OUTPUT));
Takashi Iwai3f3b7c12009-07-17 14:36:59 +020012287 else /* mono */
Takashi Iwai0afe5f82009-10-02 09:20:00 +020012288 err = add_pb_sw_ctrl(spec, ALC_CTL_WIDGET_MUTE, ctlname,
Takashi Iwai3f3b7c12009-07-17 14:36:59 +020012289 HDA_COMPOSE_AMP_VAL(nid, 2, idx, HDA_OUTPUT));
Kailang Yanga361d842007-06-05 12:30:55 +020012290 if (err < 0)
12291 return err;
12292 return 0;
12293}
12294
12295/* add playback controls from the parsed DAC table */
12296static int alc268_auto_create_multi_out_ctls(struct alc_spec *spec,
12297 const struct auto_pin_cfg *cfg)
12298{
12299 hda_nid_t nid;
12300 int err;
12301
Kailang Yanga361d842007-06-05 12:30:55 +020012302 spec->multiout.dac_nids = spec->private_dac_nids;
Kailang Yanga361d842007-06-05 12:30:55 +020012303
12304 nid = cfg->line_out_pins[0];
Takashi Iwai3f3b7c12009-07-17 14:36:59 +020012305 if (nid) {
12306 const char *name;
12307 if (cfg->line_out_type == AUTO_PIN_SPEAKER_OUT)
12308 name = "Speaker";
12309 else
12310 name = "Front";
12311 err = alc268_new_analog_output(spec, nid, name, 0);
12312 if (err < 0)
12313 return err;
12314 }
Kailang Yanga361d842007-06-05 12:30:55 +020012315
12316 nid = cfg->speaker_pins[0];
12317 if (nid == 0x1d) {
Takashi Iwai0afe5f82009-10-02 09:20:00 +020012318 err = add_pb_vol_ctrl(spec, ALC_CTL_WIDGET_VOL, "Speaker",
Kailang Yanga361d842007-06-05 12:30:55 +020012319 HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_INPUT));
12320 if (err < 0)
12321 return err;
Takashi Iwai3f3b7c12009-07-17 14:36:59 +020012322 } else {
12323 err = alc268_new_analog_output(spec, nid, "Speaker", 0);
12324 if (err < 0)
12325 return err;
Kailang Yanga361d842007-06-05 12:30:55 +020012326 }
12327 nid = cfg->hp_pins[0];
Takashi Iwai3f3b7c12009-07-17 14:36:59 +020012328 if (nid) {
12329 err = alc268_new_analog_output(spec, nid, "Headphone", 0);
12330 if (err < 0)
12331 return err;
12332 }
Kailang Yanga361d842007-06-05 12:30:55 +020012333
12334 nid = cfg->line_out_pins[1] | cfg->line_out_pins[2];
12335 if (nid == 0x16) {
Takashi Iwai0afe5f82009-10-02 09:20:00 +020012336 err = add_pb_sw_ctrl(spec, ALC_CTL_WIDGET_MUTE, "Mono",
Takashi Iwai3f3b7c12009-07-17 14:36:59 +020012337 HDA_COMPOSE_AMP_VAL(nid, 2, 0, HDA_OUTPUT));
Kailang Yanga361d842007-06-05 12:30:55 +020012338 if (err < 0)
12339 return err;
12340 }
Kailang Yangea1fb292008-08-26 12:58:38 +020012341 return 0;
Kailang Yanga361d842007-06-05 12:30:55 +020012342}
12343
12344/* create playback/capture controls for input pins */
Takashi Iwai05f5f472009-08-25 13:10:18 +020012345static int alc268_auto_create_input_ctls(struct hda_codec *codec,
Kailang Yanga361d842007-06-05 12:30:55 +020012346 const struct auto_pin_cfg *cfg)
12347{
Takashi Iwai05f5f472009-08-25 13:10:18 +020012348 return alc_auto_create_input_ctls(codec, cfg, 0, 0x23, 0x24);
Kailang Yanga361d842007-06-05 12:30:55 +020012349}
12350
Takashi Iwaie9af4f32009-08-29 23:23:08 +020012351static void alc268_auto_set_output_and_unmute(struct hda_codec *codec,
12352 hda_nid_t nid, int pin_type)
12353{
12354 int idx;
12355
12356 alc_set_pin_output(codec, nid, pin_type);
12357 if (nid == 0x14 || nid == 0x16)
12358 idx = 0;
12359 else
12360 idx = 1;
12361 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_CONNECT_SEL, idx);
12362}
12363
12364static void alc268_auto_init_multi_out(struct hda_codec *codec)
12365{
12366 struct alc_spec *spec = codec->spec;
12367 hda_nid_t nid = spec->autocfg.line_out_pins[0];
12368 if (nid) {
12369 int pin_type = get_pin_type(spec->autocfg.line_out_type);
12370 alc268_auto_set_output_and_unmute(codec, nid, pin_type);
12371 }
12372}
12373
12374static void alc268_auto_init_hp_out(struct hda_codec *codec)
12375{
12376 struct alc_spec *spec = codec->spec;
12377 hda_nid_t pin;
12378
12379 pin = spec->autocfg.hp_pins[0];
12380 if (pin)
12381 alc268_auto_set_output_and_unmute(codec, pin, PIN_HP);
12382 pin = spec->autocfg.speaker_pins[0];
12383 if (pin)
12384 alc268_auto_set_output_and_unmute(codec, pin, PIN_OUT);
12385}
12386
Kailang Yanga361d842007-06-05 12:30:55 +020012387static void alc268_auto_init_mono_speaker_out(struct hda_codec *codec)
12388{
12389 struct alc_spec *spec = codec->spec;
12390 hda_nid_t speaker_nid = spec->autocfg.speaker_pins[0];
12391 hda_nid_t hp_nid = spec->autocfg.hp_pins[0];
12392 hda_nid_t line_nid = spec->autocfg.line_out_pins[0];
12393 unsigned int dac_vol1, dac_vol2;
12394
Takashi Iwaie9af4f32009-08-29 23:23:08 +020012395 if (line_nid == 0x1d || speaker_nid == 0x1d) {
Kailang Yanga361d842007-06-05 12:30:55 +020012396 snd_hda_codec_write(codec, speaker_nid, 0,
12397 AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
Takashi Iwaie9af4f32009-08-29 23:23:08 +020012398 /* mute mixer inputs from 0x1d */
Kailang Yanga361d842007-06-05 12:30:55 +020012399 snd_hda_codec_write(codec, 0x0f, 0,
12400 AC_VERB_SET_AMP_GAIN_MUTE,
12401 AMP_IN_UNMUTE(1));
12402 snd_hda_codec_write(codec, 0x10, 0,
12403 AC_VERB_SET_AMP_GAIN_MUTE,
12404 AMP_IN_UNMUTE(1));
12405 } else {
Takashi Iwaie9af4f32009-08-29 23:23:08 +020012406 /* unmute mixer inputs from 0x1d */
Kailang Yanga361d842007-06-05 12:30:55 +020012407 snd_hda_codec_write(codec, 0x0f, 0,
12408 AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1));
12409 snd_hda_codec_write(codec, 0x10, 0,
12410 AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1));
12411 }
12412
12413 dac_vol1 = dac_vol2 = 0xb000 | 0x40; /* set max volume */
Kailang Yangea1fb292008-08-26 12:58:38 +020012414 if (line_nid == 0x14)
Kailang Yanga361d842007-06-05 12:30:55 +020012415 dac_vol2 = AMP_OUT_ZERO;
12416 else if (line_nid == 0x15)
12417 dac_vol1 = AMP_OUT_ZERO;
Kailang Yangea1fb292008-08-26 12:58:38 +020012418 if (hp_nid == 0x14)
Kailang Yanga361d842007-06-05 12:30:55 +020012419 dac_vol2 = AMP_OUT_ZERO;
12420 else if (hp_nid == 0x15)
12421 dac_vol1 = AMP_OUT_ZERO;
12422 if (line_nid != 0x16 || hp_nid != 0x16 ||
12423 spec->autocfg.line_out_pins[1] != 0x16 ||
12424 spec->autocfg.line_out_pins[2] != 0x16)
12425 dac_vol1 = dac_vol2 = AMP_OUT_ZERO;
12426
12427 snd_hda_codec_write(codec, 0x02, 0,
12428 AC_VERB_SET_AMP_GAIN_MUTE, dac_vol1);
12429 snd_hda_codec_write(codec, 0x03, 0,
12430 AC_VERB_SET_AMP_GAIN_MUTE, dac_vol2);
12431}
12432
Sasha Alexandrdef319f2009-06-16 16:00:15 -040012433/* pcm configuration: identical with ALC880 */
Kailang Yanga361d842007-06-05 12:30:55 +020012434#define alc268_pcm_analog_playback alc880_pcm_analog_playback
12435#define alc268_pcm_analog_capture alc880_pcm_analog_capture
Takashi Iwai63300792008-01-24 15:31:36 +010012436#define alc268_pcm_analog_alt_capture alc880_pcm_analog_alt_capture
Kailang Yanga361d842007-06-05 12:30:55 +020012437#define alc268_pcm_digital_playback alc880_pcm_digital_playback
12438
12439/*
12440 * BIOS auto configuration
12441 */
12442static int alc268_parse_auto_config(struct hda_codec *codec)
12443{
12444 struct alc_spec *spec = codec->spec;
12445 int err;
12446 static hda_nid_t alc268_ignore[] = { 0 };
12447
12448 err = snd_hda_parse_pin_def_config(codec, &spec->autocfg,
12449 alc268_ignore);
12450 if (err < 0)
12451 return err;
Takashi Iwai7e0e44d2009-02-19 08:15:49 +010012452 if (!spec->autocfg.line_outs) {
12453 if (spec->autocfg.dig_outs || spec->autocfg.dig_in_pin) {
12454 spec->multiout.max_channels = 2;
12455 spec->no_analog = 1;
12456 goto dig_only;
12457 }
Kailang Yanga361d842007-06-05 12:30:55 +020012458 return 0; /* can't find valid BIOS pin config */
Takashi Iwai7e0e44d2009-02-19 08:15:49 +010012459 }
Kailang Yanga361d842007-06-05 12:30:55 +020012460 err = alc268_auto_create_multi_out_ctls(spec, &spec->autocfg);
12461 if (err < 0)
12462 return err;
Takashi Iwai05f5f472009-08-25 13:10:18 +020012463 err = alc268_auto_create_input_ctls(codec, &spec->autocfg);
Kailang Yanga361d842007-06-05 12:30:55 +020012464 if (err < 0)
12465 return err;
12466
12467 spec->multiout.max_channels = 2;
12468
Takashi Iwai7e0e44d2009-02-19 08:15:49 +010012469 dig_only:
Kailang Yanga361d842007-06-05 12:30:55 +020012470 /* digital only support output */
Takashi Iwai7e0e44d2009-02-19 08:15:49 +010012471 if (spec->autocfg.dig_outs) {
Kailang Yanga361d842007-06-05 12:30:55 +020012472 spec->multiout.dig_out_nid = ALC268_DIGOUT_NID;
Takashi Iwai7e0e44d2009-02-19 08:15:49 +010012473 spec->dig_out_type = spec->autocfg.dig_out_type[0];
12474 }
Takashi Iwai603c4012008-07-30 15:01:44 +020012475 if (spec->kctls.list)
Takashi Iwaid88897e2008-10-31 15:01:37 +010012476 add_mixer(spec, spec->kctls.list);
Kailang Yanga361d842007-06-05 12:30:55 +020012477
Takashi Iwai892981f2009-03-02 08:04:35 +010012478 if (!spec->no_analog && spec->autocfg.speaker_pins[0] != 0x1d)
Takashi Iwaid88897e2008-10-31 15:01:37 +010012479 add_mixer(spec, alc268_beep_mixer);
Takashi Iwaiaef9d312008-02-19 13:16:41 +010012480
Takashi Iwaid88897e2008-10-31 15:01:37 +010012481 add_verb(spec, alc268_volume_init_verbs);
Herton Ronaldo Krzesinski59085892009-08-11 22:33:09 -030012482 spec->num_mux_defs = 2;
Herton Ronaldo Krzesinski61b9b9b2009-01-28 09:16:33 -020012483 spec->input_mux = &spec->private_imux[0];
Kailang Yanga361d842007-06-05 12:30:55 +020012484
Takashi Iwai776e1842007-08-29 15:07:11 +020012485 err = alc_auto_add_mic_boost(codec);
12486 if (err < 0)
12487 return err;
12488
Takashi Iwai1d955eb2009-06-29 11:33:53 +020012489 alc_ssid_check(codec, 0x15, 0x1b, 0x14);
12490
Kailang Yanga361d842007-06-05 12:30:55 +020012491 return 1;
12492}
12493
Kailang Yanga361d842007-06-05 12:30:55 +020012494#define alc268_auto_init_analog_input alc882_auto_init_analog_input
12495
12496/* init callback for auto-configuration model -- overriding the default init */
12497static void alc268_auto_init(struct hda_codec *codec)
12498{
Takashi Iwaif6c7e542008-02-12 18:32:23 +010012499 struct alc_spec *spec = codec->spec;
Kailang Yanga361d842007-06-05 12:30:55 +020012500 alc268_auto_init_multi_out(codec);
12501 alc268_auto_init_hp_out(codec);
12502 alc268_auto_init_mono_speaker_out(codec);
12503 alc268_auto_init_analog_input(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +010012504 if (spec->unsol_event)
Kailang Yang7fb0d782008-10-15 11:12:35 +020012505 alc_inithook(codec);
Kailang Yanga361d842007-06-05 12:30:55 +020012506}
12507
12508/*
12509 * configuration and preset
12510 */
12511static const char *alc268_models[ALC268_MODEL_LAST] = {
Herton Ronaldo Krzesinskieb5a6622008-04-14 13:46:28 +020012512 [ALC267_QUANTA_IL1] = "quanta-il1",
Kailang Yanga361d842007-06-05 12:30:55 +020012513 [ALC268_3ST] = "3stack",
Takashi Iwai983f8ae2007-08-15 16:44:04 +020012514 [ALC268_TOSHIBA] = "toshiba",
Takashi Iwaid2738092007-08-16 14:59:45 +020012515 [ALC268_ACER] = "acer",
Takashi Iwaic238b4f2008-11-05 14:57:20 +010012516 [ALC268_ACER_DMIC] = "acer-dmic",
Kailang Yang8ef355d2008-08-26 13:10:22 +020012517 [ALC268_ACER_ASPIRE_ONE] = "acer-aspire",
Takashi Iwai3866f0b2008-01-15 12:37:42 +010012518 [ALC268_DELL] = "dell",
Mirco Tischlerf12462c2008-02-04 12:33:59 +010012519 [ALC268_ZEPTO] = "zepto",
Jonathan Woithe86c53bd2008-01-08 12:33:19 +010012520#ifdef CONFIG_SND_DEBUG
12521 [ALC268_TEST] = "test",
12522#endif
Kailang Yanga361d842007-06-05 12:30:55 +020012523 [ALC268_AUTO] = "auto",
12524};
12525
12526static struct snd_pci_quirk alc268_cfg_tbl[] = {
Takashi Iwaia0b8f7d2008-04-22 19:39:49 +020012527 SND_PCI_QUIRK(0x1025, 0x011e, "Acer Aspire 5720z", ALC268_ACER),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010012528 SND_PCI_QUIRK(0x1025, 0x0126, "Acer", ALC268_ACER),
Andy Shevchenkodafc8352008-01-25 11:53:50 +010012529 SND_PCI_QUIRK(0x1025, 0x012e, "Acer Aspire 5310", ALC268_ACER),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010012530 SND_PCI_QUIRK(0x1025, 0x0130, "Acer Extensa 5210", ALC268_ACER),
Andy Shevchenko29a52c22008-01-24 17:29:00 +010012531 SND_PCI_QUIRK(0x1025, 0x0136, "Acer Aspire 5315", ALC268_ACER),
Kailang Yang8ef355d2008-08-26 13:10:22 +020012532 SND_PCI_QUIRK(0x1025, 0x015b, "Acer Aspire One",
12533 ALC268_ACER_ASPIRE_ONE),
Takashi Iwai3866f0b2008-01-15 12:37:42 +010012534 SND_PCI_QUIRK(0x1028, 0x0253, "Dell OEM", ALC268_DELL),
Daniel T Chena1bf8082009-11-01 18:32:29 -050012535 SND_PCI_QUIRK_MASK(0x1028, 0xfff0, 0x02b0,
12536 "Dell Inspiron Mini9/Vostro A90", ALC268_DELL),
Takashi Iwai33d78672009-09-08 11:03:41 +020012537 /* almost compatible with toshiba but with optional digital outs;
12538 * auto-probing seems working fine
12539 */
Takashi Iwai8871e5b2009-06-02 01:02:50 +020012540 SND_PCI_QUIRK_MASK(0x103c, 0xff00, 0x3000, "HP TX25xx series",
Takashi Iwai33d78672009-09-08 11:03:41 +020012541 ALC268_AUTO),
Kailang Yanga361d842007-06-05 12:30:55 +020012542 SND_PCI_QUIRK(0x1043, 0x1205, "ASUS W7J", ALC268_3ST),
Takashi Iwai8871e5b2009-06-02 01:02:50 +020012543 SND_PCI_QUIRK(0x1170, 0x0040, "ZEPTO", ALC268_ZEPTO),
Tony Vroon378bd6a2008-06-04 12:08:30 +020012544 SND_PCI_QUIRK(0x14c0, 0x0025, "COMPAL IFL90/JFL-92", ALC268_TOSHIBA),
Takashi Iwaib875bf32007-09-06 15:00:27 +020012545 SND_PCI_QUIRK(0x152d, 0x0763, "Diverse (CPR2000)", ALC268_ACER),
Herton Ronaldo Krzesinskieb5a6622008-04-14 13:46:28 +020012546 SND_PCI_QUIRK(0x152d, 0x0771, "Quanta IL1", ALC267_QUANTA_IL1),
Takashi Iwai8871e5b2009-06-02 01:02:50 +020012547 SND_PCI_QUIRK(0x1854, 0x1775, "LG R510", ALC268_DELL),
Kailang Yanga361d842007-06-05 12:30:55 +020012548 {}
12549};
12550
Takashi Iwai3abf2f32009-08-19 20:05:02 +020012551/* Toshiba laptops have no unique PCI SSID but only codec SSID */
12552static struct snd_pci_quirk alc268_ssid_cfg_tbl[] = {
12553 SND_PCI_QUIRK(0x1179, 0xff0a, "TOSHIBA X-200", ALC268_AUTO),
12554 SND_PCI_QUIRK(0x1179, 0xff0e, "TOSHIBA X-200 HDMI", ALC268_AUTO),
12555 SND_PCI_QUIRK_MASK(0x1179, 0xff00, 0xff00, "TOSHIBA A/Lx05",
12556 ALC268_TOSHIBA),
12557 {}
12558};
12559
Kailang Yanga361d842007-06-05 12:30:55 +020012560static struct alc_config_preset alc268_presets[] = {
Herton Ronaldo Krzesinskieb5a6622008-04-14 13:46:28 +020012561 [ALC267_QUANTA_IL1] = {
Takashi Iwaifdbc6622009-08-19 00:18:10 +020012562 .mixers = { alc267_quanta_il1_mixer, alc268_beep_mixer,
12563 alc268_capture_nosrc_mixer },
Herton Ronaldo Krzesinskieb5a6622008-04-14 13:46:28 +020012564 .init_verbs = { alc268_base_init_verbs, alc268_eapd_verbs,
12565 alc267_quanta_il1_verbs },
12566 .num_dacs = ARRAY_SIZE(alc268_dac_nids),
12567 .dac_nids = alc268_dac_nids,
12568 .num_adc_nids = ARRAY_SIZE(alc268_adc_nids_alt),
12569 .adc_nids = alc268_adc_nids_alt,
12570 .hp_nid = 0x03,
12571 .num_channel_mode = ARRAY_SIZE(alc268_modes),
12572 .channel_mode = alc268_modes,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020012573 .unsol_event = alc_sku_unsol_event,
12574 .setup = alc267_quanta_il1_setup,
12575 .init_hook = alc_inithook,
Herton Ronaldo Krzesinskieb5a6622008-04-14 13:46:28 +020012576 },
Kailang Yanga361d842007-06-05 12:30:55 +020012577 [ALC268_3ST] = {
Takashi Iwaiaef9d312008-02-19 13:16:41 +010012578 .mixers = { alc268_base_mixer, alc268_capture_alt_mixer,
12579 alc268_beep_mixer },
Kailang Yanga361d842007-06-05 12:30:55 +020012580 .init_verbs = { alc268_base_init_verbs },
12581 .num_dacs = ARRAY_SIZE(alc268_dac_nids),
12582 .dac_nids = alc268_dac_nids,
12583 .num_adc_nids = ARRAY_SIZE(alc268_adc_nids_alt),
12584 .adc_nids = alc268_adc_nids_alt,
Takashi Iwaie1406342008-02-11 18:32:32 +010012585 .capsrc_nids = alc268_capsrc_nids,
Kailang Yanga361d842007-06-05 12:30:55 +020012586 .hp_nid = 0x03,
12587 .dig_out_nid = ALC268_DIGOUT_NID,
12588 .num_channel_mode = ARRAY_SIZE(alc268_modes),
12589 .channel_mode = alc268_modes,
12590 .input_mux = &alc268_capture_source,
12591 },
Kailang Yangd1a991a2007-08-15 16:21:59 +020012592 [ALC268_TOSHIBA] = {
Takashi Iwai42171c12009-05-08 14:11:43 +020012593 .mixers = { alc268_toshiba_mixer, alc268_capture_alt_mixer,
Takashi Iwaiaef9d312008-02-19 13:16:41 +010012594 alc268_beep_mixer },
Takashi Iwaid2738092007-08-16 14:59:45 +020012595 .init_verbs = { alc268_base_init_verbs, alc268_eapd_verbs,
12596 alc268_toshiba_verbs },
Kailang Yangd1a991a2007-08-15 16:21:59 +020012597 .num_dacs = ARRAY_SIZE(alc268_dac_nids),
12598 .dac_nids = alc268_dac_nids,
12599 .num_adc_nids = ARRAY_SIZE(alc268_adc_nids_alt),
12600 .adc_nids = alc268_adc_nids_alt,
Takashi Iwaie1406342008-02-11 18:32:32 +010012601 .capsrc_nids = alc268_capsrc_nids,
Kailang Yangd1a991a2007-08-15 16:21:59 +020012602 .hp_nid = 0x03,
12603 .num_channel_mode = ARRAY_SIZE(alc268_modes),
12604 .channel_mode = alc268_modes,
12605 .input_mux = &alc268_capture_source,
Takashi Iwaid2738092007-08-16 14:59:45 +020012606 .unsol_event = alc268_toshiba_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020012607 .setup = alc268_toshiba_setup,
12608 .init_hook = alc268_toshiba_automute,
Takashi Iwaid2738092007-08-16 14:59:45 +020012609 },
12610 [ALC268_ACER] = {
Takashi Iwai432fd132009-09-30 08:13:44 +020012611 .mixers = { alc268_acer_mixer, alc268_capture_alt_mixer,
Takashi Iwaiaef9d312008-02-19 13:16:41 +010012612 alc268_beep_mixer },
Takashi Iwaid2738092007-08-16 14:59:45 +020012613 .init_verbs = { alc268_base_init_verbs, alc268_eapd_verbs,
12614 alc268_acer_verbs },
12615 .num_dacs = ARRAY_SIZE(alc268_dac_nids),
12616 .dac_nids = alc268_dac_nids,
12617 .num_adc_nids = ARRAY_SIZE(alc268_adc_nids_alt),
12618 .adc_nids = alc268_adc_nids_alt,
Takashi Iwaie1406342008-02-11 18:32:32 +010012619 .capsrc_nids = alc268_capsrc_nids,
Takashi Iwaid2738092007-08-16 14:59:45 +020012620 .hp_nid = 0x02,
12621 .num_channel_mode = ARRAY_SIZE(alc268_modes),
12622 .channel_mode = alc268_modes,
Takashi Iwai0ccb5412008-03-06 16:58:35 +010012623 .input_mux = &alc268_acer_capture_source,
Takashi Iwaid2738092007-08-16 14:59:45 +020012624 .unsol_event = alc268_acer_unsol_event,
Takashi Iwai889c4392007-08-23 18:56:52 +020012625 .init_hook = alc268_acer_init_hook,
Kailang Yangd1a991a2007-08-15 16:21:59 +020012626 },
Takashi Iwaic238b4f2008-11-05 14:57:20 +010012627 [ALC268_ACER_DMIC] = {
12628 .mixers = { alc268_acer_dmic_mixer, alc268_capture_alt_mixer,
12629 alc268_beep_mixer },
12630 .init_verbs = { alc268_base_init_verbs, alc268_eapd_verbs,
12631 alc268_acer_verbs },
12632 .num_dacs = ARRAY_SIZE(alc268_dac_nids),
12633 .dac_nids = alc268_dac_nids,
12634 .num_adc_nids = ARRAY_SIZE(alc268_adc_nids_alt),
12635 .adc_nids = alc268_adc_nids_alt,
12636 .capsrc_nids = alc268_capsrc_nids,
12637 .hp_nid = 0x02,
12638 .num_channel_mode = ARRAY_SIZE(alc268_modes),
12639 .channel_mode = alc268_modes,
12640 .input_mux = &alc268_acer_dmic_capture_source,
12641 .unsol_event = alc268_acer_unsol_event,
12642 .init_hook = alc268_acer_init_hook,
12643 },
Kailang Yang8ef355d2008-08-26 13:10:22 +020012644 [ALC268_ACER_ASPIRE_ONE] = {
12645 .mixers = { alc268_acer_aspire_one_mixer,
Takashi Iwai22971e32009-02-10 11:56:44 +010012646 alc268_beep_mixer,
Takashi Iwaifdbc6622009-08-19 00:18:10 +020012647 alc268_capture_nosrc_mixer },
Kailang Yang8ef355d2008-08-26 13:10:22 +020012648 .init_verbs = { alc268_base_init_verbs, alc268_eapd_verbs,
12649 alc268_acer_aspire_one_verbs },
12650 .num_dacs = ARRAY_SIZE(alc268_dac_nids),
12651 .dac_nids = alc268_dac_nids,
12652 .num_adc_nids = ARRAY_SIZE(alc268_adc_nids_alt),
12653 .adc_nids = alc268_adc_nids_alt,
12654 .capsrc_nids = alc268_capsrc_nids,
12655 .hp_nid = 0x03,
12656 .num_channel_mode = ARRAY_SIZE(alc268_modes),
12657 .channel_mode = alc268_modes,
Kailang Yang8ef355d2008-08-26 13:10:22 +020012658 .unsol_event = alc268_acer_lc_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020012659 .setup = alc268_acer_lc_setup,
Kailang Yang8ef355d2008-08-26 13:10:22 +020012660 .init_hook = alc268_acer_lc_init_hook,
12661 },
Takashi Iwai3866f0b2008-01-15 12:37:42 +010012662 [ALC268_DELL] = {
Takashi Iwaifdbc6622009-08-19 00:18:10 +020012663 .mixers = { alc268_dell_mixer, alc268_beep_mixer,
12664 alc268_capture_nosrc_mixer },
Takashi Iwai3866f0b2008-01-15 12:37:42 +010012665 .init_verbs = { alc268_base_init_verbs, alc268_eapd_verbs,
12666 alc268_dell_verbs },
12667 .num_dacs = ARRAY_SIZE(alc268_dac_nids),
12668 .dac_nids = alc268_dac_nids,
Takashi Iwaifdbc6622009-08-19 00:18:10 +020012669 .num_adc_nids = ARRAY_SIZE(alc268_adc_nids_alt),
12670 .adc_nids = alc268_adc_nids_alt,
12671 .capsrc_nids = alc268_capsrc_nids,
Takashi Iwai3866f0b2008-01-15 12:37:42 +010012672 .hp_nid = 0x02,
12673 .num_channel_mode = ARRAY_SIZE(alc268_modes),
12674 .channel_mode = alc268_modes,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020012675 .unsol_event = alc_sku_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020012676 .setup = alc268_dell_setup,
12677 .init_hook = alc_inithook,
Takashi Iwai3866f0b2008-01-15 12:37:42 +010012678 },
Mirco Tischlerf12462c2008-02-04 12:33:59 +010012679 [ALC268_ZEPTO] = {
Takashi Iwaiaef9d312008-02-19 13:16:41 +010012680 .mixers = { alc268_base_mixer, alc268_capture_alt_mixer,
12681 alc268_beep_mixer },
Mirco Tischlerf12462c2008-02-04 12:33:59 +010012682 .init_verbs = { alc268_base_init_verbs, alc268_eapd_verbs,
12683 alc268_toshiba_verbs },
12684 .num_dacs = ARRAY_SIZE(alc268_dac_nids),
12685 .dac_nids = alc268_dac_nids,
12686 .num_adc_nids = ARRAY_SIZE(alc268_adc_nids_alt),
12687 .adc_nids = alc268_adc_nids_alt,
Takashi Iwaie1406342008-02-11 18:32:32 +010012688 .capsrc_nids = alc268_capsrc_nids,
Mirco Tischlerf12462c2008-02-04 12:33:59 +010012689 .hp_nid = 0x03,
12690 .dig_out_nid = ALC268_DIGOUT_NID,
12691 .num_channel_mode = ARRAY_SIZE(alc268_modes),
12692 .channel_mode = alc268_modes,
12693 .input_mux = &alc268_capture_source,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020012694 .setup = alc268_toshiba_setup,
12695 .init_hook = alc268_toshiba_automute,
Mirco Tischlerf12462c2008-02-04 12:33:59 +010012696 },
Jonathan Woithe86c53bd2008-01-08 12:33:19 +010012697#ifdef CONFIG_SND_DEBUG
12698 [ALC268_TEST] = {
12699 .mixers = { alc268_test_mixer, alc268_capture_mixer },
12700 .init_verbs = { alc268_base_init_verbs, alc268_eapd_verbs,
12701 alc268_volume_init_verbs },
12702 .num_dacs = ARRAY_SIZE(alc268_dac_nids),
12703 .dac_nids = alc268_dac_nids,
12704 .num_adc_nids = ARRAY_SIZE(alc268_adc_nids_alt),
12705 .adc_nids = alc268_adc_nids_alt,
Takashi Iwaie1406342008-02-11 18:32:32 +010012706 .capsrc_nids = alc268_capsrc_nids,
Jonathan Woithe86c53bd2008-01-08 12:33:19 +010012707 .hp_nid = 0x03,
12708 .dig_out_nid = ALC268_DIGOUT_NID,
12709 .num_channel_mode = ARRAY_SIZE(alc268_modes),
12710 .channel_mode = alc268_modes,
12711 .input_mux = &alc268_capture_source,
12712 },
12713#endif
Kailang Yanga361d842007-06-05 12:30:55 +020012714};
12715
12716static int patch_alc268(struct hda_codec *codec)
12717{
12718 struct alc_spec *spec;
12719 int board_config;
Takashi Iwai22971e32009-02-10 11:56:44 +010012720 int i, has_beep, err;
Kailang Yanga361d842007-06-05 12:30:55 +020012721
12722 spec = kcalloc(1, sizeof(*spec), GFP_KERNEL);
12723 if (spec == NULL)
12724 return -ENOMEM;
12725
12726 codec->spec = spec;
12727
12728 board_config = snd_hda_check_board_config(codec, ALC268_MODEL_LAST,
12729 alc268_models,
12730 alc268_cfg_tbl);
12731
Takashi Iwai3abf2f32009-08-19 20:05:02 +020012732 if (board_config < 0 || board_config >= ALC268_MODEL_LAST)
12733 board_config = snd_hda_check_board_codec_sid_config(codec,
12734 ALC882_MODEL_LAST, alc268_models, alc268_ssid_cfg_tbl);
12735
Kailang Yanga361d842007-06-05 12:30:55 +020012736 if (board_config < 0 || board_config >= ALC268_MODEL_LAST) {
Takashi Iwai9a11f1a2009-07-28 16:01:20 +020012737 printk(KERN_INFO "hda_codec: %s: BIOS auto-probing.\n",
12738 codec->chip_name);
Kailang Yanga361d842007-06-05 12:30:55 +020012739 board_config = ALC268_AUTO;
12740 }
12741
12742 if (board_config == ALC268_AUTO) {
12743 /* automatic parse from the BIOS config */
12744 err = alc268_parse_auto_config(codec);
12745 if (err < 0) {
12746 alc_free(codec);
12747 return err;
12748 } else if (!err) {
12749 printk(KERN_INFO
12750 "hda_codec: Cannot set up configuration "
12751 "from BIOS. Using base mode...\n");
12752 board_config = ALC268_3ST;
12753 }
12754 }
12755
12756 if (board_config != ALC268_AUTO)
Takashi Iwaie9c364c2009-08-11 17:16:13 +020012757 setup_preset(codec, &alc268_presets[board_config]);
Kailang Yanga361d842007-06-05 12:30:55 +020012758
Kailang Yanga361d842007-06-05 12:30:55 +020012759 spec->stream_analog_playback = &alc268_pcm_analog_playback;
12760 spec->stream_analog_capture = &alc268_pcm_analog_capture;
Takashi Iwai63300792008-01-24 15:31:36 +010012761 spec->stream_analog_alt_capture = &alc268_pcm_analog_alt_capture;
Kailang Yanga361d842007-06-05 12:30:55 +020012762
Kailang Yanga361d842007-06-05 12:30:55 +020012763 spec->stream_digital_playback = &alc268_pcm_digital_playback;
12764
Takashi Iwai22971e32009-02-10 11:56:44 +010012765 has_beep = 0;
12766 for (i = 0; i < spec->num_mixers; i++) {
12767 if (spec->mixers[i] == alc268_beep_mixer) {
12768 has_beep = 1;
12769 break;
12770 }
12771 }
12772
12773 if (has_beep) {
12774 err = snd_hda_attach_beep_device(codec, 0x1);
12775 if (err < 0) {
12776 alc_free(codec);
12777 return err;
12778 }
12779 if (!query_amp_caps(codec, 0x1d, HDA_INPUT))
12780 /* override the amp caps for beep generator */
12781 snd_hda_override_amp_caps(codec, 0x1d, HDA_INPUT,
Takashi Iwaiaef9d312008-02-19 13:16:41 +010012782 (0x0c << AC_AMPCAP_OFFSET_SHIFT) |
12783 (0x0c << AC_AMPCAP_NUM_STEPS_SHIFT) |
12784 (0x07 << AC_AMPCAP_STEP_SIZE_SHIFT) |
12785 (0 << AC_AMPCAP_MUTE_SHIFT));
Takashi Iwai22971e32009-02-10 11:56:44 +010012786 }
Takashi Iwaiaef9d312008-02-19 13:16:41 +010012787
Takashi Iwai7e0e44d2009-02-19 08:15:49 +010012788 if (!spec->no_analog && !spec->adc_nids && spec->input_mux) {
Takashi Iwai3866f0b2008-01-15 12:37:42 +010012789 /* check whether NID 0x07 is valid */
12790 unsigned int wcap = get_wcaps(codec, 0x07);
Takashi Iwai85860c02008-02-19 15:00:15 +010012791 int i;
Kailang Yanga361d842007-06-05 12:30:55 +020012792
Takashi Iwaidefb5ab2009-10-07 15:12:27 +020012793 spec->capsrc_nids = alc268_capsrc_nids;
Takashi Iwai3866f0b2008-01-15 12:37:42 +010012794 /* get type */
Takashi Iwaia22d5432009-07-27 12:54:26 +020012795 wcap = get_wcaps_type(wcap);
Takashi Iwaifdbc6622009-08-19 00:18:10 +020012796 if (spec->auto_mic ||
12797 wcap != AC_WID_AUD_IN || spec->input_mux->num_items == 1) {
Takashi Iwai3866f0b2008-01-15 12:37:42 +010012798 spec->adc_nids = alc268_adc_nids_alt;
12799 spec->num_adc_nids = ARRAY_SIZE(alc268_adc_nids_alt);
Takashi Iwaidefb5ab2009-10-07 15:12:27 +020012800 if (spec->auto_mic)
12801 fixup_automic_adc(codec);
Takashi Iwaifdbc6622009-08-19 00:18:10 +020012802 if (spec->auto_mic || spec->input_mux->num_items == 1)
12803 add_mixer(spec, alc268_capture_nosrc_mixer);
12804 else
12805 add_mixer(spec, alc268_capture_alt_mixer);
Takashi Iwai3866f0b2008-01-15 12:37:42 +010012806 } else {
12807 spec->adc_nids = alc268_adc_nids;
12808 spec->num_adc_nids = ARRAY_SIZE(alc268_adc_nids);
Takashi Iwaid88897e2008-10-31 15:01:37 +010012809 add_mixer(spec, alc268_capture_mixer);
Kailang Yanga361d842007-06-05 12:30:55 +020012810 }
Takashi Iwai85860c02008-02-19 15:00:15 +010012811 /* set default input source */
12812 for (i = 0; i < spec->num_adc_nids; i++)
12813 snd_hda_codec_write_cache(codec, alc268_capsrc_nids[i],
12814 0, AC_VERB_SET_CONNECT_SEL,
Herton Ronaldo Krzesinski59085892009-08-11 22:33:09 -030012815 i < spec->num_mux_defs ?
12816 spec->input_mux[i].items[0].index :
Takashi Iwai85860c02008-02-19 15:00:15 +010012817 spec->input_mux->items[0].index);
Kailang Yanga361d842007-06-05 12:30:55 +020012818 }
Takashi Iwai2134ea42008-01-10 16:53:55 +010012819
12820 spec->vmaster_nid = 0x02;
12821
Kailang Yanga361d842007-06-05 12:30:55 +020012822 codec->patch_ops = alc_patch_ops;
12823 if (board_config == ALC268_AUTO)
12824 spec->init_hook = alc268_auto_init;
Kailang Yangea1fb292008-08-26 12:58:38 +020012825
Takashi Iwaidaead532008-11-28 12:55:36 +010012826 codec->proc_widget_hook = print_realtek_coef;
12827
Kailang Yanga361d842007-06-05 12:30:55 +020012828 return 0;
12829}
12830
12831/*
Kailang Yangf6a92242007-12-13 16:52:54 +010012832 * ALC269 channel source setting (2 channel)
12833 */
12834#define ALC269_DIGOUT_NID ALC880_DIGOUT_NID
12835
12836#define alc269_dac_nids alc260_dac_nids
12837
12838static hda_nid_t alc269_adc_nids[1] = {
12839 /* ADC1 */
Kailang Yangf53281e2008-07-18 12:36:43 +020012840 0x08,
12841};
12842
Takashi Iwaie01bf502008-08-21 16:25:07 +020012843static hda_nid_t alc269_capsrc_nids[1] = {
12844 0x23,
12845};
12846
12847/* NOTE: ADC2 (0x07) is connected from a recording *MIXER* (0x24),
12848 * not a mux!
12849 */
12850
Kailang Yangf6a92242007-12-13 16:52:54 +010012851#define alc269_modes alc260_modes
12852#define alc269_capture_source alc880_lg_lw_capture_source
12853
12854static struct snd_kcontrol_new alc269_base_mixer[] = {
12855 HDA_CODEC_VOLUME("Front Playback Volume", 0x02, 0x0, HDA_OUTPUT),
12856 HDA_CODEC_MUTE("Front Playback Switch", 0x14, 0x0, HDA_OUTPUT),
12857 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
12858 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
12859 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
12860 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
12861 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
12862 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
12863 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
12864 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
12865 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
12866 HDA_CODEC_MUTE_MONO("Mono Playback Switch", 0x16, 2, 0x0, HDA_OUTPUT),
12867 { } /* end */
12868};
12869
Kailang Yang60db6b52008-08-26 13:13:00 +020012870static struct snd_kcontrol_new alc269_quanta_fl1_mixer[] = {
12871 /* output mixer control */
12872 HDA_BIND_VOL("Master Playback Volume", &alc268_acer_bind_master_vol),
12873 {
12874 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
12875 .name = "Master Playback Switch",
12876 .info = snd_hda_mixer_amp_switch_info,
12877 .get = snd_hda_mixer_amp_switch_get,
12878 .put = alc268_acer_master_sw_put,
12879 .private_value = HDA_COMPOSE_AMP_VAL(0x14, 3, 0, HDA_OUTPUT),
12880 },
12881 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
12882 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
12883 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
12884 HDA_CODEC_VOLUME("Internal Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
12885 HDA_CODEC_MUTE("Internal Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
12886 HDA_CODEC_VOLUME("Internal Mic Boost", 0x19, 0, HDA_INPUT),
Kailang Yang60db6b52008-08-26 13:13:00 +020012887 { }
12888};
12889
Tony Vroon64154832008-11-06 15:08:49 +000012890static struct snd_kcontrol_new alc269_lifebook_mixer[] = {
12891 /* output mixer control */
12892 HDA_BIND_VOL("Master Playback Volume", &alc268_acer_bind_master_vol),
12893 {
12894 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
12895 .name = "Master Playback Switch",
12896 .info = snd_hda_mixer_amp_switch_info,
12897 .get = snd_hda_mixer_amp_switch_get,
12898 .put = alc268_acer_master_sw_put,
12899 .private_value = HDA_COMPOSE_AMP_VAL(0x14, 3, 0, HDA_OUTPUT),
12900 },
12901 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
12902 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
12903 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
12904 HDA_CODEC_VOLUME("Internal Mic Playback Volume", 0x0b, 0x01, HDA_INPUT),
12905 HDA_CODEC_MUTE("Internal Mic Playback Switch", 0x0b, 0x01, HDA_INPUT),
12906 HDA_CODEC_VOLUME("Internal Mic Boost", 0x19, 0, HDA_INPUT),
12907 HDA_CODEC_VOLUME("Dock Mic Playback Volume", 0x0b, 0x03, HDA_INPUT),
12908 HDA_CODEC_MUTE("Dock Mic Playback Switch", 0x0b, 0x03, HDA_INPUT),
12909 HDA_CODEC_VOLUME("Dock Mic Boost", 0x1b, 0, HDA_INPUT),
Tony Vroon64154832008-11-06 15:08:49 +000012910 { }
12911};
12912
Kailang Yangf53281e2008-07-18 12:36:43 +020012913static struct snd_kcontrol_new alc269_eeepc_mixer[] = {
Takashi Iwaiaa202452009-07-03 15:00:54 +020012914 HDA_CODEC_MUTE("Speaker Playback Switch", 0x14, 0x0, HDA_OUTPUT),
Darren Salt508f7112009-07-08 15:29:49 +010012915 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x02, 0x0, HDA_OUTPUT),
Takashi Iwaiaa202452009-07-03 15:00:54 +020012916 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
Darren Salt508f7112009-07-08 15:29:49 +010012917 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x03, 0x0, HDA_OUTPUT),
Kailang Yangf53281e2008-07-18 12:36:43 +020012918 { } /* end */
12919};
12920
Kailang Yangf6a92242007-12-13 16:52:54 +010012921/* capture mixer elements */
Kailang Yangf53281e2008-07-18 12:36:43 +020012922static struct snd_kcontrol_new alc269_epc_capture_mixer[] = {
12923 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
12924 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
Takashi Iwai26f5df22008-11-03 17:39:46 +010012925 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
12926 { } /* end */
12927};
12928
12929/* FSC amilo */
Takashi Iwaiaa202452009-07-03 15:00:54 +020012930#define alc269_fujitsu_mixer alc269_eeepc_mixer
Kailang Yangf53281e2008-07-18 12:36:43 +020012931
Kailang Yang60db6b52008-08-26 13:13:00 +020012932static struct hda_verb alc269_quanta_fl1_verbs[] = {
12933 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01},
12934 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
12935 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
12936 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
12937 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
12938 {0x1d, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
12939 { }
12940};
12941
Tony Vroon64154832008-11-06 15:08:49 +000012942static struct hda_verb alc269_lifebook_verbs[] = {
12943 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01},
12944 {0x1a, AC_VERB_SET_CONNECT_SEL, 0x01},
12945 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
12946 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
12947 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
12948 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
12949 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
12950 {0x1a, AC_VERB_SET_UNSOLICITED_ENABLE, ALC880_HP_EVENT | AC_USRSP_EN},
12951 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
12952 {0x1d, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
12953 { }
12954};
12955
Kailang Yang60db6b52008-08-26 13:13:00 +020012956/* toggle speaker-output according to the hp-jack state */
12957static void alc269_quanta_fl1_speaker_automute(struct hda_codec *codec)
12958{
12959 unsigned int present;
12960 unsigned char bits;
12961
Wu Fengguang864f92b2009-11-18 12:38:02 +080012962 present = snd_hda_jack_detect(codec, 0x15);
Kailang Yang60db6b52008-08-26 13:13:00 +020012963 bits = present ? AMP_IN_MUTE(0) : 0;
12964 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_INPUT, 0,
12965 AMP_IN_MUTE(0), bits);
12966 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_INPUT, 1,
12967 AMP_IN_MUTE(0), bits);
12968
12969 snd_hda_codec_write(codec, 0x20, 0,
12970 AC_VERB_SET_COEF_INDEX, 0x0c);
12971 snd_hda_codec_write(codec, 0x20, 0,
12972 AC_VERB_SET_PROC_COEF, 0x680);
12973
12974 snd_hda_codec_write(codec, 0x20, 0,
12975 AC_VERB_SET_COEF_INDEX, 0x0c);
12976 snd_hda_codec_write(codec, 0x20, 0,
12977 AC_VERB_SET_PROC_COEF, 0x480);
12978}
12979
Tony Vroon64154832008-11-06 15:08:49 +000012980/* toggle speaker-output according to the hp-jacks state */
12981static void alc269_lifebook_speaker_automute(struct hda_codec *codec)
12982{
12983 unsigned int present;
12984 unsigned char bits;
12985
12986 /* Check laptop headphone socket */
Wu Fengguang864f92b2009-11-18 12:38:02 +080012987 present = snd_hda_jack_detect(codec, 0x15);
Tony Vroon64154832008-11-06 15:08:49 +000012988
12989 /* Check port replicator headphone socket */
Wu Fengguang864f92b2009-11-18 12:38:02 +080012990 present |= snd_hda_jack_detect(codec, 0x1a);
Tony Vroon64154832008-11-06 15:08:49 +000012991
12992 bits = present ? AMP_IN_MUTE(0) : 0;
12993 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_INPUT, 0,
12994 AMP_IN_MUTE(0), bits);
12995 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_INPUT, 1,
12996 AMP_IN_MUTE(0), bits);
12997
12998 snd_hda_codec_write(codec, 0x20, 0,
12999 AC_VERB_SET_COEF_INDEX, 0x0c);
13000 snd_hda_codec_write(codec, 0x20, 0,
13001 AC_VERB_SET_PROC_COEF, 0x680);
13002
13003 snd_hda_codec_write(codec, 0x20, 0,
13004 AC_VERB_SET_COEF_INDEX, 0x0c);
13005 snd_hda_codec_write(codec, 0x20, 0,
13006 AC_VERB_SET_PROC_COEF, 0x480);
13007}
13008
Tony Vroon64154832008-11-06 15:08:49 +000013009static void alc269_lifebook_mic_autoswitch(struct hda_codec *codec)
13010{
13011 unsigned int present_laptop;
13012 unsigned int present_dock;
13013
Wu Fengguang864f92b2009-11-18 12:38:02 +080013014 present_laptop = snd_hda_jack_detect(codec, 0x18);
13015 present_dock = snd_hda_jack_detect(codec, 0x1b);
Tony Vroon64154832008-11-06 15:08:49 +000013016
13017 /* Laptop mic port overrides dock mic port, design decision */
13018 if (present_dock)
13019 snd_hda_codec_write(codec, 0x23, 0,
13020 AC_VERB_SET_CONNECT_SEL, 0x3);
13021 if (present_laptop)
13022 snd_hda_codec_write(codec, 0x23, 0,
13023 AC_VERB_SET_CONNECT_SEL, 0x0);
13024 if (!present_dock && !present_laptop)
13025 snd_hda_codec_write(codec, 0x23, 0,
13026 AC_VERB_SET_CONNECT_SEL, 0x1);
13027}
13028
Kailang Yang60db6b52008-08-26 13:13:00 +020013029static void alc269_quanta_fl1_unsol_event(struct hda_codec *codec,
13030 unsigned int res)
13031{
Takashi Iwai4f5d17062009-08-11 18:17:46 +020013032 switch (res >> 26) {
13033 case ALC880_HP_EVENT:
Kailang Yang60db6b52008-08-26 13:13:00 +020013034 alc269_quanta_fl1_speaker_automute(codec);
Takashi Iwai4f5d17062009-08-11 18:17:46 +020013035 break;
13036 case ALC880_MIC_EVENT:
13037 alc_mic_automute(codec);
13038 break;
13039 }
Kailang Yang60db6b52008-08-26 13:13:00 +020013040}
13041
Tony Vroon64154832008-11-06 15:08:49 +000013042static void alc269_lifebook_unsol_event(struct hda_codec *codec,
13043 unsigned int res)
13044{
13045 if ((res >> 26) == ALC880_HP_EVENT)
13046 alc269_lifebook_speaker_automute(codec);
13047 if ((res >> 26) == ALC880_MIC_EVENT)
13048 alc269_lifebook_mic_autoswitch(codec);
13049}
13050
Takashi Iwai4f5d17062009-08-11 18:17:46 +020013051static void alc269_quanta_fl1_setup(struct hda_codec *codec)
13052{
13053 struct alc_spec *spec = codec->spec;
13054 spec->ext_mic.pin = 0x18;
13055 spec->ext_mic.mux_idx = 0;
13056 spec->int_mic.pin = 0x19;
13057 spec->int_mic.mux_idx = 1;
13058 spec->auto_mic = 1;
13059}
13060
Kailang Yang60db6b52008-08-26 13:13:00 +020013061static void alc269_quanta_fl1_init_hook(struct hda_codec *codec)
13062{
13063 alc269_quanta_fl1_speaker_automute(codec);
Takashi Iwai4f5d17062009-08-11 18:17:46 +020013064 alc_mic_automute(codec);
Kailang Yang60db6b52008-08-26 13:13:00 +020013065}
13066
Tony Vroon64154832008-11-06 15:08:49 +000013067static void alc269_lifebook_init_hook(struct hda_codec *codec)
13068{
13069 alc269_lifebook_speaker_automute(codec);
13070 alc269_lifebook_mic_autoswitch(codec);
13071}
13072
Kailang Yang60db6b52008-08-26 13:13:00 +020013073static struct hda_verb alc269_eeepc_dmic_init_verbs[] = {
13074 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01},
13075 {0x23, AC_VERB_SET_CONNECT_SEL, 0x05},
13076 {0x02, AC_VERB_SET_AMP_GAIN_MUTE, 0xb026 },
13077 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, (0x7019 | (0x00 << 8))},
13078 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
13079 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
13080 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
13081 {}
13082};
13083
13084static struct hda_verb alc269_eeepc_amic_init_verbs[] = {
13085 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01},
13086 {0x23, AC_VERB_SET_CONNECT_SEL, 0x01},
13087 {0x02, AC_VERB_SET_AMP_GAIN_MUTE, 0xb026 },
13088 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, (0x701b | (0x00 << 8))},
13089 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
13090 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
13091 {}
13092};
13093
13094/* toggle speaker-output according to the hp-jack state */
13095static void alc269_speaker_automute(struct hda_codec *codec)
13096{
13097 unsigned int present;
13098 unsigned char bits;
13099
Wu Fengguang864f92b2009-11-18 12:38:02 +080013100 present = snd_hda_jack_detect(codec, 0x15);
Kailang Yang60db6b52008-08-26 13:13:00 +020013101 bits = present ? AMP_IN_MUTE(0) : 0;
13102 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_INPUT, 0,
13103 AMP_IN_MUTE(0), bits);
13104 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_INPUT, 1,
13105 AMP_IN_MUTE(0), bits);
13106}
13107
Kailang Yang60db6b52008-08-26 13:13:00 +020013108/* unsolicited event for HP jack sensing */
Takashi Iwai4f5d17062009-08-11 18:17:46 +020013109static void alc269_eeepc_unsol_event(struct hda_codec *codec,
Kailang Yang60db6b52008-08-26 13:13:00 +020013110 unsigned int res)
13111{
Takashi Iwai4f5d17062009-08-11 18:17:46 +020013112 switch (res >> 26) {
13113 case ALC880_HP_EVENT:
Kailang Yang60db6b52008-08-26 13:13:00 +020013114 alc269_speaker_automute(codec);
Takashi Iwai4f5d17062009-08-11 18:17:46 +020013115 break;
13116 case ALC880_MIC_EVENT:
13117 alc_mic_automute(codec);
13118 break;
13119 }
Kailang Yang60db6b52008-08-26 13:13:00 +020013120}
13121
Takashi Iwai4f5d17062009-08-11 18:17:46 +020013122static void alc269_eeepc_dmic_setup(struct hda_codec *codec)
13123{
13124 struct alc_spec *spec = codec->spec;
13125 spec->ext_mic.pin = 0x18;
13126 spec->ext_mic.mux_idx = 0;
13127 spec->int_mic.pin = 0x12;
13128 spec->int_mic.mux_idx = 5;
13129 spec->auto_mic = 1;
13130}
13131
13132static void alc269_eeepc_amic_setup(struct hda_codec *codec)
13133{
13134 struct alc_spec *spec = codec->spec;
13135 spec->ext_mic.pin = 0x18;
13136 spec->ext_mic.mux_idx = 0;
13137 spec->int_mic.pin = 0x19;
13138 spec->int_mic.mux_idx = 1;
13139 spec->auto_mic = 1;
13140}
13141
13142static void alc269_eeepc_inithook(struct hda_codec *codec)
Kailang Yang60db6b52008-08-26 13:13:00 +020013143{
13144 alc269_speaker_automute(codec);
Takashi Iwai4f5d17062009-08-11 18:17:46 +020013145 alc_mic_automute(codec);
Kailang Yang60db6b52008-08-26 13:13:00 +020013146}
13147
Kailang Yangf6a92242007-12-13 16:52:54 +010013148/*
13149 * generic initialization of ADC, input mixers and output mixers
13150 */
13151static struct hda_verb alc269_init_verbs[] = {
13152 /*
13153 * Unmute ADC0 and set the default input to mic-in
13154 */
Kailang Yang60db6b52008-08-26 13:13:00 +020013155 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Kailang Yangf6a92242007-12-13 16:52:54 +010013156
13157 /* Mute input amps (PCBeep, Line In, Mic 1 & Mic 2) of the
13158 * analog-loopback mixer widget
13159 * Note: PASD motherboards uses the Line In 2 as the input for
13160 * front panel mic (mic 2)
13161 */
13162 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
13163 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
13164 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
13165 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
13166 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
13167 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
13168
13169 /*
13170 * Set up output mixers (0x0c - 0x0e)
13171 */
13172 /* set vol=0 to output mixers */
13173 {0x02, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
13174 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
13175
13176 /* set up input amps for analog loopback */
13177 /* Amp Indices: DAC = 0, mixer = 1 */
13178 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13179 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13180 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13181 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13182 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13183 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13184
13185 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
13186 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
13187 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
13188 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
13189 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
13190 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
13191 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
13192
13193 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13194 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13195 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
13196 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
13197 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
13198 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
13199 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
13200
13201 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
13202 {0x15, AC_VERB_SET_CONNECT_SEL, 0x00},
13203
13204 /* FIXME: use matrix-type input source selection */
13205 /* Mixer elements: 0x18, 19, 1a, 1b, 1d, 0b */
13206 /* Input mixer1: unmute Mic, F-Mic, Line, CD inputs */
Kailang Yang60db6b52008-08-26 13:13:00 +020013207 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
13208 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Kailang Yangf6a92242007-12-13 16:52:54 +010013209 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
13210 {0x24, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
13211
13212 /* set EAPD */
13213 {0x14, AC_VERB_SET_EAPD_BTLENABLE, 2},
13214 {0x15, AC_VERB_SET_EAPD_BTLENABLE, 2},
13215 { }
13216};
13217
Takashi Iwai9d0b71b2009-08-24 14:10:30 +020013218#define alc269_auto_create_multi_out_ctls \
13219 alc268_auto_create_multi_out_ctls
Takashi Iwai05f5f472009-08-25 13:10:18 +020013220#define alc269_auto_create_input_ctls \
13221 alc268_auto_create_input_ctls
Kailang Yangf6a92242007-12-13 16:52:54 +010013222
13223#ifdef CONFIG_SND_HDA_POWER_SAVE
13224#define alc269_loopbacks alc880_loopbacks
13225#endif
13226
Sasha Alexandrdef319f2009-06-16 16:00:15 -040013227/* pcm configuration: identical with ALC880 */
Kailang Yangf6a92242007-12-13 16:52:54 +010013228#define alc269_pcm_analog_playback alc880_pcm_analog_playback
13229#define alc269_pcm_analog_capture alc880_pcm_analog_capture
13230#define alc269_pcm_digital_playback alc880_pcm_digital_playback
13231#define alc269_pcm_digital_capture alc880_pcm_digital_capture
13232
Takashi Iwaif03d3112009-03-05 14:18:16 +010013233static struct hda_pcm_stream alc269_44k_pcm_analog_playback = {
13234 .substreams = 1,
13235 .channels_min = 2,
13236 .channels_max = 8,
13237 .rates = SNDRV_PCM_RATE_44100, /* fixed rate */
13238 /* NID is set in alc_build_pcms */
13239 .ops = {
13240 .open = alc880_playback_pcm_open,
13241 .prepare = alc880_playback_pcm_prepare,
13242 .cleanup = alc880_playback_pcm_cleanup
13243 },
13244};
13245
13246static struct hda_pcm_stream alc269_44k_pcm_analog_capture = {
13247 .substreams = 1,
13248 .channels_min = 2,
13249 .channels_max = 2,
13250 .rates = SNDRV_PCM_RATE_44100, /* fixed rate */
13251 /* NID is set in alc_build_pcms */
13252};
13253
Kailang Yangf6a92242007-12-13 16:52:54 +010013254/*
13255 * BIOS auto configuration
13256 */
13257static int alc269_parse_auto_config(struct hda_codec *codec)
13258{
13259 struct alc_spec *spec = codec->spec;
Takashi Iwaicfb9fb52009-02-06 17:34:03 +010013260 int err;
Kailang Yangf6a92242007-12-13 16:52:54 +010013261 static hda_nid_t alc269_ignore[] = { 0x1d, 0 };
13262
13263 err = snd_hda_parse_pin_def_config(codec, &spec->autocfg,
13264 alc269_ignore);
13265 if (err < 0)
13266 return err;
13267
13268 err = alc269_auto_create_multi_out_ctls(spec, &spec->autocfg);
13269 if (err < 0)
13270 return err;
Takashi Iwai05f5f472009-08-25 13:10:18 +020013271 err = alc269_auto_create_input_ctls(codec, &spec->autocfg);
Kailang Yangf6a92242007-12-13 16:52:54 +010013272 if (err < 0)
13273 return err;
13274
13275 spec->multiout.max_channels = spec->multiout.num_dacs * 2;
13276
Takashi Iwai0852d7a2009-02-11 11:35:15 +010013277 if (spec->autocfg.dig_outs)
Kailang Yangf6a92242007-12-13 16:52:54 +010013278 spec->multiout.dig_out_nid = ALC269_DIGOUT_NID;
13279
Takashi Iwai603c4012008-07-30 15:01:44 +020013280 if (spec->kctls.list)
Takashi Iwaid88897e2008-10-31 15:01:37 +010013281 add_mixer(spec, spec->kctls.list);
Kailang Yangf6a92242007-12-13 16:52:54 +010013282
Takashi Iwaid88897e2008-10-31 15:01:37 +010013283 add_verb(spec, alc269_init_verbs);
Kailang Yangf6a92242007-12-13 16:52:54 +010013284 spec->num_mux_defs = 1;
Herton Ronaldo Krzesinski61b9b9b2009-01-28 09:16:33 -020013285 spec->input_mux = &spec->private_imux[0];
Takashi Iwaie01bf502008-08-21 16:25:07 +020013286 /* set default input source */
13287 snd_hda_codec_write_cache(codec, alc269_capsrc_nids[0],
13288 0, AC_VERB_SET_CONNECT_SEL,
13289 spec->input_mux->items[0].index);
Kailang Yangf6a92242007-12-13 16:52:54 +010013290
13291 err = alc_auto_add_mic_boost(codec);
13292 if (err < 0)
13293 return err;
13294
Takashi Iwai7e0e44d2009-02-19 08:15:49 +010013295 if (!spec->cap_mixer && !spec->no_analog)
Takashi Iwaib59bdf32009-08-11 09:47:30 +020013296 set_capture_mixer(codec);
Kailang Yangf53281e2008-07-18 12:36:43 +020013297
Takashi Iwai1d955eb2009-06-29 11:33:53 +020013298 alc_ssid_check(codec, 0x15, 0x1b, 0x14);
13299
Kailang Yangf6a92242007-12-13 16:52:54 +010013300 return 1;
13301}
13302
Takashi Iwaie9af4f32009-08-29 23:23:08 +020013303#define alc269_auto_init_multi_out alc268_auto_init_multi_out
13304#define alc269_auto_init_hp_out alc268_auto_init_hp_out
Kailang Yangf6a92242007-12-13 16:52:54 +010013305#define alc269_auto_init_analog_input alc882_auto_init_analog_input
13306
13307
13308/* init callback for auto-configuration model -- overriding the default init */
13309static void alc269_auto_init(struct hda_codec *codec)
13310{
Takashi Iwaif6c7e542008-02-12 18:32:23 +010013311 struct alc_spec *spec = codec->spec;
Kailang Yangf6a92242007-12-13 16:52:54 +010013312 alc269_auto_init_multi_out(codec);
13313 alc269_auto_init_hp_out(codec);
13314 alc269_auto_init_analog_input(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +010013315 if (spec->unsol_event)
Kailang Yang7fb0d782008-10-15 11:12:35 +020013316 alc_inithook(codec);
Kailang Yangf6a92242007-12-13 16:52:54 +010013317}
13318
13319/*
13320 * configuration and preset
13321 */
13322static const char *alc269_models[ALC269_MODEL_LAST] = {
Kailang Yang60db6b52008-08-26 13:13:00 +020013323 [ALC269_BASIC] = "basic",
Takashi Iwai2922c9a2008-08-27 18:12:42 +020013324 [ALC269_QUANTA_FL1] = "quanta",
13325 [ALC269_ASUS_EEEPC_P703] = "eeepc-p703",
Takashi Iwai26f5df22008-11-03 17:39:46 +010013326 [ALC269_ASUS_EEEPC_P901] = "eeepc-p901",
Tony Vroon64154832008-11-06 15:08:49 +000013327 [ALC269_FUJITSU] = "fujitsu",
Takashi Iwai3d3792c2009-09-11 07:50:47 +020013328 [ALC269_LIFEBOOK] = "lifebook",
13329 [ALC269_AUTO] = "auto",
Kailang Yangf6a92242007-12-13 16:52:54 +010013330};
13331
13332static struct snd_pci_quirk alc269_cfg_tbl[] = {
Kailang Yang60db6b52008-08-26 13:13:00 +020013333 SND_PCI_QUIRK(0x17aa, 0x3bf8, "Quanta FL1", ALC269_QUANTA_FL1),
Kailang Yangf53281e2008-07-18 12:36:43 +020013334 SND_PCI_QUIRK(0x1043, 0x8330, "ASUS Eeepc P703 P900A",
13335 ALC269_ASUS_EEEPC_P703),
Kailang Yang622e84c2009-04-21 07:39:04 +020013336 SND_PCI_QUIRK(0x1043, 0x1883, "ASUS F81Se", ALC269_ASUS_EEEPC_P703),
13337 SND_PCI_QUIRK(0x1043, 0x16a3, "ASUS F5Q", ALC269_ASUS_EEEPC_P703),
13338 SND_PCI_QUIRK(0x1043, 0x1723, "ASUS P80", ALC269_ASUS_EEEPC_P703),
13339 SND_PCI_QUIRK(0x1043, 0x1773, "ASUS U20A", ALC269_ASUS_EEEPC_P703),
13340 SND_PCI_QUIRK(0x1043, 0x1743, "ASUS U80", ALC269_ASUS_EEEPC_P703),
13341 SND_PCI_QUIRK(0x1043, 0x1653, "ASUS U50", ALC269_ASUS_EEEPC_P703),
Kailang Yangf53281e2008-07-18 12:36:43 +020013342 SND_PCI_QUIRK(0x1043, 0x831a, "ASUS Eeepc P901",
13343 ALC269_ASUS_EEEPC_P901),
Kailang Yang60db6b52008-08-26 13:13:00 +020013344 SND_PCI_QUIRK(0x1043, 0x834a, "ASUS Eeepc S101",
13345 ALC269_ASUS_EEEPC_P901),
Kailang Yang622e84c2009-04-21 07:39:04 +020013346 SND_PCI_QUIRK(0x1043, 0x16e3, "ASUS UX50", ALC269_ASUS_EEEPC_P901),
Takashi Iwai26f5df22008-11-03 17:39:46 +010013347 SND_PCI_QUIRK(0x1734, 0x115d, "FSC Amilo", ALC269_FUJITSU),
Tony Vroon64154832008-11-06 15:08:49 +000013348 SND_PCI_QUIRK(0x10cf, 0x1475, "Lifebook ICH9M-based", ALC269_LIFEBOOK),
Kailang Yangf6a92242007-12-13 16:52:54 +010013349 {}
13350};
13351
13352static struct alc_config_preset alc269_presets[] = {
13353 [ALC269_BASIC] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010013354 .mixers = { alc269_base_mixer },
Kailang Yangf6a92242007-12-13 16:52:54 +010013355 .init_verbs = { alc269_init_verbs },
13356 .num_dacs = ARRAY_SIZE(alc269_dac_nids),
13357 .dac_nids = alc269_dac_nids,
13358 .hp_nid = 0x03,
13359 .num_channel_mode = ARRAY_SIZE(alc269_modes),
13360 .channel_mode = alc269_modes,
13361 .input_mux = &alc269_capture_source,
13362 },
Kailang Yang60db6b52008-08-26 13:13:00 +020013363 [ALC269_QUANTA_FL1] = {
13364 .mixers = { alc269_quanta_fl1_mixer },
13365 .init_verbs = { alc269_init_verbs, alc269_quanta_fl1_verbs },
13366 .num_dacs = ARRAY_SIZE(alc269_dac_nids),
13367 .dac_nids = alc269_dac_nids,
13368 .hp_nid = 0x03,
13369 .num_channel_mode = ARRAY_SIZE(alc269_modes),
13370 .channel_mode = alc269_modes,
13371 .input_mux = &alc269_capture_source,
13372 .unsol_event = alc269_quanta_fl1_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020013373 .setup = alc269_quanta_fl1_setup,
Kailang Yang60db6b52008-08-26 13:13:00 +020013374 .init_hook = alc269_quanta_fl1_init_hook,
13375 },
Kailang Yangf53281e2008-07-18 12:36:43 +020013376 [ALC269_ASUS_EEEPC_P703] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010013377 .mixers = { alc269_eeepc_mixer },
13378 .cap_mixer = alc269_epc_capture_mixer,
Kailang Yangf53281e2008-07-18 12:36:43 +020013379 .init_verbs = { alc269_init_verbs,
13380 alc269_eeepc_amic_init_verbs },
13381 .num_dacs = ARRAY_SIZE(alc269_dac_nids),
13382 .dac_nids = alc269_dac_nids,
13383 .hp_nid = 0x03,
13384 .num_channel_mode = ARRAY_SIZE(alc269_modes),
13385 .channel_mode = alc269_modes,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020013386 .unsol_event = alc269_eeepc_unsol_event,
13387 .setup = alc269_eeepc_amic_setup,
13388 .init_hook = alc269_eeepc_inithook,
Kailang Yangf53281e2008-07-18 12:36:43 +020013389 },
13390 [ALC269_ASUS_EEEPC_P901] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010013391 .mixers = { alc269_eeepc_mixer },
13392 .cap_mixer = alc269_epc_capture_mixer,
Kailang Yangf53281e2008-07-18 12:36:43 +020013393 .init_verbs = { alc269_init_verbs,
13394 alc269_eeepc_dmic_init_verbs },
13395 .num_dacs = ARRAY_SIZE(alc269_dac_nids),
13396 .dac_nids = alc269_dac_nids,
13397 .hp_nid = 0x03,
13398 .num_channel_mode = ARRAY_SIZE(alc269_modes),
13399 .channel_mode = alc269_modes,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020013400 .unsol_event = alc269_eeepc_unsol_event,
13401 .setup = alc269_eeepc_dmic_setup,
13402 .init_hook = alc269_eeepc_inithook,
Kailang Yangf53281e2008-07-18 12:36:43 +020013403 },
Takashi Iwai26f5df22008-11-03 17:39:46 +010013404 [ALC269_FUJITSU] = {
Takashi Iwai45bdd1c2009-02-06 16:11:25 +010013405 .mixers = { alc269_fujitsu_mixer },
Takashi Iwai26f5df22008-11-03 17:39:46 +010013406 .cap_mixer = alc269_epc_capture_mixer,
13407 .init_verbs = { alc269_init_verbs,
13408 alc269_eeepc_dmic_init_verbs },
13409 .num_dacs = ARRAY_SIZE(alc269_dac_nids),
13410 .dac_nids = alc269_dac_nids,
13411 .hp_nid = 0x03,
13412 .num_channel_mode = ARRAY_SIZE(alc269_modes),
13413 .channel_mode = alc269_modes,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020013414 .unsol_event = alc269_eeepc_unsol_event,
13415 .setup = alc269_eeepc_dmic_setup,
13416 .init_hook = alc269_eeepc_inithook,
Takashi Iwai26f5df22008-11-03 17:39:46 +010013417 },
Tony Vroon64154832008-11-06 15:08:49 +000013418 [ALC269_LIFEBOOK] = {
13419 .mixers = { alc269_lifebook_mixer },
13420 .init_verbs = { alc269_init_verbs, alc269_lifebook_verbs },
13421 .num_dacs = ARRAY_SIZE(alc269_dac_nids),
13422 .dac_nids = alc269_dac_nids,
13423 .hp_nid = 0x03,
13424 .num_channel_mode = ARRAY_SIZE(alc269_modes),
13425 .channel_mode = alc269_modes,
13426 .input_mux = &alc269_capture_source,
13427 .unsol_event = alc269_lifebook_unsol_event,
13428 .init_hook = alc269_lifebook_init_hook,
13429 },
Kailang Yangf6a92242007-12-13 16:52:54 +010013430};
13431
13432static int patch_alc269(struct hda_codec *codec)
13433{
13434 struct alc_spec *spec;
13435 int board_config;
13436 int err;
13437
13438 spec = kzalloc(sizeof(*spec), GFP_KERNEL);
13439 if (spec == NULL)
13440 return -ENOMEM;
13441
13442 codec->spec = spec;
13443
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +020013444 alc_fix_pll_init(codec, 0x20, 0x04, 15);
13445
Kailang Yangf6a92242007-12-13 16:52:54 +010013446 board_config = snd_hda_check_board_config(codec, ALC269_MODEL_LAST,
13447 alc269_models,
13448 alc269_cfg_tbl);
13449
13450 if (board_config < 0) {
Takashi Iwai9a11f1a2009-07-28 16:01:20 +020013451 printk(KERN_INFO "hda_codec: %s: BIOS auto-probing.\n",
13452 codec->chip_name);
Kailang Yangf6a92242007-12-13 16:52:54 +010013453 board_config = ALC269_AUTO;
13454 }
13455
13456 if (board_config == ALC269_AUTO) {
13457 /* automatic parse from the BIOS config */
13458 err = alc269_parse_auto_config(codec);
13459 if (err < 0) {
13460 alc_free(codec);
13461 return err;
13462 } else if (!err) {
13463 printk(KERN_INFO
13464 "hda_codec: Cannot set up configuration "
13465 "from BIOS. Using base mode...\n");
13466 board_config = ALC269_BASIC;
13467 }
13468 }
13469
Kusanagi Kouichi680cd532009-02-05 00:00:58 +090013470 err = snd_hda_attach_beep_device(codec, 0x1);
13471 if (err < 0) {
13472 alc_free(codec);
13473 return err;
13474 }
13475
Kailang Yangf6a92242007-12-13 16:52:54 +010013476 if (board_config != ALC269_AUTO)
Takashi Iwaie9c364c2009-08-11 17:16:13 +020013477 setup_preset(codec, &alc269_presets[board_config]);
Kailang Yangf6a92242007-12-13 16:52:54 +010013478
Takashi Iwaif03d3112009-03-05 14:18:16 +010013479 if (codec->subsystem_id == 0x17aa3bf8) {
13480 /* Due to a hardware problem on Lenovo Ideadpad, we need to
13481 * fix the sample rate of analog I/O to 44.1kHz
13482 */
13483 spec->stream_analog_playback = &alc269_44k_pcm_analog_playback;
13484 spec->stream_analog_capture = &alc269_44k_pcm_analog_capture;
13485 } else {
13486 spec->stream_analog_playback = &alc269_pcm_analog_playback;
13487 spec->stream_analog_capture = &alc269_pcm_analog_capture;
13488 }
Kailang Yangf6a92242007-12-13 16:52:54 +010013489 spec->stream_digital_playback = &alc269_pcm_digital_playback;
13490 spec->stream_digital_capture = &alc269_pcm_digital_capture;
13491
13492 spec->adc_nids = alc269_adc_nids;
13493 spec->num_adc_nids = ARRAY_SIZE(alc269_adc_nids);
Takashi Iwaie01bf502008-08-21 16:25:07 +020013494 spec->capsrc_nids = alc269_capsrc_nids;
Takashi Iwaif9e336f2008-10-31 16:37:07 +010013495 if (!spec->cap_mixer)
Takashi Iwaib59bdf32009-08-11 09:47:30 +020013496 set_capture_mixer(codec);
Takashi Iwai45bdd1c2009-02-06 16:11:25 +010013497 set_beep_amp(spec, 0x0b, 0x04, HDA_INPUT);
Kailang Yangf6a92242007-12-13 16:52:54 +010013498
Takashi Iwai100d5eb2009-08-10 11:55:51 +020013499 spec->vmaster_nid = 0x02;
13500
Kailang Yangf6a92242007-12-13 16:52:54 +010013501 codec->patch_ops = alc_patch_ops;
13502 if (board_config == ALC269_AUTO)
13503 spec->init_hook = alc269_auto_init;
13504#ifdef CONFIG_SND_HDA_POWER_SAVE
13505 if (!spec->loopback.amplist)
13506 spec->loopback.amplist = alc269_loopbacks;
13507#endif
Takashi Iwaidaead532008-11-28 12:55:36 +010013508 codec->proc_widget_hook = print_realtek_coef;
Kailang Yangf6a92242007-12-13 16:52:54 +010013509
13510 return 0;
13511}
13512
13513/*
Kailang Yangdf694da2005-12-05 19:42:22 +010013514 * ALC861 channel source setting (2/6 channel selection for 3-stack)
13515 */
13516
13517/*
13518 * set the path ways for 2 channel output
13519 * need to set the codec line out and mic 1 pin widgets to inputs
13520 */
13521static struct hda_verb alc861_threestack_ch2_init[] = {
13522 /* set pin widget 1Ah (line in) for input */
13523 { 0x0c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013524 /* set pin widget 18h (mic1/2) for input, for mic also enable
13525 * the vref
13526 */
Kailang Yangdf694da2005-12-05 19:42:22 +010013527 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
13528
Takashi Iwai9c7f8522006-06-28 15:08:22 +020013529 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0xb00c },
13530#if 0
13531 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8)) }, /*mic*/
13532 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x02 << 8)) }, /*line-in*/
13533#endif
Kailang Yangdf694da2005-12-05 19:42:22 +010013534 { } /* end */
13535};
13536/*
13537 * 6ch mode
13538 * need to set the codec line out and mic 1 pin widgets to outputs
13539 */
13540static struct hda_verb alc861_threestack_ch6_init[] = {
13541 /* set pin widget 1Ah (line in) for output (Back Surround)*/
13542 { 0x0c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
13543 /* set pin widget 18h (mic1) for output (CLFE)*/
13544 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
13545
13546 { 0x0c, AC_VERB_SET_CONNECT_SEL, 0x00 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +020013547 { 0x0d, AC_VERB_SET_CONNECT_SEL, 0x00 },
Kailang Yangdf694da2005-12-05 19:42:22 +010013548
Takashi Iwai9c7f8522006-06-28 15:08:22 +020013549 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0xb080 },
13550#if 0
13551 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x01 << 8)) }, /*mic*/
13552 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8)) }, /*line in*/
13553#endif
Kailang Yangdf694da2005-12-05 19:42:22 +010013554 { } /* end */
13555};
13556
13557static struct hda_channel_mode alc861_threestack_modes[2] = {
13558 { 2, alc861_threestack_ch2_init },
13559 { 6, alc861_threestack_ch6_init },
13560};
Takashi Iwai22309c32006-08-09 16:57:28 +020013561/* Set mic1 as input and unmute the mixer */
13562static struct hda_verb alc861_uniwill_m31_ch2_init[] = {
13563 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
13564 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x01 << 8)) }, /*mic*/
13565 { } /* end */
13566};
13567/* Set mic1 as output and mute mixer */
13568static struct hda_verb alc861_uniwill_m31_ch4_init[] = {
13569 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
13570 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8)) }, /*mic*/
13571 { } /* end */
13572};
13573
13574static struct hda_channel_mode alc861_uniwill_m31_modes[2] = {
13575 { 2, alc861_uniwill_m31_ch2_init },
13576 { 4, alc861_uniwill_m31_ch4_init },
13577};
Kailang Yangdf694da2005-12-05 19:42:22 +010013578
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020013579/* Set mic1 and line-in as input and unmute the mixer */
13580static struct hda_verb alc861_asus_ch2_init[] = {
13581 /* set pin widget 1Ah (line in) for input */
13582 { 0x0c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013583 /* set pin widget 18h (mic1/2) for input, for mic also enable
13584 * the vref
13585 */
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020013586 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
13587
13588 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0xb00c },
13589#if 0
13590 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x01 << 8)) }, /*mic*/
13591 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7000 | (0x02 << 8)) }, /*line-in*/
13592#endif
13593 { } /* end */
13594};
13595/* Set mic1 nad line-in as output and mute mixer */
13596static struct hda_verb alc861_asus_ch6_init[] = {
13597 /* set pin widget 1Ah (line in) for output (Back Surround)*/
13598 { 0x0c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
13599 /* { 0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE }, */
13600 /* set pin widget 18h (mic1) for output (CLFE)*/
13601 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
13602 /* { 0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE }, */
13603 { 0x0c, AC_VERB_SET_CONNECT_SEL, 0x00 },
13604 { 0x0d, AC_VERB_SET_CONNECT_SEL, 0x00 },
13605
13606 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0xb080 },
13607#if 0
13608 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x01 << 8)) }, /*mic*/
13609 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, (0x7080 | (0x02 << 8)) }, /*line in*/
13610#endif
13611 { } /* end */
13612};
13613
13614static struct hda_channel_mode alc861_asus_modes[2] = {
13615 { 2, alc861_asus_ch2_init },
13616 { 6, alc861_asus_ch6_init },
13617};
13618
Kailang Yangdf694da2005-12-05 19:42:22 +010013619/* patch-ALC861 */
13620
13621static struct snd_kcontrol_new alc861_base_mixer[] = {
13622 /* output mixer control */
13623 HDA_CODEC_MUTE("Front Playback Switch", 0x03, 0x0, HDA_OUTPUT),
13624 HDA_CODEC_MUTE("Surround Playback Switch", 0x06, 0x0, HDA_OUTPUT),
13625 HDA_CODEC_MUTE_MONO("Center Playback Switch", 0x05, 1, 0x0, HDA_OUTPUT),
13626 HDA_CODEC_MUTE_MONO("LFE Playback Switch", 0x05, 2, 0x0, HDA_OUTPUT),
13627 HDA_CODEC_MUTE("Side Playback Switch", 0x04, 0x0, HDA_OUTPUT),
13628
13629 /*Input mixer control */
13630 /* HDA_CODEC_VOLUME("Input Playback Volume", 0x15, 0x0, HDA_OUTPUT),
13631 HDA_CODEC_MUTE("Input Playback Switch", 0x15, 0x0, HDA_OUTPUT), */
13632 HDA_CODEC_VOLUME("CD Playback Volume", 0x15, 0x0, HDA_INPUT),
13633 HDA_CODEC_MUTE("CD Playback Switch", 0x15, 0x0, HDA_INPUT),
13634 HDA_CODEC_VOLUME("Line Playback Volume", 0x15, 0x02, HDA_INPUT),
13635 HDA_CODEC_MUTE("Line Playback Switch", 0x15, 0x02, HDA_INPUT),
13636 HDA_CODEC_VOLUME("Mic Playback Volume", 0x15, 0x01, HDA_INPUT),
13637 HDA_CODEC_MUTE("Mic Playback Switch", 0x15, 0x01, HDA_INPUT),
13638 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x10, 0x01, HDA_OUTPUT),
13639 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1a, 0x03, HDA_INPUT),
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013640
Kailang Yangdf694da2005-12-05 19:42:22 +010013641 { } /* end */
13642};
13643
13644static struct snd_kcontrol_new alc861_3ST_mixer[] = {
13645 /* output mixer control */
13646 HDA_CODEC_MUTE("Front Playback Switch", 0x03, 0x0, HDA_OUTPUT),
13647 HDA_CODEC_MUTE("Surround Playback Switch", 0x06, 0x0, HDA_OUTPUT),
13648 HDA_CODEC_MUTE_MONO("Center Playback Switch", 0x05, 1, 0x0, HDA_OUTPUT),
13649 HDA_CODEC_MUTE_MONO("LFE Playback Switch", 0x05, 2, 0x0, HDA_OUTPUT),
13650 /*HDA_CODEC_MUTE("Side Playback Switch", 0x04, 0x0, HDA_OUTPUT), */
13651
13652 /* Input mixer control */
13653 /* HDA_CODEC_VOLUME("Input Playback Volume", 0x15, 0x0, HDA_OUTPUT),
13654 HDA_CODEC_MUTE("Input Playback Switch", 0x15, 0x0, HDA_OUTPUT), */
13655 HDA_CODEC_VOLUME("CD Playback Volume", 0x15, 0x0, HDA_INPUT),
13656 HDA_CODEC_MUTE("CD Playback Switch", 0x15, 0x0, HDA_INPUT),
13657 HDA_CODEC_VOLUME("Line Playback Volume", 0x15, 0x02, HDA_INPUT),
13658 HDA_CODEC_MUTE("Line Playback Switch", 0x15, 0x02, HDA_INPUT),
13659 HDA_CODEC_VOLUME("Mic Playback Volume", 0x15, 0x01, HDA_INPUT),
13660 HDA_CODEC_MUTE("Mic Playback Switch", 0x15, 0x01, HDA_INPUT),
13661 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x10, 0x01, HDA_OUTPUT),
13662 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1a, 0x03, HDA_INPUT),
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013663
Kailang Yangdf694da2005-12-05 19:42:22 +010013664 {
13665 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
13666 .name = "Channel Mode",
13667 .info = alc_ch_mode_info,
13668 .get = alc_ch_mode_get,
13669 .put = alc_ch_mode_put,
13670 .private_value = ARRAY_SIZE(alc861_threestack_modes),
13671 },
13672 { } /* end */
Tobin Davisa53d1ae2006-10-17 12:00:28 +020013673};
13674
Takashi Iwaid1d985f2006-11-23 19:27:12 +010013675static struct snd_kcontrol_new alc861_toshiba_mixer[] = {
Tobin Davisa53d1ae2006-10-17 12:00:28 +020013676 /* output mixer control */
13677 HDA_CODEC_MUTE("Master Playback Switch", 0x03, 0x0, HDA_OUTPUT),
13678 HDA_CODEC_VOLUME("Mic Playback Volume", 0x15, 0x01, HDA_INPUT),
13679 HDA_CODEC_MUTE("Mic Playback Switch", 0x15, 0x01, HDA_INPUT),
Kailang Yangea1fb292008-08-26 12:58:38 +020013680
Tobin Davisa53d1ae2006-10-17 12:00:28 +020013681 { } /* end */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013682};
Tobin Davisa53d1ae2006-10-17 12:00:28 +020013683
Takashi Iwai22309c32006-08-09 16:57:28 +020013684static struct snd_kcontrol_new alc861_uniwill_m31_mixer[] = {
13685 /* output mixer control */
13686 HDA_CODEC_MUTE("Front Playback Switch", 0x03, 0x0, HDA_OUTPUT),
13687 HDA_CODEC_MUTE("Surround Playback Switch", 0x06, 0x0, HDA_OUTPUT),
13688 HDA_CODEC_MUTE_MONO("Center Playback Switch", 0x05, 1, 0x0, HDA_OUTPUT),
13689 HDA_CODEC_MUTE_MONO("LFE Playback Switch", 0x05, 2, 0x0, HDA_OUTPUT),
13690 /*HDA_CODEC_MUTE("Side Playback Switch", 0x04, 0x0, HDA_OUTPUT), */
13691
13692 /* Input mixer control */
13693 /* HDA_CODEC_VOLUME("Input Playback Volume", 0x15, 0x0, HDA_OUTPUT),
13694 HDA_CODEC_MUTE("Input Playback Switch", 0x15, 0x0, HDA_OUTPUT), */
13695 HDA_CODEC_VOLUME("CD Playback Volume", 0x15, 0x0, HDA_INPUT),
13696 HDA_CODEC_MUTE("CD Playback Switch", 0x15, 0x0, HDA_INPUT),
13697 HDA_CODEC_VOLUME("Line Playback Volume", 0x15, 0x02, HDA_INPUT),
13698 HDA_CODEC_MUTE("Line Playback Switch", 0x15, 0x02, HDA_INPUT),
13699 HDA_CODEC_VOLUME("Mic Playback Volume", 0x15, 0x01, HDA_INPUT),
13700 HDA_CODEC_MUTE("Mic Playback Switch", 0x15, 0x01, HDA_INPUT),
13701 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x10, 0x01, HDA_OUTPUT),
13702 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1a, 0x03, HDA_INPUT),
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013703
Takashi Iwai22309c32006-08-09 16:57:28 +020013704 {
13705 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
13706 .name = "Channel Mode",
13707 .info = alc_ch_mode_info,
13708 .get = alc_ch_mode_get,
13709 .put = alc_ch_mode_put,
13710 .private_value = ARRAY_SIZE(alc861_uniwill_m31_modes),
13711 },
13712 { } /* end */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013713};
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020013714
13715static struct snd_kcontrol_new alc861_asus_mixer[] = {
13716 /* output mixer control */
13717 HDA_CODEC_MUTE("Front Playback Switch", 0x03, 0x0, HDA_OUTPUT),
13718 HDA_CODEC_MUTE("Surround Playback Switch", 0x06, 0x0, HDA_OUTPUT),
13719 HDA_CODEC_MUTE_MONO("Center Playback Switch", 0x05, 1, 0x0, HDA_OUTPUT),
13720 HDA_CODEC_MUTE_MONO("LFE Playback Switch", 0x05, 2, 0x0, HDA_OUTPUT),
13721 HDA_CODEC_MUTE("Side Playback Switch", 0x04, 0x0, HDA_OUTPUT),
13722
13723 /* Input mixer control */
13724 HDA_CODEC_VOLUME("Input Playback Volume", 0x15, 0x0, HDA_OUTPUT),
13725 HDA_CODEC_MUTE("Input Playback Switch", 0x15, 0x0, HDA_OUTPUT),
13726 HDA_CODEC_VOLUME("CD Playback Volume", 0x15, 0x0, HDA_INPUT),
13727 HDA_CODEC_MUTE("CD Playback Switch", 0x15, 0x0, HDA_INPUT),
13728 HDA_CODEC_VOLUME("Line Playback Volume", 0x15, 0x02, HDA_INPUT),
13729 HDA_CODEC_MUTE("Line Playback Switch", 0x15, 0x02, HDA_INPUT),
13730 HDA_CODEC_VOLUME("Mic Playback Volume", 0x15, 0x01, HDA_INPUT),
13731 HDA_CODEC_MUTE("Mic Playback Switch", 0x15, 0x01, HDA_INPUT),
13732 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x10, 0x01, HDA_OUTPUT),
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013733 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1a, 0x03, HDA_OUTPUT),
13734
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020013735 {
13736 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
13737 .name = "Channel Mode",
13738 .info = alc_ch_mode_info,
13739 .get = alc_ch_mode_get,
13740 .put = alc_ch_mode_put,
13741 .private_value = ARRAY_SIZE(alc861_asus_modes),
13742 },
13743 { }
Takashi Iwai56bb0ca2006-11-22 11:52:52 +010013744};
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020013745
Takashi Iwai56bb0ca2006-11-22 11:52:52 +010013746/* additional mixer */
Takashi Iwaid1d985f2006-11-23 19:27:12 +010013747static struct snd_kcontrol_new alc861_asus_laptop_mixer[] = {
Takashi Iwai56bb0ca2006-11-22 11:52:52 +010013748 HDA_CODEC_VOLUME("CD Playback Volume", 0x15, 0x0, HDA_INPUT),
13749 HDA_CODEC_MUTE("CD Playback Switch", 0x15, 0x0, HDA_INPUT),
Takashi Iwai56bb0ca2006-11-22 11:52:52 +010013750 { }
13751};
13752
Kailang Yangdf694da2005-12-05 19:42:22 +010013753/*
13754 * generic initialization of ADC, input mixers and output mixers
13755 */
13756static struct hda_verb alc861_base_init_verbs[] = {
13757 /*
13758 * Unmute ADC0 and set the default input to mic-in
13759 */
13760 /* port-A for surround (rear panel) */
13761 { 0x0e, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
13762 { 0x0e, AC_VERB_SET_CONNECT_SEL, 0x00 },
13763 /* port-B for mic-in (rear panel) with vref */
13764 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
13765 /* port-C for line-in (rear panel) */
13766 { 0x0c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
13767 /* port-D for Front */
13768 { 0x0b, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
13769 { 0x0b, AC_VERB_SET_CONNECT_SEL, 0x00 },
13770 /* port-E for HP out (front panel) */
13771 { 0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc0 },
13772 /* route front PCM to HP */
Takashi Iwai9dece1d2006-11-16 17:12:49 +010013773 { 0x0f, AC_VERB_SET_CONNECT_SEL, 0x00 },
Kailang Yangdf694da2005-12-05 19:42:22 +010013774 /* port-F for mic-in (front panel) with vref */
13775 { 0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
13776 /* port-G for CLFE (rear panel) */
13777 { 0x1f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
13778 { 0x1f, AC_VERB_SET_CONNECT_SEL, 0x00 },
13779 /* port-H for side (rear panel) */
13780 { 0x20, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
13781 { 0x20, AC_VERB_SET_CONNECT_SEL, 0x00 },
13782 /* CD-in */
13783 { 0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
13784 /* route front mic to ADC1*/
13785 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
13786 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Kailang Yangea1fb292008-08-26 12:58:38 +020013787
Kailang Yangdf694da2005-12-05 19:42:22 +010013788 /* Unmute DAC0~3 & spdif out*/
13789 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13790 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13791 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13792 {0x06, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13793 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Kailang Yangea1fb292008-08-26 12:58:38 +020013794
Kailang Yangdf694da2005-12-05 19:42:22 +010013795 /* Unmute Mixer 14 (mic) 1c (Line in)*/
13796 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13797 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13798 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13799 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Kailang Yangea1fb292008-08-26 12:58:38 +020013800
Kailang Yangdf694da2005-12-05 19:42:22 +010013801 /* Unmute Stereo Mixer 15 */
13802 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13803 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13804 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013805 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0xb00c}, /* Output 0~12 step */
Kailang Yangdf694da2005-12-05 19:42:22 +010013806
13807 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13808 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13809 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13810 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13811 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13812 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13813 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13814 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013815 /* hp used DAC 3 (Front) */
13816 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(3)},
Kailang Yangdf694da2005-12-05 19:42:22 +010013817 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
13818
13819 { }
13820};
13821
13822static struct hda_verb alc861_threestack_init_verbs[] = {
13823 /*
13824 * Unmute ADC0 and set the default input to mic-in
13825 */
13826 /* port-A for surround (rear panel) */
13827 { 0x0e, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
13828 /* port-B for mic-in (rear panel) with vref */
13829 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
13830 /* port-C for line-in (rear panel) */
13831 { 0x0c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
13832 /* port-D for Front */
13833 { 0x0b, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
13834 { 0x0b, AC_VERB_SET_CONNECT_SEL, 0x00 },
13835 /* port-E for HP out (front panel) */
13836 { 0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc0 },
13837 /* route front PCM to HP */
Takashi Iwai9dece1d2006-11-16 17:12:49 +010013838 { 0x0f, AC_VERB_SET_CONNECT_SEL, 0x00 },
Kailang Yangdf694da2005-12-05 19:42:22 +010013839 /* port-F for mic-in (front panel) with vref */
13840 { 0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
13841 /* port-G for CLFE (rear panel) */
13842 { 0x1f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
13843 /* port-H for side (rear panel) */
13844 { 0x20, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
13845 /* CD-in */
13846 { 0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
13847 /* route front mic to ADC1*/
13848 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
13849 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13850 /* Unmute DAC0~3 & spdif out*/
13851 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13852 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13853 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13854 {0x06, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13855 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Kailang Yangea1fb292008-08-26 12:58:38 +020013856
Kailang Yangdf694da2005-12-05 19:42:22 +010013857 /* Unmute Mixer 14 (mic) 1c (Line in)*/
13858 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13859 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13860 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13861 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Kailang Yangea1fb292008-08-26 12:58:38 +020013862
Kailang Yangdf694da2005-12-05 19:42:22 +010013863 /* Unmute Stereo Mixer 15 */
13864 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13865 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13866 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013867 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0xb00c}, /* Output 0~12 step */
Kailang Yangdf694da2005-12-05 19:42:22 +010013868
13869 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13870 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13871 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13872 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13873 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13874 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13875 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13876 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013877 /* hp used DAC 3 (Front) */
13878 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(3)},
Kailang Yangdf694da2005-12-05 19:42:22 +010013879 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
13880 { }
13881};
Takashi Iwai22309c32006-08-09 16:57:28 +020013882
13883static struct hda_verb alc861_uniwill_m31_init_verbs[] = {
13884 /*
13885 * Unmute ADC0 and set the default input to mic-in
13886 */
13887 /* port-A for surround (rear panel) */
13888 { 0x0e, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
13889 /* port-B for mic-in (rear panel) with vref */
13890 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
13891 /* port-C for line-in (rear panel) */
13892 { 0x0c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
13893 /* port-D for Front */
13894 { 0x0b, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
13895 { 0x0b, AC_VERB_SET_CONNECT_SEL, 0x00 },
13896 /* port-E for HP out (front panel) */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013897 /* this has to be set to VREF80 */
13898 { 0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
Takashi Iwai22309c32006-08-09 16:57:28 +020013899 /* route front PCM to HP */
Takashi Iwai9dece1d2006-11-16 17:12:49 +010013900 { 0x0f, AC_VERB_SET_CONNECT_SEL, 0x00 },
Takashi Iwai22309c32006-08-09 16:57:28 +020013901 /* port-F for mic-in (front panel) with vref */
13902 { 0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
13903 /* port-G for CLFE (rear panel) */
13904 { 0x1f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
13905 /* port-H for side (rear panel) */
13906 { 0x20, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
13907 /* CD-in */
13908 { 0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
13909 /* route front mic to ADC1*/
13910 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
13911 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13912 /* Unmute DAC0~3 & spdif out*/
13913 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13914 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13915 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13916 {0x06, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13917 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Kailang Yangea1fb292008-08-26 12:58:38 +020013918
Takashi Iwai22309c32006-08-09 16:57:28 +020013919 /* Unmute Mixer 14 (mic) 1c (Line in)*/
13920 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13921 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13922 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13923 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Kailang Yangea1fb292008-08-26 12:58:38 +020013924
Takashi Iwai22309c32006-08-09 16:57:28 +020013925 /* Unmute Stereo Mixer 15 */
13926 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13927 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13928 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013929 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0xb00c}, /* Output 0~12 step */
Takashi Iwai22309c32006-08-09 16:57:28 +020013930
13931 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13932 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13933 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13934 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13935 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13936 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13937 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13938 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013939 /* hp used DAC 3 (Front) */
13940 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(3)},
Takashi Iwai22309c32006-08-09 16:57:28 +020013941 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
13942 { }
13943};
13944
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020013945static struct hda_verb alc861_asus_init_verbs[] = {
13946 /*
13947 * Unmute ADC0 and set the default input to mic-in
13948 */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013949 /* port-A for surround (rear panel)
13950 * according to codec#0 this is the HP jack
13951 */
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020013952 { 0x0e, AC_VERB_SET_PIN_WIDGET_CONTROL, 0xc0 }, /* was 0x00 */
13953 /* route front PCM to HP */
13954 { 0x0e, AC_VERB_SET_CONNECT_SEL, 0x01 },
13955 /* port-B for mic-in (rear panel) with vref */
13956 { 0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
13957 /* port-C for line-in (rear panel) */
13958 { 0x0c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
13959 /* port-D for Front */
13960 { 0x0b, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
13961 { 0x0b, AC_VERB_SET_CONNECT_SEL, 0x00 },
13962 /* port-E for HP out (front panel) */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013963 /* this has to be set to VREF80 */
13964 { 0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020013965 /* route front PCM to HP */
Takashi Iwai9dece1d2006-11-16 17:12:49 +010013966 { 0x0f, AC_VERB_SET_CONNECT_SEL, 0x00 },
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020013967 /* port-F for mic-in (front panel) with vref */
13968 { 0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24 },
13969 /* port-G for CLFE (rear panel) */
13970 { 0x1f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
13971 /* port-H for side (rear panel) */
13972 { 0x20, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x40 },
13973 /* CD-in */
13974 { 0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20 },
13975 /* route front mic to ADC1*/
13976 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00},
13977 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13978 /* Unmute DAC0~3 & spdif out*/
13979 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13980 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13981 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13982 {0x06, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13983 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
13984 /* Unmute Mixer 14 (mic) 1c (Line in)*/
13985 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13986 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13987 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13988 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Kailang Yangea1fb292008-08-26 12:58:38 +020013989
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020013990 /* Unmute Stereo Mixer 15 */
13991 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13992 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13993 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020013994 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0xb00c}, /* Output 0~12 step */
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020013995
13996 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13997 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
13998 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
13999 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
14000 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
14001 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
14002 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
14003 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014004 /* hp used DAC 3 (Front) */
14005 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(3)},
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020014006 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
14007 { }
14008};
14009
Takashi Iwai56bb0ca2006-11-22 11:52:52 +010014010/* additional init verbs for ASUS laptops */
14011static struct hda_verb alc861_asus_laptop_init_verbs[] = {
14012 { 0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x45 }, /* HP-out */
14013 { 0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2) }, /* mute line-in */
14014 { }
14015};
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020014016
Kailang Yangdf694da2005-12-05 19:42:22 +010014017/*
14018 * generic initialization of ADC, input mixers and output mixers
14019 */
14020static struct hda_verb alc861_auto_init_verbs[] = {
14021 /*
14022 * Unmute ADC0 and set the default input to mic-in
14023 */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014024 /* {0x08, AC_VERB_SET_CONNECT_SEL, 0x00}, */
Kailang Yangdf694da2005-12-05 19:42:22 +010014025 {0x08, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Kailang Yangea1fb292008-08-26 12:58:38 +020014026
Kailang Yangdf694da2005-12-05 19:42:22 +010014027 /* Unmute DAC0~3 & spdif out*/
14028 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
14029 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
14030 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
14031 {0x06, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
14032 {0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Kailang Yangea1fb292008-08-26 12:58:38 +020014033
Kailang Yangdf694da2005-12-05 19:42:22 +010014034 /* Unmute Mixer 14 (mic) 1c (Line in)*/
14035 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
14036 {0x014, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
14037 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
14038 {0x01c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Kailang Yangea1fb292008-08-26 12:58:38 +020014039
Kailang Yangdf694da2005-12-05 19:42:22 +010014040 /* Unmute Stereo Mixer 15 */
14041 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
14042 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
14043 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
14044 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, 0xb00c},
14045
Takashi Iwai1c209302009-07-22 15:17:45 +020014046 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
14047 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
14048 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
14049 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
14050 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
14051 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
14052 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
14053 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Kailang Yangdf694da2005-12-05 19:42:22 +010014054
14055 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
14056 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Takashi Iwai1c209302009-07-22 15:17:45 +020014057 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
14058 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
Kailang Yangdf694da2005-12-05 19:42:22 +010014059 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
14060 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Takashi Iwai1c209302009-07-22 15:17:45 +020014061 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
14062 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
Kailang Yangdf694da2005-12-05 19:42:22 +010014063
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014064 {0x08, AC_VERB_SET_CONNECT_SEL, 0x00}, /* set Mic 1 */
Kailang Yangdf694da2005-12-05 19:42:22 +010014065
14066 { }
14067};
14068
Tobin Davisa53d1ae2006-10-17 12:00:28 +020014069static struct hda_verb alc861_toshiba_init_verbs[] = {
14070 {0x0f, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014071
Tobin Davisa53d1ae2006-10-17 12:00:28 +020014072 { }
14073};
14074
14075/* toggle speaker-output according to the hp-jack state */
14076static void alc861_toshiba_automute(struct hda_codec *codec)
14077{
Wu Fengguang864f92b2009-11-18 12:38:02 +080014078 unsigned int present = snd_hda_jack_detect(codec, 0x0f);
Tobin Davisa53d1ae2006-10-17 12:00:28 +020014079
Takashi Iwai47fd8302007-08-10 17:11:07 +020014080 snd_hda_codec_amp_stereo(codec, 0x16, HDA_INPUT, 0,
14081 HDA_AMP_MUTE, present ? HDA_AMP_MUTE : 0);
14082 snd_hda_codec_amp_stereo(codec, 0x1a, HDA_INPUT, 3,
14083 HDA_AMP_MUTE, present ? 0 : HDA_AMP_MUTE);
Tobin Davisa53d1ae2006-10-17 12:00:28 +020014084}
14085
14086static void alc861_toshiba_unsol_event(struct hda_codec *codec,
14087 unsigned int res)
14088{
Tobin Davisa53d1ae2006-10-17 12:00:28 +020014089 if ((res >> 26) == ALC880_HP_EVENT)
14090 alc861_toshiba_automute(codec);
14091}
14092
Sasha Alexandrdef319f2009-06-16 16:00:15 -040014093/* pcm configuration: identical with ALC880 */
Kailang Yangdf694da2005-12-05 19:42:22 +010014094#define alc861_pcm_analog_playback alc880_pcm_analog_playback
14095#define alc861_pcm_analog_capture alc880_pcm_analog_capture
14096#define alc861_pcm_digital_playback alc880_pcm_digital_playback
14097#define alc861_pcm_digital_capture alc880_pcm_digital_capture
14098
14099
14100#define ALC861_DIGOUT_NID 0x07
14101
14102static struct hda_channel_mode alc861_8ch_modes[1] = {
14103 { 8, NULL }
14104};
14105
14106static hda_nid_t alc861_dac_nids[4] = {
14107 /* front, surround, clfe, side */
14108 0x03, 0x06, 0x05, 0x04
14109};
14110
Takashi Iwai9c7f8522006-06-28 15:08:22 +020014111static hda_nid_t alc660_dac_nids[3] = {
14112 /* front, clfe, surround */
14113 0x03, 0x05, 0x06
14114};
14115
Kailang Yangdf694da2005-12-05 19:42:22 +010014116static hda_nid_t alc861_adc_nids[1] = {
14117 /* ADC0-2 */
14118 0x08,
14119};
14120
14121static struct hda_input_mux alc861_capture_source = {
14122 .num_items = 5,
14123 .items = {
14124 { "Mic", 0x0 },
14125 { "Front Mic", 0x3 },
14126 { "Line", 0x1 },
14127 { "CD", 0x4 },
14128 { "Mixer", 0x5 },
14129 },
14130};
14131
Takashi Iwai1c209302009-07-22 15:17:45 +020014132static hda_nid_t alc861_look_for_dac(struct hda_codec *codec, hda_nid_t pin)
14133{
14134 struct alc_spec *spec = codec->spec;
14135 hda_nid_t mix, srcs[5];
14136 int i, j, num;
14137
14138 if (snd_hda_get_connections(codec, pin, &mix, 1) != 1)
14139 return 0;
14140 num = snd_hda_get_connections(codec, mix, srcs, ARRAY_SIZE(srcs));
14141 if (num < 0)
14142 return 0;
14143 for (i = 0; i < num; i++) {
14144 unsigned int type;
Takashi Iwaia22d5432009-07-27 12:54:26 +020014145 type = get_wcaps_type(get_wcaps(codec, srcs[i]));
Takashi Iwai1c209302009-07-22 15:17:45 +020014146 if (type != AC_WID_AUD_OUT)
14147 continue;
14148 for (j = 0; j < spec->multiout.num_dacs; j++)
14149 if (spec->multiout.dac_nids[j] == srcs[i])
14150 break;
14151 if (j >= spec->multiout.num_dacs)
14152 return srcs[i];
14153 }
14154 return 0;
14155}
14156
Kailang Yangdf694da2005-12-05 19:42:22 +010014157/* fill in the dac_nids table from the parsed pin configuration */
Takashi Iwai1c209302009-07-22 15:17:45 +020014158static int alc861_auto_fill_dac_nids(struct hda_codec *codec,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014159 const struct auto_pin_cfg *cfg)
Kailang Yangdf694da2005-12-05 19:42:22 +010014160{
Takashi Iwai1c209302009-07-22 15:17:45 +020014161 struct alc_spec *spec = codec->spec;
Kailang Yangdf694da2005-12-05 19:42:22 +010014162 int i;
Takashi Iwai1c209302009-07-22 15:17:45 +020014163 hda_nid_t nid, dac;
Kailang Yangdf694da2005-12-05 19:42:22 +010014164
14165 spec->multiout.dac_nids = spec->private_dac_nids;
14166 for (i = 0; i < cfg->line_outs; i++) {
14167 nid = cfg->line_out_pins[i];
Takashi Iwai1c209302009-07-22 15:17:45 +020014168 dac = alc861_look_for_dac(codec, nid);
14169 if (!dac)
14170 continue;
14171 spec->multiout.dac_nids[spec->multiout.num_dacs++] = dac;
Kailang Yangdf694da2005-12-05 19:42:22 +010014172 }
Kailang Yangdf694da2005-12-05 19:42:22 +010014173 return 0;
14174}
14175
Takashi Iwai1c209302009-07-22 15:17:45 +020014176static int alc861_create_out_sw(struct hda_codec *codec, const char *pfx,
14177 hda_nid_t nid, unsigned int chs)
Kailang Yangdf694da2005-12-05 19:42:22 +010014178{
Takashi Iwai0afe5f82009-10-02 09:20:00 +020014179 return add_pb_sw_ctrl(codec->spec, ALC_CTL_WIDGET_MUTE, pfx,
Takashi Iwai1c209302009-07-22 15:17:45 +020014180 HDA_COMPOSE_AMP_VAL(nid, chs, 0, HDA_OUTPUT));
14181}
14182
14183/* add playback controls from the parsed DAC table */
14184static int alc861_auto_create_multi_out_ctls(struct hda_codec *codec,
14185 const struct auto_pin_cfg *cfg)
14186{
14187 struct alc_spec *spec = codec->spec;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014188 static const char *chname[4] = {
14189 "Front", "Surround", NULL /*CLFE*/, "Side"
14190 };
Kailang Yangdf694da2005-12-05 19:42:22 +010014191 hda_nid_t nid;
Takashi Iwai1c209302009-07-22 15:17:45 +020014192 int i, err;
14193
14194 if (cfg->line_outs == 1) {
14195 const char *pfx = NULL;
14196 if (!cfg->hp_outs)
14197 pfx = "Master";
14198 else if (cfg->line_out_type == AUTO_PIN_SPEAKER_OUT)
14199 pfx = "Speaker";
14200 if (pfx) {
14201 nid = spec->multiout.dac_nids[0];
14202 return alc861_create_out_sw(codec, pfx, nid, 3);
14203 }
14204 }
Kailang Yangdf694da2005-12-05 19:42:22 +010014205
14206 for (i = 0; i < cfg->line_outs; i++) {
14207 nid = spec->multiout.dac_nids[i];
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014208 if (!nid)
Kailang Yangdf694da2005-12-05 19:42:22 +010014209 continue;
Takashi Iwai1c209302009-07-22 15:17:45 +020014210 if (i == 2) {
Kailang Yangdf694da2005-12-05 19:42:22 +010014211 /* Center/LFE */
Takashi Iwai1c209302009-07-22 15:17:45 +020014212 err = alc861_create_out_sw(codec, "Center", nid, 1);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014213 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +010014214 return err;
Takashi Iwai1c209302009-07-22 15:17:45 +020014215 err = alc861_create_out_sw(codec, "LFE", nid, 2);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014216 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +010014217 return err;
14218 } else {
Takashi Iwai1c209302009-07-22 15:17:45 +020014219 err = alc861_create_out_sw(codec, chname[i], nid, 3);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014220 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +010014221 return err;
14222 }
14223 }
14224 return 0;
14225}
14226
Takashi Iwai1c209302009-07-22 15:17:45 +020014227static int alc861_auto_create_hp_ctls(struct hda_codec *codec, hda_nid_t pin)
Kailang Yangdf694da2005-12-05 19:42:22 +010014228{
Takashi Iwai1c209302009-07-22 15:17:45 +020014229 struct alc_spec *spec = codec->spec;
Kailang Yangdf694da2005-12-05 19:42:22 +010014230 int err;
14231 hda_nid_t nid;
14232
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014233 if (!pin)
Kailang Yangdf694da2005-12-05 19:42:22 +010014234 return 0;
14235
14236 if ((pin >= 0x0b && pin <= 0x10) || pin == 0x1f || pin == 0x20) {
Takashi Iwai1c209302009-07-22 15:17:45 +020014237 nid = alc861_look_for_dac(codec, pin);
14238 if (nid) {
14239 err = alc861_create_out_sw(codec, "Headphone", nid, 3);
14240 if (err < 0)
14241 return err;
14242 spec->multiout.hp_nid = nid;
14243 }
Kailang Yangdf694da2005-12-05 19:42:22 +010014244 }
14245 return 0;
14246}
14247
14248/* create playback/capture controls for input pins */
Takashi Iwai05f5f472009-08-25 13:10:18 +020014249static int alc861_auto_create_input_ctls(struct hda_codec *codec,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014250 const struct auto_pin_cfg *cfg)
Kailang Yangdf694da2005-12-05 19:42:22 +010014251{
Takashi Iwai05f5f472009-08-25 13:10:18 +020014252 return alc_auto_create_input_ctls(codec, cfg, 0x15, 0x08, 0);
Kailang Yangdf694da2005-12-05 19:42:22 +010014253}
14254
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014255static void alc861_auto_set_output_and_unmute(struct hda_codec *codec,
14256 hda_nid_t nid,
Takashi Iwai1c209302009-07-22 15:17:45 +020014257 int pin_type, hda_nid_t dac)
Kailang Yangdf694da2005-12-05 19:42:22 +010014258{
Takashi Iwai1c209302009-07-22 15:17:45 +020014259 hda_nid_t mix, srcs[5];
14260 int i, num;
14261
Jacek Luczak564c5be2008-05-03 18:41:23 +020014262 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
14263 pin_type);
Takashi Iwai1c209302009-07-22 15:17:45 +020014264 snd_hda_codec_write(codec, dac, 0, AC_VERB_SET_AMP_GAIN_MUTE,
Jacek Luczak564c5be2008-05-03 18:41:23 +020014265 AMP_OUT_UNMUTE);
Takashi Iwai1c209302009-07-22 15:17:45 +020014266 if (snd_hda_get_connections(codec, nid, &mix, 1) != 1)
14267 return;
14268 num = snd_hda_get_connections(codec, mix, srcs, ARRAY_SIZE(srcs));
14269 if (num < 0)
14270 return;
14271 for (i = 0; i < num; i++) {
14272 unsigned int mute;
14273 if (srcs[i] == dac || srcs[i] == 0x15)
14274 mute = AMP_IN_UNMUTE(i);
14275 else
14276 mute = AMP_IN_MUTE(i);
14277 snd_hda_codec_write(codec, mix, 0, AC_VERB_SET_AMP_GAIN_MUTE,
14278 mute);
14279 }
Kailang Yangdf694da2005-12-05 19:42:22 +010014280}
14281
14282static void alc861_auto_init_multi_out(struct hda_codec *codec)
14283{
14284 struct alc_spec *spec = codec->spec;
14285 int i;
14286
14287 for (i = 0; i < spec->autocfg.line_outs; i++) {
14288 hda_nid_t nid = spec->autocfg.line_out_pins[i];
Takashi Iwaibaba8ee2007-04-23 17:17:48 +020014289 int pin_type = get_pin_type(spec->autocfg.line_out_type);
Kailang Yangdf694da2005-12-05 19:42:22 +010014290 if (nid)
Takashi Iwaibaba8ee2007-04-23 17:17:48 +020014291 alc861_auto_set_output_and_unmute(codec, nid, pin_type,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014292 spec->multiout.dac_nids[i]);
Kailang Yangdf694da2005-12-05 19:42:22 +010014293 }
14294}
14295
14296static void alc861_auto_init_hp_out(struct hda_codec *codec)
14297{
14298 struct alc_spec *spec = codec->spec;
Kailang Yangdf694da2005-12-05 19:42:22 +010014299
Takashi Iwai15870f02009-10-05 08:25:13 +020014300 if (spec->autocfg.hp_outs)
14301 alc861_auto_set_output_and_unmute(codec,
14302 spec->autocfg.hp_pins[0],
14303 PIN_HP,
Takashi Iwai1c209302009-07-22 15:17:45 +020014304 spec->multiout.hp_nid);
Takashi Iwai15870f02009-10-05 08:25:13 +020014305 if (spec->autocfg.speaker_outs)
14306 alc861_auto_set_output_and_unmute(codec,
14307 spec->autocfg.speaker_pins[0],
14308 PIN_OUT,
Takashi Iwai1c209302009-07-22 15:17:45 +020014309 spec->multiout.dac_nids[0]);
Kailang Yangdf694da2005-12-05 19:42:22 +010014310}
14311
14312static void alc861_auto_init_analog_input(struct hda_codec *codec)
14313{
14314 struct alc_spec *spec = codec->spec;
14315 int i;
14316
14317 for (i = 0; i < AUTO_PIN_LAST; i++) {
14318 hda_nid_t nid = spec->autocfg.input_pins[i];
Takashi Iwai23f0c042009-02-26 13:03:58 +010014319 if (nid >= 0x0c && nid <= 0x11)
14320 alc_set_input_pin(codec, nid, i);
Kailang Yangdf694da2005-12-05 19:42:22 +010014321 }
14322}
14323
14324/* parse the BIOS configuration and set up the alc_spec */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014325/* return 1 if successful, 0 if the proper config is not found,
14326 * or a negative error code
14327 */
Kailang Yangdf694da2005-12-05 19:42:22 +010014328static int alc861_parse_auto_config(struct hda_codec *codec)
14329{
14330 struct alc_spec *spec = codec->spec;
14331 int err;
14332 static hda_nid_t alc861_ignore[] = { 0x1d, 0 };
14333
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014334 err = snd_hda_parse_pin_def_config(codec, &spec->autocfg,
14335 alc861_ignore);
14336 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +010014337 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014338 if (!spec->autocfg.line_outs)
Kailang Yangdf694da2005-12-05 19:42:22 +010014339 return 0; /* can't find valid BIOS pin config */
14340
Takashi Iwai1c209302009-07-22 15:17:45 +020014341 err = alc861_auto_fill_dac_nids(codec, &spec->autocfg);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014342 if (err < 0)
14343 return err;
Takashi Iwai1c209302009-07-22 15:17:45 +020014344 err = alc861_auto_create_multi_out_ctls(codec, &spec->autocfg);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014345 if (err < 0)
14346 return err;
Takashi Iwai1c209302009-07-22 15:17:45 +020014347 err = alc861_auto_create_hp_ctls(codec, spec->autocfg.hp_pins[0]);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014348 if (err < 0)
14349 return err;
Takashi Iwai05f5f472009-08-25 13:10:18 +020014350 err = alc861_auto_create_input_ctls(codec, &spec->autocfg);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014351 if (err < 0)
Kailang Yangdf694da2005-12-05 19:42:22 +010014352 return err;
14353
14354 spec->multiout.max_channels = spec->multiout.num_dacs * 2;
14355
Takashi Iwai0852d7a2009-02-11 11:35:15 +010014356 if (spec->autocfg.dig_outs)
Kailang Yangdf694da2005-12-05 19:42:22 +010014357 spec->multiout.dig_out_nid = ALC861_DIGOUT_NID;
14358
Takashi Iwai603c4012008-07-30 15:01:44 +020014359 if (spec->kctls.list)
Takashi Iwaid88897e2008-10-31 15:01:37 +010014360 add_mixer(spec, spec->kctls.list);
Kailang Yangdf694da2005-12-05 19:42:22 +010014361
Takashi Iwaid88897e2008-10-31 15:01:37 +010014362 add_verb(spec, alc861_auto_init_verbs);
Kailang Yangdf694da2005-12-05 19:42:22 +010014363
Jonathan Woithea1e8d2d2006-03-28 12:47:09 +020014364 spec->num_mux_defs = 1;
Herton Ronaldo Krzesinski61b9b9b2009-01-28 09:16:33 -020014365 spec->input_mux = &spec->private_imux[0];
Kailang Yangdf694da2005-12-05 19:42:22 +010014366
14367 spec->adc_nids = alc861_adc_nids;
14368 spec->num_adc_nids = ARRAY_SIZE(alc861_adc_nids);
Takashi Iwaib59bdf32009-08-11 09:47:30 +020014369 set_capture_mixer(codec);
Kailang Yangdf694da2005-12-05 19:42:22 +010014370
Takashi Iwai4a79ba32009-04-22 16:31:35 +020014371 alc_ssid_check(codec, 0x0e, 0x0f, 0x0b);
14372
Kailang Yangdf694da2005-12-05 19:42:22 +010014373 return 1;
14374}
14375
Takashi Iwaiae6b8132006-03-03 16:47:17 +010014376/* additional initialization for auto-configuration model */
14377static void alc861_auto_init(struct hda_codec *codec)
Kailang Yangdf694da2005-12-05 19:42:22 +010014378{
Takashi Iwaif6c7e542008-02-12 18:32:23 +010014379 struct alc_spec *spec = codec->spec;
Kailang Yangdf694da2005-12-05 19:42:22 +010014380 alc861_auto_init_multi_out(codec);
14381 alc861_auto_init_hp_out(codec);
14382 alc861_auto_init_analog_input(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +010014383 if (spec->unsol_event)
Kailang Yang7fb0d782008-10-15 11:12:35 +020014384 alc_inithook(codec);
Kailang Yangdf694da2005-12-05 19:42:22 +010014385}
14386
Takashi Iwaicb53c622007-08-10 17:21:45 +020014387#ifdef CONFIG_SND_HDA_POWER_SAVE
14388static struct hda_amp_list alc861_loopbacks[] = {
14389 { 0x15, HDA_INPUT, 0 },
14390 { 0x15, HDA_INPUT, 1 },
14391 { 0x15, HDA_INPUT, 2 },
14392 { 0x15, HDA_INPUT, 3 },
14393 { } /* end */
14394};
14395#endif
14396
Kailang Yangdf694da2005-12-05 19:42:22 +010014397
14398/*
14399 * configuration and preset
14400 */
Takashi Iwaif5fcc132006-11-24 17:07:44 +010014401static const char *alc861_models[ALC861_MODEL_LAST] = {
14402 [ALC861_3ST] = "3stack",
14403 [ALC660_3ST] = "3stack-660",
14404 [ALC861_3ST_DIG] = "3stack-dig",
14405 [ALC861_6ST_DIG] = "6stack-dig",
14406 [ALC861_UNIWILL_M31] = "uniwill-m31",
14407 [ALC861_TOSHIBA] = "toshiba",
14408 [ALC861_ASUS] = "asus",
14409 [ALC861_ASUS_LAPTOP] = "asus-laptop",
14410 [ALC861_AUTO] = "auto",
14411};
14412
14413static struct snd_pci_quirk alc861_cfg_tbl[] = {
Takashi Iwai687a47b2007-01-10 11:25:58 +010014414 SND_PCI_QUIRK(0x1043, 0x1205, "ASUS W7J", ALC861_3ST),
Takashi Iwaif5fcc132006-11-24 17:07:44 +010014415 SND_PCI_QUIRK(0x1043, 0x1335, "ASUS F2/3", ALC861_ASUS_LAPTOP),
14416 SND_PCI_QUIRK(0x1043, 0x1338, "ASUS F2/3", ALC861_ASUS_LAPTOP),
14417 SND_PCI_QUIRK(0x1043, 0x1393, "ASUS", ALC861_ASUS),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010014418 SND_PCI_QUIRK(0x1043, 0x13d7, "ASUS A9rp", ALC861_ASUS_LAPTOP),
Kailang Yang83c34212007-07-05 11:43:05 +020014419 SND_PCI_QUIRK(0x1043, 0x81cb, "ASUS P1-AH2", ALC861_3ST_DIG),
Tobin Davisad5e7732007-01-08 10:57:32 +010014420 SND_PCI_QUIRK(0x1179, 0xff00, "Toshiba", ALC861_TOSHIBA),
Takashi Iwai341d4eb2007-07-09 17:53:18 +020014421 /* FIXME: the entry below breaks Toshiba A100 (model=auto works!)
14422 * Any other models that need this preset?
14423 */
14424 /* SND_PCI_QUIRK(0x1179, 0xff10, "Toshiba", ALC861_TOSHIBA), */
Claudio Matsuokaef64adb2007-07-14 00:26:16 +020014425 SND_PCI_QUIRK(0x1462, 0x7254, "HP dx2200 (MSI MS-7254)", ALC861_3ST),
14426 SND_PCI_QUIRK(0x1462, 0x7297, "HP dx2250 (MSI MS-7297)", ALC861_3ST),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010014427 SND_PCI_QUIRK(0x1584, 0x2b01, "Uniwill X40AIx", ALC861_UNIWILL_M31),
14428 SND_PCI_QUIRK(0x1584, 0x9072, "Uniwill m31", ALC861_UNIWILL_M31),
14429 SND_PCI_QUIRK(0x1584, 0x9075, "Airis Praxis N1212", ALC861_ASUS_LAPTOP),
14430 /* FIXME: the below seems conflict */
14431 /* SND_PCI_QUIRK(0x1584, 0x9075, "Uniwill", ALC861_UNIWILL_M31), */
14432 SND_PCI_QUIRK(0x1849, 0x0660, "Asrock 939SLI32", ALC660_3ST),
14433 SND_PCI_QUIRK(0x8086, 0xd600, "Intel", ALC861_3ST),
Kailang Yangdf694da2005-12-05 19:42:22 +010014434 {}
14435};
14436
14437static struct alc_config_preset alc861_presets[] = {
14438 [ALC861_3ST] = {
14439 .mixers = { alc861_3ST_mixer },
14440 .init_verbs = { alc861_threestack_init_verbs },
14441 .num_dacs = ARRAY_SIZE(alc861_dac_nids),
14442 .dac_nids = alc861_dac_nids,
14443 .num_channel_mode = ARRAY_SIZE(alc861_threestack_modes),
14444 .channel_mode = alc861_threestack_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +020014445 .need_dac_fix = 1,
Kailang Yangdf694da2005-12-05 19:42:22 +010014446 .num_adc_nids = ARRAY_SIZE(alc861_adc_nids),
14447 .adc_nids = alc861_adc_nids,
14448 .input_mux = &alc861_capture_source,
14449 },
14450 [ALC861_3ST_DIG] = {
14451 .mixers = { alc861_base_mixer },
14452 .init_verbs = { alc861_threestack_init_verbs },
14453 .num_dacs = ARRAY_SIZE(alc861_dac_nids),
14454 .dac_nids = alc861_dac_nids,
14455 .dig_out_nid = ALC861_DIGOUT_NID,
14456 .num_channel_mode = ARRAY_SIZE(alc861_threestack_modes),
14457 .channel_mode = alc861_threestack_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +020014458 .need_dac_fix = 1,
Kailang Yangdf694da2005-12-05 19:42:22 +010014459 .num_adc_nids = ARRAY_SIZE(alc861_adc_nids),
14460 .adc_nids = alc861_adc_nids,
14461 .input_mux = &alc861_capture_source,
14462 },
14463 [ALC861_6ST_DIG] = {
14464 .mixers = { alc861_base_mixer },
14465 .init_verbs = { alc861_base_init_verbs },
14466 .num_dacs = ARRAY_SIZE(alc861_dac_nids),
14467 .dac_nids = alc861_dac_nids,
14468 .dig_out_nid = ALC861_DIGOUT_NID,
14469 .num_channel_mode = ARRAY_SIZE(alc861_8ch_modes),
14470 .channel_mode = alc861_8ch_modes,
14471 .num_adc_nids = ARRAY_SIZE(alc861_adc_nids),
14472 .adc_nids = alc861_adc_nids,
14473 .input_mux = &alc861_capture_source,
14474 },
Takashi Iwai9c7f8522006-06-28 15:08:22 +020014475 [ALC660_3ST] = {
14476 .mixers = { alc861_3ST_mixer },
14477 .init_verbs = { alc861_threestack_init_verbs },
14478 .num_dacs = ARRAY_SIZE(alc660_dac_nids),
14479 .dac_nids = alc660_dac_nids,
14480 .num_channel_mode = ARRAY_SIZE(alc861_threestack_modes),
14481 .channel_mode = alc861_threestack_modes,
Takashi Iwai4e195a72006-07-28 14:47:34 +020014482 .need_dac_fix = 1,
Takashi Iwai9c7f8522006-06-28 15:08:22 +020014483 .num_adc_nids = ARRAY_SIZE(alc861_adc_nids),
14484 .adc_nids = alc861_adc_nids,
14485 .input_mux = &alc861_capture_source,
14486 },
Takashi Iwai22309c32006-08-09 16:57:28 +020014487 [ALC861_UNIWILL_M31] = {
14488 .mixers = { alc861_uniwill_m31_mixer },
14489 .init_verbs = { alc861_uniwill_m31_init_verbs },
14490 .num_dacs = ARRAY_SIZE(alc861_dac_nids),
14491 .dac_nids = alc861_dac_nids,
14492 .dig_out_nid = ALC861_DIGOUT_NID,
14493 .num_channel_mode = ARRAY_SIZE(alc861_uniwill_m31_modes),
14494 .channel_mode = alc861_uniwill_m31_modes,
14495 .need_dac_fix = 1,
14496 .num_adc_nids = ARRAY_SIZE(alc861_adc_nids),
14497 .adc_nids = alc861_adc_nids,
14498 .input_mux = &alc861_capture_source,
14499 },
Tobin Davisa53d1ae2006-10-17 12:00:28 +020014500 [ALC861_TOSHIBA] = {
14501 .mixers = { alc861_toshiba_mixer },
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014502 .init_verbs = { alc861_base_init_verbs,
14503 alc861_toshiba_init_verbs },
Tobin Davisa53d1ae2006-10-17 12:00:28 +020014504 .num_dacs = ARRAY_SIZE(alc861_dac_nids),
14505 .dac_nids = alc861_dac_nids,
14506 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
14507 .channel_mode = alc883_3ST_2ch_modes,
14508 .num_adc_nids = ARRAY_SIZE(alc861_adc_nids),
14509 .adc_nids = alc861_adc_nids,
14510 .input_mux = &alc861_capture_source,
14511 .unsol_event = alc861_toshiba_unsol_event,
14512 .init_hook = alc861_toshiba_automute,
14513 },
Mariusz Domanski7cdbff92006-10-23 13:42:56 +020014514 [ALC861_ASUS] = {
14515 .mixers = { alc861_asus_mixer },
14516 .init_verbs = { alc861_asus_init_verbs },
14517 .num_dacs = ARRAY_SIZE(alc861_dac_nids),
14518 .dac_nids = alc861_dac_nids,
14519 .dig_out_nid = ALC861_DIGOUT_NID,
14520 .num_channel_mode = ARRAY_SIZE(alc861_asus_modes),
14521 .channel_mode = alc861_asus_modes,
14522 .need_dac_fix = 1,
14523 .hp_nid = 0x06,
14524 .num_adc_nids = ARRAY_SIZE(alc861_adc_nids),
14525 .adc_nids = alc861_adc_nids,
14526 .input_mux = &alc861_capture_source,
14527 },
Takashi Iwai56bb0ca2006-11-22 11:52:52 +010014528 [ALC861_ASUS_LAPTOP] = {
14529 .mixers = { alc861_toshiba_mixer, alc861_asus_laptop_mixer },
14530 .init_verbs = { alc861_asus_init_verbs,
14531 alc861_asus_laptop_init_verbs },
14532 .num_dacs = ARRAY_SIZE(alc861_dac_nids),
14533 .dac_nids = alc861_dac_nids,
14534 .dig_out_nid = ALC861_DIGOUT_NID,
14535 .num_channel_mode = ARRAY_SIZE(alc883_3ST_2ch_modes),
14536 .channel_mode = alc883_3ST_2ch_modes,
14537 .need_dac_fix = 1,
14538 .num_adc_nids = ARRAY_SIZE(alc861_adc_nids),
14539 .adc_nids = alc861_adc_nids,
14540 .input_mux = &alc861_capture_source,
14541 },
14542};
Kailang Yangdf694da2005-12-05 19:42:22 +010014543
14544
14545static int patch_alc861(struct hda_codec *codec)
14546{
14547 struct alc_spec *spec;
14548 int board_config;
14549 int err;
14550
Robert P. J. Daydc041e02006-12-19 14:44:15 +010014551 spec = kzalloc(sizeof(*spec), GFP_KERNEL);
Kailang Yangdf694da2005-12-05 19:42:22 +010014552 if (spec == NULL)
14553 return -ENOMEM;
14554
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014555 codec->spec = spec;
Kailang Yangdf694da2005-12-05 19:42:22 +010014556
Takashi Iwaif5fcc132006-11-24 17:07:44 +010014557 board_config = snd_hda_check_board_config(codec, ALC861_MODEL_LAST,
14558 alc861_models,
14559 alc861_cfg_tbl);
Takashi Iwai9c7f8522006-06-28 15:08:22 +020014560
Takashi Iwaif5fcc132006-11-24 17:07:44 +010014561 if (board_config < 0) {
Takashi Iwai9a11f1a2009-07-28 16:01:20 +020014562 printk(KERN_INFO "hda_codec: %s: BIOS auto-probing.\n",
14563 codec->chip_name);
Kailang Yangdf694da2005-12-05 19:42:22 +010014564 board_config = ALC861_AUTO;
14565 }
14566
14567 if (board_config == ALC861_AUTO) {
14568 /* automatic parse from the BIOS config */
14569 err = alc861_parse_auto_config(codec);
14570 if (err < 0) {
14571 alc_free(codec);
14572 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020014573 } else if (!err) {
Takashi Iwai9c7f8522006-06-28 15:08:22 +020014574 printk(KERN_INFO
14575 "hda_codec: Cannot set up configuration "
14576 "from BIOS. Using base mode...\n");
Kailang Yangdf694da2005-12-05 19:42:22 +010014577 board_config = ALC861_3ST_DIG;
14578 }
14579 }
14580
Kusanagi Kouichi680cd532009-02-05 00:00:58 +090014581 err = snd_hda_attach_beep_device(codec, 0x23);
14582 if (err < 0) {
14583 alc_free(codec);
14584 return err;
14585 }
14586
Kailang Yangdf694da2005-12-05 19:42:22 +010014587 if (board_config != ALC861_AUTO)
Takashi Iwaie9c364c2009-08-11 17:16:13 +020014588 setup_preset(codec, &alc861_presets[board_config]);
Kailang Yangdf694da2005-12-05 19:42:22 +010014589
Kailang Yangdf694da2005-12-05 19:42:22 +010014590 spec->stream_analog_playback = &alc861_pcm_analog_playback;
14591 spec->stream_analog_capture = &alc861_pcm_analog_capture;
14592
Kailang Yangdf694da2005-12-05 19:42:22 +010014593 spec->stream_digital_playback = &alc861_pcm_digital_playback;
14594 spec->stream_digital_capture = &alc861_pcm_digital_capture;
14595
Takashi Iwai45bdd1c2009-02-06 16:11:25 +010014596 set_beep_amp(spec, 0x23, 0, HDA_OUTPUT);
14597
Takashi Iwai2134ea42008-01-10 16:53:55 +010014598 spec->vmaster_nid = 0x03;
14599
Kailang Yangdf694da2005-12-05 19:42:22 +010014600 codec->patch_ops = alc_patch_ops;
14601 if (board_config == ALC861_AUTO)
Takashi Iwaiae6b8132006-03-03 16:47:17 +010014602 spec->init_hook = alc861_auto_init;
Takashi Iwaicb53c622007-08-10 17:21:45 +020014603#ifdef CONFIG_SND_HDA_POWER_SAVE
14604 if (!spec->loopback.amplist)
14605 spec->loopback.amplist = alc861_loopbacks;
14606#endif
Takashi Iwaidaead532008-11-28 12:55:36 +010014607 codec->proc_widget_hook = print_realtek_coef;
Kailang Yangea1fb292008-08-26 12:58:38 +020014608
Kailang Yangdf694da2005-12-05 19:42:22 +010014609 return 0;
14610}
14611
14612/*
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010014613 * ALC861-VD support
14614 *
14615 * Based on ALC882
14616 *
14617 * In addition, an independent DAC
14618 */
14619#define ALC861VD_DIGOUT_NID 0x06
14620
14621static hda_nid_t alc861vd_dac_nids[4] = {
14622 /* front, surr, clfe, side surr */
14623 0x02, 0x03, 0x04, 0x05
14624};
14625
14626/* dac_nids for ALC660vd are in a different order - according to
14627 * Realtek's driver.
Sasha Alexandrdef319f2009-06-16 16:00:15 -040014628 * This should probably result in a different mixer for 6stack models
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010014629 * of ALC660vd codecs, but for now there is only 3stack mixer
14630 * - and it is the same as in 861vd.
14631 * adc_nids in ALC660vd are (is) the same as in 861vd
14632 */
14633static hda_nid_t alc660vd_dac_nids[3] = {
14634 /* front, rear, clfe, rear_surr */
14635 0x02, 0x04, 0x03
14636};
14637
14638static hda_nid_t alc861vd_adc_nids[1] = {
14639 /* ADC0 */
14640 0x09,
14641};
14642
Takashi Iwaie1406342008-02-11 18:32:32 +010014643static hda_nid_t alc861vd_capsrc_nids[1] = { 0x22 };
14644
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010014645/* input MUX */
14646/* FIXME: should be a matrix-type input source selection */
14647static struct hda_input_mux alc861vd_capture_source = {
14648 .num_items = 4,
14649 .items = {
14650 { "Mic", 0x0 },
14651 { "Front Mic", 0x1 },
14652 { "Line", 0x2 },
14653 { "CD", 0x4 },
14654 },
14655};
14656
Kailang Yang272a5272007-05-14 11:00:38 +020014657static struct hda_input_mux alc861vd_dallas_capture_source = {
Tobin Davisb419f342008-03-07 11:57:51 +010014658 .num_items = 2,
Kailang Yang272a5272007-05-14 11:00:38 +020014659 .items = {
Tobin Davisb419f342008-03-07 11:57:51 +010014660 { "Ext Mic", 0x0 },
14661 { "Int Mic", 0x1 },
Kailang Yang272a5272007-05-14 11:00:38 +020014662 },
14663};
14664
Kailang Yangd1a991a2007-08-15 16:21:59 +020014665static struct hda_input_mux alc861vd_hp_capture_source = {
14666 .num_items = 2,
14667 .items = {
14668 { "Front Mic", 0x0 },
14669 { "ATAPI Mic", 0x1 },
14670 },
14671};
14672
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010014673/*
14674 * 2ch mode
14675 */
14676static struct hda_channel_mode alc861vd_3stack_2ch_modes[1] = {
14677 { 2, NULL }
14678};
14679
14680/*
14681 * 6ch mode
14682 */
14683static struct hda_verb alc861vd_6stack_ch6_init[] = {
14684 { 0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
14685 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
14686 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
14687 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
14688 { } /* end */
14689};
14690
14691/*
14692 * 8ch mode
14693 */
14694static struct hda_verb alc861vd_6stack_ch8_init[] = {
14695 { 0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
14696 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
14697 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
14698 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
14699 { } /* end */
14700};
14701
14702static struct hda_channel_mode alc861vd_6stack_modes[2] = {
14703 { 6, alc861vd_6stack_ch6_init },
14704 { 8, alc861vd_6stack_ch8_init },
14705};
14706
14707static struct snd_kcontrol_new alc861vd_chmode_mixer[] = {
14708 {
14709 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
14710 .name = "Channel Mode",
14711 .info = alc_ch_mode_info,
14712 .get = alc_ch_mode_get,
14713 .put = alc_ch_mode_put,
14714 },
14715 { } /* end */
14716};
14717
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010014718/* Pin assignment: Front=0x14, Rear=0x15, CLFE=0x16, Side=0x17
14719 * Mic=0x18, Front Mic=0x19, Line-In=0x1a, HP=0x1b
14720 */
14721static struct snd_kcontrol_new alc861vd_6st_mixer[] = {
14722 HDA_CODEC_VOLUME("Front Playback Volume", 0x02, 0x0, HDA_OUTPUT),
14723 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
14724
14725 HDA_CODEC_VOLUME("Surround Playback Volume", 0x03, 0x0, HDA_OUTPUT),
14726 HDA_BIND_MUTE("Surround Playback Switch", 0x0d, 2, HDA_INPUT),
14727
14728 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x04, 1, 0x0,
14729 HDA_OUTPUT),
14730 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x04, 2, 0x0,
14731 HDA_OUTPUT),
14732 HDA_BIND_MUTE_MONO("Center Playback Switch", 0x0e, 1, 2, HDA_INPUT),
14733 HDA_BIND_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 2, HDA_INPUT),
14734
14735 HDA_CODEC_VOLUME("Side Playback Volume", 0x05, 0x0, HDA_OUTPUT),
14736 HDA_BIND_MUTE("Side Playback Switch", 0x0f, 2, HDA_INPUT),
14737
14738 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
14739
14740 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
14741 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
14742 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
14743
14744 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
14745 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
14746 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
14747
14748 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
14749 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
14750
14751 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
14752 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
14753
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010014754 { } /* end */
14755};
14756
14757static struct snd_kcontrol_new alc861vd_3st_mixer[] = {
14758 HDA_CODEC_VOLUME("Front Playback Volume", 0x02, 0x0, HDA_OUTPUT),
14759 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
14760
14761 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
14762
14763 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
14764 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
14765 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
14766
14767 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
14768 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
14769 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
14770
14771 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
14772 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
14773
14774 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
14775 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
14776
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010014777 { } /* end */
14778};
14779
Kailang Yangbdd148a2007-05-08 15:19:08 +020014780static struct snd_kcontrol_new alc861vd_lenovo_mixer[] = {
14781 HDA_CODEC_VOLUME("Front Playback Volume", 0x02, 0x0, HDA_OUTPUT),
14782 /*HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),*/
14783 HDA_CODEC_MUTE("Front Playback Switch", 0x14, 0x0, HDA_OUTPUT),
14784
14785 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
14786
14787 HDA_CODEC_VOLUME("Mic Boost", 0x18, 0, HDA_INPUT),
14788 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
14789 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
14790
14791 HDA_CODEC_VOLUME("Front Mic Boost", 0x19, 0, HDA_INPUT),
14792 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
14793 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
14794
14795 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
14796 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
14797
14798 { } /* end */
14799};
14800
Tobin Davisb419f342008-03-07 11:57:51 +010014801/* Pin assignment: Speaker=0x14, HP = 0x15,
14802 * Ext Mic=0x18, Int Mic = 0x19, CD = 0x1c, PC Beep = 0x1d
Kailang Yang272a5272007-05-14 11:00:38 +020014803 */
14804static struct snd_kcontrol_new alc861vd_dallas_mixer[] = {
Tobin Davisb419f342008-03-07 11:57:51 +010014805 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x02, 0x0, HDA_OUTPUT),
14806 HDA_BIND_MUTE("Speaker Playback Switch", 0x0c, 2, HDA_INPUT),
Kailang Yang272a5272007-05-14 11:00:38 +020014807 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x03, 0x0, HDA_OUTPUT),
14808 HDA_BIND_MUTE("Headphone Playback Switch", 0x0d, 2, HDA_INPUT),
Tobin Davisb419f342008-03-07 11:57:51 +010014809 HDA_CODEC_VOLUME("Ext Mic Boost", 0x18, 0, HDA_INPUT),
14810 HDA_CODEC_VOLUME("Ext Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
14811 HDA_CODEC_MUTE("Ext Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
14812 HDA_CODEC_VOLUME("Int Mic Boost", 0x19, 0, HDA_INPUT),
14813 HDA_CODEC_VOLUME("Int Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
14814 HDA_CODEC_MUTE("Int Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Kailang Yang272a5272007-05-14 11:00:38 +020014815 { } /* end */
14816};
14817
Kailang Yangd1a991a2007-08-15 16:21:59 +020014818/* Pin assignment: Speaker=0x14, Line-out = 0x15,
14819 * Front Mic=0x18, ATAPI Mic = 0x19,
14820 */
14821static struct snd_kcontrol_new alc861vd_hp_mixer[] = {
14822 HDA_CODEC_VOLUME("Front Playback Volume", 0x02, 0x0, HDA_OUTPUT),
14823 HDA_BIND_MUTE("Front Playback Switch", 0x0c, 2, HDA_INPUT),
14824 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x03, 0x0, HDA_OUTPUT),
14825 HDA_BIND_MUTE("Headphone Playback Switch", 0x0d, 2, HDA_INPUT),
14826 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
14827 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
14828 HDA_CODEC_VOLUME("ATAPI Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
14829 HDA_CODEC_MUTE("ATAPI Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Kailang Yangea1fb292008-08-26 12:58:38 +020014830
Kailang Yangd1a991a2007-08-15 16:21:59 +020014831 { } /* end */
14832};
14833
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010014834/*
14835 * generic initialization of ADC, input mixers and output mixers
14836 */
14837static struct hda_verb alc861vd_volume_init_verbs[] = {
14838 /*
14839 * Unmute ADC0 and set the default input to mic-in
14840 */
14841 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
14842 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
14843
14844 /* Unmute input amps (CD, Line In, Mic 1 & Mic 2) of
14845 * the analog-loopback mixer widget
14846 */
14847 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +020014848 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
14849 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
14850 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
14851 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
14852 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010014853
14854 /* Capture mixer: unmute Mic, F-Mic, Line, CD inputs */
Kailang Yangbdd148a2007-05-08 15:19:08 +020014855 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
14856 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
14857 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2)},
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010014858 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(4)},
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010014859
14860 /*
14861 * Set up output mixers (0x02 - 0x05)
14862 */
14863 /* set vol=0 to output mixers */
14864 {0x02, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
14865 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
14866 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
14867 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
14868
14869 /* set up input amps for analog loopback */
14870 /* Amp Indices: DAC = 0, mixer = 1 */
14871 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
14872 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
14873 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
14874 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
14875 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
14876 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
14877 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
14878 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
14879
14880 { }
14881};
14882
14883/*
14884 * 3-stack pin configuration:
14885 * front = 0x14, mic/clfe = 0x18, HP = 0x19, line/surr = 0x1a, f-mic = 0x1b
14886 */
14887static struct hda_verb alc861vd_3stack_init_verbs[] = {
14888 /*
14889 * Set pin mode and muting
14890 */
14891 /* set front pin widgets 0x14 for output */
14892 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
14893 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
14894 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
14895
14896 /* Mic (rear) pin: input vref at 80% */
14897 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
14898 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
14899 /* Front Mic pin: input vref at 80% */
14900 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
14901 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
14902 /* Line In pin: input */
14903 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
14904 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
14905 /* Line-2 In: Headphone output (output 0 - 0x0c) */
14906 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
14907 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
14908 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
14909 /* CD pin widget for input */
14910 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
14911
14912 { }
14913};
14914
14915/*
14916 * 6-stack pin configuration:
14917 */
14918static struct hda_verb alc861vd_6stack_init_verbs[] = {
14919 /*
14920 * Set pin mode and muting
14921 */
14922 /* set front pin widgets 0x14 for output */
14923 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
14924 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
14925 {0x14, AC_VERB_SET_CONNECT_SEL, 0x00},
14926
14927 /* Rear Pin: output 1 (0x0d) */
14928 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
14929 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
14930 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01},
14931 /* CLFE Pin: output 2 (0x0e) */
14932 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
14933 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
14934 {0x16, AC_VERB_SET_CONNECT_SEL, 0x02},
14935 /* Side Pin: output 3 (0x0f) */
14936 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
14937 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
14938 {0x17, AC_VERB_SET_CONNECT_SEL, 0x03},
14939
14940 /* Mic (rear) pin: input vref at 80% */
14941 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
14942 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
14943 /* Front Mic pin: input vref at 80% */
14944 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
14945 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
14946 /* Line In pin: input */
14947 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
14948 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
14949 /* Line-2 In: Headphone output (output 0 - 0x0c) */
14950 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
14951 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
14952 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
14953 /* CD pin widget for input */
14954 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
14955
14956 { }
14957};
14958
Kailang Yangbdd148a2007-05-08 15:19:08 +020014959static struct hda_verb alc861vd_eapd_verbs[] = {
14960 {0x14, AC_VERB_SET_EAPD_BTLENABLE, 2},
14961 { }
14962};
14963
Kailang Yangf9423e72008-05-27 12:32:25 +020014964static struct hda_verb alc660vd_eapd_verbs[] = {
14965 {0x14, AC_VERB_SET_EAPD_BTLENABLE, 2},
14966 {0x15, AC_VERB_SET_EAPD_BTLENABLE, 2},
14967 { }
14968};
14969
Kailang Yangbdd148a2007-05-08 15:19:08 +020014970static struct hda_verb alc861vd_lenovo_unsol_verbs[] = {
14971 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
14972 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
14973 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(5)},
14974 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
Kailang Yangea1fb292008-08-26 12:58:38 +020014975 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
Kailang Yangbdd148a2007-05-08 15:19:08 +020014976 {}
14977};
14978
Kailang Yangbdd148a2007-05-08 15:19:08 +020014979static void alc861vd_lenovo_mic_automute(struct hda_codec *codec)
14980{
14981 unsigned int present;
14982 unsigned char bits;
14983
Wu Fengguang864f92b2009-11-18 12:38:02 +080014984 present = snd_hda_jack_detect(codec, 0x18);
Takashi Iwai47fd8302007-08-10 17:11:07 +020014985 bits = present ? HDA_AMP_MUTE : 0;
Wu Fengguang864f92b2009-11-18 12:38:02 +080014986
Takashi Iwai47fd8302007-08-10 17:11:07 +020014987 snd_hda_codec_amp_stereo(codec, 0x0b, HDA_INPUT, 1,
14988 HDA_AMP_MUTE, bits);
Kailang Yangbdd148a2007-05-08 15:19:08 +020014989}
14990
Takashi Iwai4f5d17062009-08-11 18:17:46 +020014991static void alc861vd_lenovo_setup(struct hda_codec *codec)
Kailang Yangbdd148a2007-05-08 15:19:08 +020014992{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020014993 struct alc_spec *spec = codec->spec;
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020014994 spec->autocfg.hp_pins[0] = 0x1b;
14995 spec->autocfg.speaker_pins[0] = 0x14;
Takashi Iwai4f5d17062009-08-11 18:17:46 +020014996}
14997
14998static void alc861vd_lenovo_init_hook(struct hda_codec *codec)
14999{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020015000 alc_automute_amp(codec);
Kailang Yangbdd148a2007-05-08 15:19:08 +020015001 alc861vd_lenovo_mic_automute(codec);
15002}
15003
15004static void alc861vd_lenovo_unsol_event(struct hda_codec *codec,
15005 unsigned int res)
15006{
15007 switch (res >> 26) {
Kailang Yangbdd148a2007-05-08 15:19:08 +020015008 case ALC880_MIC_EVENT:
15009 alc861vd_lenovo_mic_automute(codec);
15010 break;
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020015011 default:
15012 alc_automute_amp_unsol_event(codec, res);
15013 break;
Kailang Yangbdd148a2007-05-08 15:19:08 +020015014 }
15015}
15016
Kailang Yang272a5272007-05-14 11:00:38 +020015017static struct hda_verb alc861vd_dallas_verbs[] = {
15018 {0x02, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
15019 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
15020 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
15021 {0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
15022
15023 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
15024 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
15025 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
15026 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
15027 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
15028 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
15029 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
15030 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
Kailang Yangea1fb292008-08-26 12:58:38 +020015031
Kailang Yang272a5272007-05-14 11:00:38 +020015032 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
15033 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
15034 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
15035 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
15036 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
15037 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
15038 {0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
15039 {0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
15040
15041 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF50},
15042 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
15043 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF50},
15044 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
15045 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
15046 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
15047 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
15048 {0x1d, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
15049
15050 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
15051 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
15052 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
15053 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
15054
15055 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
Kailang Yangea1fb292008-08-26 12:58:38 +020015056 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
Kailang Yang272a5272007-05-14 11:00:38 +020015057 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
15058
15059 { } /* end */
15060};
15061
15062/* toggle speaker-output according to the hp-jack state */
Takashi Iwai4f5d17062009-08-11 18:17:46 +020015063static void alc861vd_dallas_setup(struct hda_codec *codec)
Kailang Yang272a5272007-05-14 11:00:38 +020015064{
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020015065 struct alc_spec *spec = codec->spec;
Kailang Yang272a5272007-05-14 11:00:38 +020015066
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020015067 spec->autocfg.hp_pins[0] = 0x15;
15068 spec->autocfg.speaker_pins[0] = 0x14;
Kailang Yang272a5272007-05-14 11:00:38 +020015069}
15070
Takashi Iwaicb53c622007-08-10 17:21:45 +020015071#ifdef CONFIG_SND_HDA_POWER_SAVE
15072#define alc861vd_loopbacks alc880_loopbacks
15073#endif
15074
Sasha Alexandrdef319f2009-06-16 16:00:15 -040015075/* pcm configuration: identical with ALC880 */
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015076#define alc861vd_pcm_analog_playback alc880_pcm_analog_playback
15077#define alc861vd_pcm_analog_capture alc880_pcm_analog_capture
15078#define alc861vd_pcm_digital_playback alc880_pcm_digital_playback
15079#define alc861vd_pcm_digital_capture alc880_pcm_digital_capture
15080
15081/*
15082 * configuration and preset
15083 */
15084static const char *alc861vd_models[ALC861VD_MODEL_LAST] = {
15085 [ALC660VD_3ST] = "3stack-660",
Takashi Iwai983f8ae2007-08-15 16:44:04 +020015086 [ALC660VD_3ST_DIG] = "3stack-660-digout",
Takashi Iwai13c94742008-11-05 08:06:08 +010015087 [ALC660VD_ASUS_V1S] = "asus-v1s",
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015088 [ALC861VD_3ST] = "3stack",
15089 [ALC861VD_3ST_DIG] = "3stack-digout",
15090 [ALC861VD_6ST_DIG] = "6stack-digout",
Kailang Yangbdd148a2007-05-08 15:19:08 +020015091 [ALC861VD_LENOVO] = "lenovo",
Kailang Yang272a5272007-05-14 11:00:38 +020015092 [ALC861VD_DALLAS] = "dallas",
Takashi Iwai983f8ae2007-08-15 16:44:04 +020015093 [ALC861VD_HP] = "hp",
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015094 [ALC861VD_AUTO] = "auto",
15095};
15096
15097static struct snd_pci_quirk alc861vd_cfg_tbl[] = {
Takashi Iwaiac3e3742007-12-17 17:14:18 +010015098 SND_PCI_QUIRK(0x1019, 0xa88d, "Realtek ALC660 demo", ALC660VD_3ST),
15099 SND_PCI_QUIRK(0x103c, 0x30bf, "HP TX1000", ALC861VD_HP),
Takashi Iwai07e038b2007-02-15 18:23:41 +010015100 SND_PCI_QUIRK(0x1043, 0x12e2, "Asus z35m", ALC660VD_3ST),
Takashi Iwaif8f25ba2009-10-06 08:31:29 +020015101 /*SND_PCI_QUIRK(0x1043, 0x1339, "Asus G1", ALC660VD_3ST),*/ /* auto */
Takashi Iwai13c94742008-11-05 08:06:08 +010015102 SND_PCI_QUIRK(0x1043, 0x1633, "Asus V1Sn", ALC660VD_ASUS_V1S),
Mike Crash6963f842007-06-25 12:12:51 +020015103 SND_PCI_QUIRK(0x1043, 0x81e7, "ASUS", ALC660VD_3ST_DIG),
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015104 SND_PCI_QUIRK(0x10de, 0x03f0, "Realtek ALC660 demo", ALC660VD_3ST),
Takashi Iwaiac3e3742007-12-17 17:14:18 +010015105 SND_PCI_QUIRK(0x1179, 0xff00, "Toshiba A135", ALC861VD_LENOVO),
Takashi Iwai38baf5a2007-08-16 17:52:43 +020015106 /*SND_PCI_QUIRK(0x1179, 0xff00, "DALLAS", ALC861VD_DALLAS),*/ /*lenovo*/
Takashi Iwaice577e82009-08-03 08:23:52 +020015107 SND_PCI_QUIRK(0x1179, 0xff01, "Toshiba A135", ALC861VD_LENOVO),
Takashi Iwai542d7c62007-08-16 18:57:30 +020015108 SND_PCI_QUIRK(0x1179, 0xff03, "Toshiba P205", ALC861VD_LENOVO),
Tobin Davisb419f342008-03-07 11:57:51 +010015109 SND_PCI_QUIRK(0x1179, 0xff31, "Toshiba L30-149", ALC861VD_DALLAS),
Takashi Iwai39c5d412007-08-15 16:24:17 +020015110 SND_PCI_QUIRK(0x1565, 0x820d, "Biostar NF61S SE", ALC861VD_6ST_DIG),
Takashi Iwaidea0a502009-02-09 17:14:52 +010015111 SND_PCI_QUIRK_VENDOR(0x17aa, "Lenovo", ALC861VD_LENOVO),
Tobin Davis625dc0b2007-07-30 21:42:10 +020015112 SND_PCI_QUIRK(0x1849, 0x0862, "ASRock K8NF6G-VSTA", ALC861VD_6ST_DIG),
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015113 {}
15114};
15115
15116static struct alc_config_preset alc861vd_presets[] = {
15117 [ALC660VD_3ST] = {
15118 .mixers = { alc861vd_3st_mixer },
15119 .init_verbs = { alc861vd_volume_init_verbs,
15120 alc861vd_3stack_init_verbs },
15121 .num_dacs = ARRAY_SIZE(alc660vd_dac_nids),
15122 .dac_nids = alc660vd_dac_nids,
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015123 .num_channel_mode = ARRAY_SIZE(alc861vd_3stack_2ch_modes),
15124 .channel_mode = alc861vd_3stack_2ch_modes,
15125 .input_mux = &alc861vd_capture_source,
15126 },
Mike Crash6963f842007-06-25 12:12:51 +020015127 [ALC660VD_3ST_DIG] = {
15128 .mixers = { alc861vd_3st_mixer },
15129 .init_verbs = { alc861vd_volume_init_verbs,
15130 alc861vd_3stack_init_verbs },
15131 .num_dacs = ARRAY_SIZE(alc660vd_dac_nids),
15132 .dac_nids = alc660vd_dac_nids,
15133 .dig_out_nid = ALC861VD_DIGOUT_NID,
Mike Crash6963f842007-06-25 12:12:51 +020015134 .num_channel_mode = ARRAY_SIZE(alc861vd_3stack_2ch_modes),
15135 .channel_mode = alc861vd_3stack_2ch_modes,
15136 .input_mux = &alc861vd_capture_source,
15137 },
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015138 [ALC861VD_3ST] = {
15139 .mixers = { alc861vd_3st_mixer },
15140 .init_verbs = { alc861vd_volume_init_verbs,
15141 alc861vd_3stack_init_verbs },
15142 .num_dacs = ARRAY_SIZE(alc861vd_dac_nids),
15143 .dac_nids = alc861vd_dac_nids,
15144 .num_channel_mode = ARRAY_SIZE(alc861vd_3stack_2ch_modes),
15145 .channel_mode = alc861vd_3stack_2ch_modes,
15146 .input_mux = &alc861vd_capture_source,
15147 },
15148 [ALC861VD_3ST_DIG] = {
15149 .mixers = { alc861vd_3st_mixer },
15150 .init_verbs = { alc861vd_volume_init_verbs,
15151 alc861vd_3stack_init_verbs },
15152 .num_dacs = ARRAY_SIZE(alc861vd_dac_nids),
15153 .dac_nids = alc861vd_dac_nids,
15154 .dig_out_nid = ALC861VD_DIGOUT_NID,
15155 .num_channel_mode = ARRAY_SIZE(alc861vd_3stack_2ch_modes),
15156 .channel_mode = alc861vd_3stack_2ch_modes,
15157 .input_mux = &alc861vd_capture_source,
15158 },
15159 [ALC861VD_6ST_DIG] = {
15160 .mixers = { alc861vd_6st_mixer, alc861vd_chmode_mixer },
15161 .init_verbs = { alc861vd_volume_init_verbs,
15162 alc861vd_6stack_init_verbs },
15163 .num_dacs = ARRAY_SIZE(alc861vd_dac_nids),
15164 .dac_nids = alc861vd_dac_nids,
15165 .dig_out_nid = ALC861VD_DIGOUT_NID,
15166 .num_channel_mode = ARRAY_SIZE(alc861vd_6stack_modes),
15167 .channel_mode = alc861vd_6stack_modes,
15168 .input_mux = &alc861vd_capture_source,
15169 },
Kailang Yangbdd148a2007-05-08 15:19:08 +020015170 [ALC861VD_LENOVO] = {
15171 .mixers = { alc861vd_lenovo_mixer },
15172 .init_verbs = { alc861vd_volume_init_verbs,
15173 alc861vd_3stack_init_verbs,
15174 alc861vd_eapd_verbs,
15175 alc861vd_lenovo_unsol_verbs },
15176 .num_dacs = ARRAY_SIZE(alc660vd_dac_nids),
15177 .dac_nids = alc660vd_dac_nids,
Kailang Yangbdd148a2007-05-08 15:19:08 +020015178 .num_channel_mode = ARRAY_SIZE(alc861vd_3stack_2ch_modes),
15179 .channel_mode = alc861vd_3stack_2ch_modes,
15180 .input_mux = &alc861vd_capture_source,
15181 .unsol_event = alc861vd_lenovo_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020015182 .setup = alc861vd_lenovo_setup,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020015183 .init_hook = alc861vd_lenovo_init_hook,
Kailang Yangbdd148a2007-05-08 15:19:08 +020015184 },
Kailang Yang272a5272007-05-14 11:00:38 +020015185 [ALC861VD_DALLAS] = {
15186 .mixers = { alc861vd_dallas_mixer },
15187 .init_verbs = { alc861vd_dallas_verbs },
15188 .num_dacs = ARRAY_SIZE(alc861vd_dac_nids),
15189 .dac_nids = alc861vd_dac_nids,
Kailang Yang272a5272007-05-14 11:00:38 +020015190 .num_channel_mode = ARRAY_SIZE(alc861vd_3stack_2ch_modes),
15191 .channel_mode = alc861vd_3stack_2ch_modes,
15192 .input_mux = &alc861vd_dallas_capture_source,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020015193 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020015194 .setup = alc861vd_dallas_setup,
15195 .init_hook = alc_automute_amp,
Kailang Yangd1a991a2007-08-15 16:21:59 +020015196 },
15197 [ALC861VD_HP] = {
15198 .mixers = { alc861vd_hp_mixer },
15199 .init_verbs = { alc861vd_dallas_verbs, alc861vd_eapd_verbs },
15200 .num_dacs = ARRAY_SIZE(alc861vd_dac_nids),
15201 .dac_nids = alc861vd_dac_nids,
Kailang Yangd1a991a2007-08-15 16:21:59 +020015202 .dig_out_nid = ALC861VD_DIGOUT_NID,
Kailang Yangd1a991a2007-08-15 16:21:59 +020015203 .num_channel_mode = ARRAY_SIZE(alc861vd_3stack_2ch_modes),
15204 .channel_mode = alc861vd_3stack_2ch_modes,
15205 .input_mux = &alc861vd_hp_capture_source,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020015206 .unsol_event = alc_automute_amp_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020015207 .setup = alc861vd_dallas_setup,
15208 .init_hook = alc_automute_amp,
Kailang Yangea1fb292008-08-26 12:58:38 +020015209 },
Takashi Iwai13c94742008-11-05 08:06:08 +010015210 [ALC660VD_ASUS_V1S] = {
15211 .mixers = { alc861vd_lenovo_mixer },
15212 .init_verbs = { alc861vd_volume_init_verbs,
15213 alc861vd_3stack_init_verbs,
15214 alc861vd_eapd_verbs,
15215 alc861vd_lenovo_unsol_verbs },
15216 .num_dacs = ARRAY_SIZE(alc660vd_dac_nids),
15217 .dac_nids = alc660vd_dac_nids,
15218 .dig_out_nid = ALC861VD_DIGOUT_NID,
15219 .num_channel_mode = ARRAY_SIZE(alc861vd_3stack_2ch_modes),
15220 .channel_mode = alc861vd_3stack_2ch_modes,
15221 .input_mux = &alc861vd_capture_source,
15222 .unsol_event = alc861vd_lenovo_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020015223 .setup = alc861vd_lenovo_setup,
Takashi Iwaia9fd4f32009-05-08 15:57:59 +020015224 .init_hook = alc861vd_lenovo_init_hook,
Takashi Iwai13c94742008-11-05 08:06:08 +010015225 },
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015226};
15227
15228/*
15229 * BIOS auto configuration
15230 */
Takashi Iwai05f5f472009-08-25 13:10:18 +020015231static int alc861vd_auto_create_input_ctls(struct hda_codec *codec,
15232 const struct auto_pin_cfg *cfg)
15233{
15234 return alc_auto_create_input_ctls(codec, cfg, 0x15, 0x09, 0);
15235}
15236
15237
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015238static void alc861vd_auto_set_output_and_unmute(struct hda_codec *codec,
15239 hda_nid_t nid, int pin_type, int dac_idx)
15240{
Takashi Iwaif6c7e542008-02-12 18:32:23 +010015241 alc_set_pin_output(codec, nid, pin_type);
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015242}
15243
15244static void alc861vd_auto_init_multi_out(struct hda_codec *codec)
15245{
15246 struct alc_spec *spec = codec->spec;
15247 int i;
15248
15249 for (i = 0; i <= HDA_SIDE; i++) {
15250 hda_nid_t nid = spec->autocfg.line_out_pins[i];
Takashi Iwaibaba8ee2007-04-23 17:17:48 +020015251 int pin_type = get_pin_type(spec->autocfg.line_out_type);
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015252 if (nid)
15253 alc861vd_auto_set_output_and_unmute(codec, nid,
Takashi Iwaibaba8ee2007-04-23 17:17:48 +020015254 pin_type, i);
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015255 }
15256}
15257
15258
15259static void alc861vd_auto_init_hp_out(struct hda_codec *codec)
15260{
15261 struct alc_spec *spec = codec->spec;
15262 hda_nid_t pin;
15263
15264 pin = spec->autocfg.hp_pins[0];
Sasha Alexandrdef319f2009-06-16 16:00:15 -040015265 if (pin) /* connect to front and use dac 0 */
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015266 alc861vd_auto_set_output_and_unmute(codec, pin, PIN_HP, 0);
Takashi Iwaif6c7e542008-02-12 18:32:23 +010015267 pin = spec->autocfg.speaker_pins[0];
15268 if (pin)
15269 alc861vd_auto_set_output_and_unmute(codec, pin, PIN_OUT, 0);
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015270}
15271
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015272#define ALC861VD_PIN_CD_NID ALC880_PIN_CD_NID
15273
15274static void alc861vd_auto_init_analog_input(struct hda_codec *codec)
15275{
15276 struct alc_spec *spec = codec->spec;
15277 int i;
15278
15279 for (i = 0; i < AUTO_PIN_LAST; i++) {
15280 hda_nid_t nid = spec->autocfg.input_pins[i];
Takashi Iwai05f5f472009-08-25 13:10:18 +020015281 if (alc_is_input_pin(codec, nid)) {
Takashi Iwai23f0c042009-02-26 13:03:58 +010015282 alc_set_input_pin(codec, nid, i);
Takashi Iwaie82c0252009-03-23 15:17:38 +010015283 if (nid != ALC861VD_PIN_CD_NID &&
15284 (get_wcaps(codec, nid) & AC_WCAP_OUT_AMP))
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015285 snd_hda_codec_write(codec, nid, 0,
15286 AC_VERB_SET_AMP_GAIN_MUTE,
15287 AMP_OUT_MUTE);
15288 }
15289 }
15290}
15291
Takashi Iwaif511b012008-08-15 16:46:42 +020015292#define alc861vd_auto_init_input_src alc882_auto_init_input_src
15293
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015294#define alc861vd_idx_to_mixer_vol(nid) ((nid) + 0x02)
15295#define alc861vd_idx_to_mixer_switch(nid) ((nid) + 0x0c)
15296
15297/* add playback controls from the parsed DAC table */
15298/* Based on ALC880 version. But ALC861VD has separate,
15299 * different NIDs for mute/unmute switch and volume control */
15300static int alc861vd_auto_create_multi_out_ctls(struct alc_spec *spec,
15301 const struct auto_pin_cfg *cfg)
15302{
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015303 static const char *chname[4] = {"Front", "Surround", "CLFE", "Side"};
15304 hda_nid_t nid_v, nid_s;
15305 int i, err;
15306
15307 for (i = 0; i < cfg->line_outs; i++) {
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020015308 if (!spec->multiout.dac_nids[i])
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015309 continue;
15310 nid_v = alc861vd_idx_to_mixer_vol(
15311 alc880_dac_to_idx(
15312 spec->multiout.dac_nids[i]));
15313 nid_s = alc861vd_idx_to_mixer_switch(
15314 alc880_dac_to_idx(
15315 spec->multiout.dac_nids[i]));
15316
15317 if (i == 2) {
15318 /* Center/LFE */
Takashi Iwai0afe5f82009-10-02 09:20:00 +020015319 err = add_pb_vol_ctrl(spec, ALC_CTL_WIDGET_VOL,
15320 "Center",
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020015321 HDA_COMPOSE_AMP_VAL(nid_v, 1, 0,
15322 HDA_OUTPUT));
15323 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015324 return err;
Takashi Iwai0afe5f82009-10-02 09:20:00 +020015325 err = add_pb_vol_ctrl(spec, ALC_CTL_WIDGET_VOL,
15326 "LFE",
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020015327 HDA_COMPOSE_AMP_VAL(nid_v, 2, 0,
15328 HDA_OUTPUT));
15329 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015330 return err;
Takashi Iwai0afe5f82009-10-02 09:20:00 +020015331 err = add_pb_sw_ctrl(spec, ALC_CTL_BIND_MUTE,
15332 "Center",
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020015333 HDA_COMPOSE_AMP_VAL(nid_s, 1, 2,
15334 HDA_INPUT));
15335 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015336 return err;
Takashi Iwai0afe5f82009-10-02 09:20:00 +020015337 err = add_pb_sw_ctrl(spec, ALC_CTL_BIND_MUTE,
15338 "LFE",
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020015339 HDA_COMPOSE_AMP_VAL(nid_s, 2, 2,
15340 HDA_INPUT));
15341 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015342 return err;
15343 } else {
Takashi Iwaia4fcd492009-08-25 16:12:15 +020015344 const char *pfx;
15345 if (cfg->line_outs == 1 &&
15346 cfg->line_out_type == AUTO_PIN_SPEAKER_OUT) {
15347 if (!cfg->hp_pins)
15348 pfx = "Speaker";
15349 else
15350 pfx = "PCM";
15351 } else
15352 pfx = chname[i];
Takashi Iwai0afe5f82009-10-02 09:20:00 +020015353 err = add_pb_vol_ctrl(spec, ALC_CTL_WIDGET_VOL, pfx,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020015354 HDA_COMPOSE_AMP_VAL(nid_v, 3, 0,
15355 HDA_OUTPUT));
15356 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015357 return err;
Takashi Iwaia4fcd492009-08-25 16:12:15 +020015358 if (cfg->line_outs == 1 &&
15359 cfg->line_out_type == AUTO_PIN_SPEAKER_OUT)
15360 pfx = "Speaker";
Takashi Iwai0afe5f82009-10-02 09:20:00 +020015361 err = add_pb_sw_ctrl(spec, ALC_CTL_BIND_MUTE, pfx,
Kailang Yangbdd148a2007-05-08 15:19:08 +020015362 HDA_COMPOSE_AMP_VAL(nid_s, 3, 2,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020015363 HDA_INPUT));
15364 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015365 return err;
15366 }
15367 }
15368 return 0;
15369}
15370
15371/* add playback controls for speaker and HP outputs */
15372/* Based on ALC880 version. But ALC861VD has separate,
15373 * different NIDs for mute/unmute switch and volume control */
15374static int alc861vd_auto_create_extra_out(struct alc_spec *spec,
15375 hda_nid_t pin, const char *pfx)
15376{
15377 hda_nid_t nid_v, nid_s;
15378 int err;
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015379
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020015380 if (!pin)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015381 return 0;
15382
15383 if (alc880_is_fixed_pin(pin)) {
15384 nid_v = alc880_idx_to_dac(alc880_fixed_pin_idx(pin));
15385 /* specify the DAC as the extra output */
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020015386 if (!spec->multiout.hp_nid)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015387 spec->multiout.hp_nid = nid_v;
15388 else
15389 spec->multiout.extra_out_nid[0] = nid_v;
15390 /* control HP volume/switch on the output mixer amp */
15391 nid_v = alc861vd_idx_to_mixer_vol(
15392 alc880_fixed_pin_idx(pin));
15393 nid_s = alc861vd_idx_to_mixer_switch(
15394 alc880_fixed_pin_idx(pin));
15395
Takashi Iwai0afe5f82009-10-02 09:20:00 +020015396 err = add_pb_vol_ctrl(spec, ALC_CTL_WIDGET_VOL, pfx,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020015397 HDA_COMPOSE_AMP_VAL(nid_v, 3, 0, HDA_OUTPUT));
15398 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015399 return err;
Takashi Iwai0afe5f82009-10-02 09:20:00 +020015400 err = add_pb_sw_ctrl(spec, ALC_CTL_BIND_MUTE, pfx,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020015401 HDA_COMPOSE_AMP_VAL(nid_s, 3, 2, HDA_INPUT));
15402 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015403 return err;
15404 } else if (alc880_is_multi_pin(pin)) {
15405 /* set manual connection */
15406 /* we have only a switch on HP-out PIN */
Takashi Iwai0afe5f82009-10-02 09:20:00 +020015407 err = add_pb_sw_ctrl(spec, ALC_CTL_WIDGET_MUTE, pfx,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020015408 HDA_COMPOSE_AMP_VAL(pin, 3, 0, HDA_OUTPUT));
15409 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015410 return err;
15411 }
15412 return 0;
15413}
15414
15415/* parse the BIOS configuration and set up the alc_spec
15416 * return 1 if successful, 0 if the proper config is not found,
15417 * or a negative error code
15418 * Based on ALC880 version - had to change it to override
15419 * alc880_auto_create_extra_out and alc880_auto_create_multi_out_ctls */
15420static int alc861vd_parse_auto_config(struct hda_codec *codec)
15421{
15422 struct alc_spec *spec = codec->spec;
15423 int err;
15424 static hda_nid_t alc861vd_ignore[] = { 0x1d, 0 };
15425
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020015426 err = snd_hda_parse_pin_def_config(codec, &spec->autocfg,
15427 alc861vd_ignore);
15428 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015429 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020015430 if (!spec->autocfg.line_outs)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015431 return 0; /* can't find valid BIOS pin config */
15432
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020015433 err = alc880_auto_fill_dac_nids(spec, &spec->autocfg);
15434 if (err < 0)
15435 return err;
15436 err = alc861vd_auto_create_multi_out_ctls(spec, &spec->autocfg);
15437 if (err < 0)
15438 return err;
15439 err = alc861vd_auto_create_extra_out(spec,
15440 spec->autocfg.speaker_pins[0],
15441 "Speaker");
15442 if (err < 0)
15443 return err;
15444 err = alc861vd_auto_create_extra_out(spec,
15445 spec->autocfg.hp_pins[0],
15446 "Headphone");
15447 if (err < 0)
15448 return err;
Takashi Iwai05f5f472009-08-25 13:10:18 +020015449 err = alc861vd_auto_create_input_ctls(codec, &spec->autocfg);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020015450 if (err < 0)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015451 return err;
15452
15453 spec->multiout.max_channels = spec->multiout.num_dacs * 2;
15454
Takashi Iwai0852d7a2009-02-11 11:35:15 +010015455 if (spec->autocfg.dig_outs)
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015456 spec->multiout.dig_out_nid = ALC861VD_DIGOUT_NID;
15457
Takashi Iwai603c4012008-07-30 15:01:44 +020015458 if (spec->kctls.list)
Takashi Iwaid88897e2008-10-31 15:01:37 +010015459 add_mixer(spec, spec->kctls.list);
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015460
Takashi Iwaid88897e2008-10-31 15:01:37 +010015461 add_verb(spec, alc861vd_volume_init_verbs);
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015462
15463 spec->num_mux_defs = 1;
Herton Ronaldo Krzesinski61b9b9b2009-01-28 09:16:33 -020015464 spec->input_mux = &spec->private_imux[0];
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015465
Takashi Iwai776e1842007-08-29 15:07:11 +020015466 err = alc_auto_add_mic_boost(codec);
15467 if (err < 0)
15468 return err;
15469
Takashi Iwai4a79ba32009-04-22 16:31:35 +020015470 alc_ssid_check(codec, 0x15, 0x1b, 0x14);
15471
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015472 return 1;
15473}
15474
15475/* additional initialization for auto-configuration model */
15476static void alc861vd_auto_init(struct hda_codec *codec)
15477{
Takashi Iwaif6c7e542008-02-12 18:32:23 +010015478 struct alc_spec *spec = codec->spec;
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015479 alc861vd_auto_init_multi_out(codec);
15480 alc861vd_auto_init_hp_out(codec);
15481 alc861vd_auto_init_analog_input(codec);
Takashi Iwaif511b012008-08-15 16:46:42 +020015482 alc861vd_auto_init_input_src(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +010015483 if (spec->unsol_event)
Kailang Yang7fb0d782008-10-15 11:12:35 +020015484 alc_inithook(codec);
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015485}
15486
Takashi Iwaif8f25ba2009-10-06 08:31:29 +020015487enum {
15488 ALC660VD_FIX_ASUS_GPIO1
15489};
15490
15491/* reset GPIO1 */
15492static const struct hda_verb alc660vd_fix_asus_gpio1_verbs[] = {
15493 {0x01, AC_VERB_SET_GPIO_MASK, 0x03},
15494 {0x01, AC_VERB_SET_GPIO_DIRECTION, 0x01},
15495 {0x01, AC_VERB_SET_GPIO_DATA, 0x01},
15496 { }
15497};
15498
15499static const struct alc_fixup alc861vd_fixups[] = {
15500 [ALC660VD_FIX_ASUS_GPIO1] = {
15501 .verbs = alc660vd_fix_asus_gpio1_verbs,
15502 },
15503};
15504
15505static struct snd_pci_quirk alc861vd_fixup_tbl[] = {
15506 SND_PCI_QUIRK(0x1043, 0x1339, "ASUS A7-K", ALC660VD_FIX_ASUS_GPIO1),
15507 {}
15508};
15509
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015510static int patch_alc861vd(struct hda_codec *codec)
15511{
15512 struct alc_spec *spec;
15513 int err, board_config;
15514
15515 spec = kzalloc(sizeof(*spec), GFP_KERNEL);
15516 if (spec == NULL)
15517 return -ENOMEM;
15518
15519 codec->spec = spec;
15520
15521 board_config = snd_hda_check_board_config(codec, ALC861VD_MODEL_LAST,
15522 alc861vd_models,
15523 alc861vd_cfg_tbl);
15524
15525 if (board_config < 0 || board_config >= ALC861VD_MODEL_LAST) {
Takashi Iwai9a11f1a2009-07-28 16:01:20 +020015526 printk(KERN_INFO "hda_codec: %s: BIOS auto-probing.\n",
15527 codec->chip_name);
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015528 board_config = ALC861VD_AUTO;
15529 }
15530
Takashi Iwaif8f25ba2009-10-06 08:31:29 +020015531 alc_pick_fixup(codec, alc861vd_fixup_tbl, alc861vd_fixups);
15532
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015533 if (board_config == ALC861VD_AUTO) {
15534 /* automatic parse from the BIOS config */
15535 err = alc861vd_parse_auto_config(codec);
15536 if (err < 0) {
15537 alc_free(codec);
15538 return err;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020015539 } else if (!err) {
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015540 printk(KERN_INFO
15541 "hda_codec: Cannot set up configuration "
15542 "from BIOS. Using base mode...\n");
15543 board_config = ALC861VD_3ST;
15544 }
15545 }
15546
Kusanagi Kouichi680cd532009-02-05 00:00:58 +090015547 err = snd_hda_attach_beep_device(codec, 0x23);
15548 if (err < 0) {
15549 alc_free(codec);
15550 return err;
15551 }
15552
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015553 if (board_config != ALC861VD_AUTO)
Takashi Iwaie9c364c2009-08-11 17:16:13 +020015554 setup_preset(codec, &alc861vd_presets[board_config]);
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015555
Kailang Yang2f893282008-05-27 12:14:47 +020015556 if (codec->vendor_id == 0x10ec0660) {
Kailang Yangf9423e72008-05-27 12:32:25 +020015557 /* always turn on EAPD */
Takashi Iwaid88897e2008-10-31 15:01:37 +010015558 add_verb(spec, alc660vd_eapd_verbs);
Kailang Yang2f893282008-05-27 12:14:47 +020015559 }
15560
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015561 spec->stream_analog_playback = &alc861vd_pcm_analog_playback;
15562 spec->stream_analog_capture = &alc861vd_pcm_analog_capture;
15563
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015564 spec->stream_digital_playback = &alc861vd_pcm_digital_playback;
15565 spec->stream_digital_capture = &alc861vd_pcm_digital_capture;
15566
Takashi Iwaidd704692009-08-11 08:45:11 +020015567 if (!spec->adc_nids) {
15568 spec->adc_nids = alc861vd_adc_nids;
15569 spec->num_adc_nids = ARRAY_SIZE(alc861vd_adc_nids);
15570 }
15571 if (!spec->capsrc_nids)
15572 spec->capsrc_nids = alc861vd_capsrc_nids;
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015573
Takashi Iwaib59bdf32009-08-11 09:47:30 +020015574 set_capture_mixer(codec);
Takashi Iwai45bdd1c2009-02-06 16:11:25 +010015575 set_beep_amp(spec, 0x0b, 0x05, HDA_INPUT);
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015576
Takashi Iwai2134ea42008-01-10 16:53:55 +010015577 spec->vmaster_nid = 0x02;
15578
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015579 codec->patch_ops = alc_patch_ops;
15580
15581 if (board_config == ALC861VD_AUTO)
15582 spec->init_hook = alc861vd_auto_init;
Takashi Iwaicb53c622007-08-10 17:21:45 +020015583#ifdef CONFIG_SND_HDA_POWER_SAVE
15584 if (!spec->loopback.amplist)
15585 spec->loopback.amplist = alc861vd_loopbacks;
15586#endif
Takashi Iwaidaead532008-11-28 12:55:36 +010015587 codec->proc_widget_hook = print_realtek_coef;
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010015588
15589 return 0;
15590}
15591
15592/*
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015593 * ALC662 support
15594 *
15595 * ALC662 is almost identical with ALC880 but has cleaner and more flexible
15596 * configuration. Each pin widget can choose any input DACs and a mixer.
15597 * Each ADC is connected from a mixer of all inputs. This makes possible
15598 * 6-channel independent captures.
15599 *
15600 * In addition, an independent DAC for the multi-playback (not used in this
15601 * driver yet).
15602 */
15603#define ALC662_DIGOUT_NID 0x06
15604#define ALC662_DIGIN_NID 0x0a
15605
15606static hda_nid_t alc662_dac_nids[4] = {
15607 /* front, rear, clfe, rear_surr */
15608 0x02, 0x03, 0x04
15609};
15610
Kailang Yang622e84c2009-04-21 07:39:04 +020015611static hda_nid_t alc272_dac_nids[2] = {
15612 0x02, 0x03
15613};
15614
Takashi Iwaib59bdf32009-08-11 09:47:30 +020015615static hda_nid_t alc662_adc_nids[2] = {
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015616 /* ADC1-2 */
Takashi Iwaib59bdf32009-08-11 09:47:30 +020015617 0x09, 0x08
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015618};
Takashi Iwaie1406342008-02-11 18:32:32 +010015619
Kailang Yang622e84c2009-04-21 07:39:04 +020015620static hda_nid_t alc272_adc_nids[1] = {
15621 /* ADC1-2 */
15622 0x08,
15623};
15624
Takashi Iwaib59bdf32009-08-11 09:47:30 +020015625static hda_nid_t alc662_capsrc_nids[2] = { 0x22, 0x23 };
Kailang Yang622e84c2009-04-21 07:39:04 +020015626static hda_nid_t alc272_capsrc_nids[1] = { 0x23 };
15627
Takashi Iwaie1406342008-02-11 18:32:32 +010015628
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015629/* input MUX */
15630/* FIXME: should be a matrix-type input source selection */
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015631static struct hda_input_mux alc662_capture_source = {
15632 .num_items = 4,
15633 .items = {
15634 { "Mic", 0x0 },
15635 { "Front Mic", 0x1 },
15636 { "Line", 0x2 },
15637 { "CD", 0x4 },
15638 },
15639};
15640
15641static struct hda_input_mux alc662_lenovo_101e_capture_source = {
15642 .num_items = 2,
15643 .items = {
15644 { "Mic", 0x1 },
15645 { "Line", 0x2 },
15646 },
15647};
Kailang Yang291702f2007-10-16 14:28:03 +020015648
Kailang Yang6dda9f42008-05-27 12:05:31 +020015649static struct hda_input_mux alc663_capture_source = {
15650 .num_items = 3,
15651 .items = {
15652 { "Mic", 0x0 },
15653 { "Front Mic", 0x1 },
15654 { "Line", 0x2 },
15655 },
15656};
15657
Takashi Iwai4f5d17062009-08-11 18:17:46 +020015658#if 0 /* set to 1 for testing other input sources below */
Chris Pockelé9541ba12009-05-12 08:08:53 +020015659static struct hda_input_mux alc272_nc10_capture_source = {
15660 .num_items = 16,
15661 .items = {
15662 { "Autoselect Mic", 0x0 },
15663 { "Internal Mic", 0x1 },
15664 { "In-0x02", 0x2 },
15665 { "In-0x03", 0x3 },
15666 { "In-0x04", 0x4 },
15667 { "In-0x05", 0x5 },
15668 { "In-0x06", 0x6 },
15669 { "In-0x07", 0x7 },
15670 { "In-0x08", 0x8 },
15671 { "In-0x09", 0x9 },
15672 { "In-0x0a", 0x0a },
15673 { "In-0x0b", 0x0b },
15674 { "In-0x0c", 0x0c },
15675 { "In-0x0d", 0x0d },
15676 { "In-0x0e", 0x0e },
15677 { "In-0x0f", 0x0f },
15678 },
15679};
15680#endif
15681
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015682/*
15683 * 2ch mode
15684 */
15685static struct hda_channel_mode alc662_3ST_2ch_modes[1] = {
15686 { 2, NULL }
15687};
15688
15689/*
15690 * 2ch mode
15691 */
15692static struct hda_verb alc662_3ST_ch2_init[] = {
15693 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 },
15694 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
15695 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
15696 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
15697 { } /* end */
15698};
15699
15700/*
15701 * 6ch mode
15702 */
15703static struct hda_verb alc662_3ST_ch6_init[] = {
15704 { 0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
15705 { 0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
15706 { 0x18, AC_VERB_SET_CONNECT_SEL, 0x02 },
15707 { 0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
15708 { 0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE },
15709 { 0x1a, AC_VERB_SET_CONNECT_SEL, 0x01 },
15710 { } /* end */
15711};
15712
15713static struct hda_channel_mode alc662_3ST_6ch_modes[2] = {
15714 { 2, alc662_3ST_ch2_init },
15715 { 6, alc662_3ST_ch6_init },
15716};
15717
15718/*
15719 * 2ch mode
15720 */
15721static struct hda_verb alc662_sixstack_ch6_init[] = {
15722 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
15723 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x00 },
15724 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
15725 { } /* end */
15726};
15727
15728/*
15729 * 6ch mode
15730 */
15731static struct hda_verb alc662_sixstack_ch8_init[] = {
15732 { 0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
15733 { 0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
15734 { 0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },
15735 { } /* end */
15736};
15737
15738static struct hda_channel_mode alc662_5stack_modes[2] = {
15739 { 2, alc662_sixstack_ch6_init },
15740 { 6, alc662_sixstack_ch8_init },
15741};
15742
15743/* Pin assignment: Front=0x14, Rear=0x15, CLFE=0x16, Side=0x17
15744 * Mic=0x18, Front Mic=0x19, Line-In=0x1a, HP=0x1b
15745 */
15746
15747static struct snd_kcontrol_new alc662_base_mixer[] = {
15748 /* output mixer control */
15749 HDA_CODEC_VOLUME("Front Playback Volume", 0x2, 0x0, HDA_OUTPUT),
Herton Ronaldo Krzesinski7055ad82008-03-14 12:52:20 +010015750 HDA_CODEC_MUTE("Front Playback Switch", 0x0c, 0x0, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015751 HDA_CODEC_VOLUME("Surround Playback Volume", 0x3, 0x0, HDA_OUTPUT),
Herton Ronaldo Krzesinski7055ad82008-03-14 12:52:20 +010015752 HDA_CODEC_MUTE("Surround Playback Switch", 0x0d, 0x0, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015753 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x04, 1, 0x0, HDA_OUTPUT),
15754 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x04, 2, 0x0, HDA_OUTPUT),
Herton Ronaldo Krzesinski7055ad82008-03-14 12:52:20 +010015755 HDA_CODEC_MUTE_MONO("Center Playback Switch", 0x0e, 1, 0x0, HDA_INPUT),
15756 HDA_CODEC_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 0x0, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015757 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
15758
15759 /*Input mixer control */
15760 HDA_CODEC_VOLUME("CD Playback Volume", 0xb, 0x4, HDA_INPUT),
15761 HDA_CODEC_MUTE("CD Playback Switch", 0xb, 0x4, HDA_INPUT),
15762 HDA_CODEC_VOLUME("Line Playback Volume", 0xb, 0x02, HDA_INPUT),
15763 HDA_CODEC_MUTE("Line Playback Switch", 0xb, 0x02, HDA_INPUT),
15764 HDA_CODEC_VOLUME("Mic Playback Volume", 0xb, 0x0, HDA_INPUT),
15765 HDA_CODEC_MUTE("Mic Playback Switch", 0xb, 0x0, HDA_INPUT),
15766 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0xb, 0x01, HDA_INPUT),
15767 HDA_CODEC_MUTE("Front Mic Playback Switch", 0xb, 0x01, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015768 { } /* end */
15769};
15770
15771static struct snd_kcontrol_new alc662_3ST_2ch_mixer[] = {
15772 HDA_CODEC_VOLUME("Front Playback Volume", 0x02, 0x0, HDA_OUTPUT),
Herton Ronaldo Krzesinski7055ad82008-03-14 12:52:20 +010015773 HDA_CODEC_MUTE("Front Playback Switch", 0x0c, 0x0, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015774 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
15775 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
15776 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
15777 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
15778 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
15779 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
15780 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
15781 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
15782 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015783 { } /* end */
15784};
15785
15786static struct snd_kcontrol_new alc662_3ST_6ch_mixer[] = {
15787 HDA_CODEC_VOLUME("Front Playback Volume", 0x02, 0x0, HDA_OUTPUT),
Herton Ronaldo Krzesinski7055ad82008-03-14 12:52:20 +010015788 HDA_CODEC_MUTE("Front Playback Switch", 0x0c, 0x0, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015789 HDA_CODEC_VOLUME("Surround Playback Volume", 0x03, 0x0, HDA_OUTPUT),
Herton Ronaldo Krzesinski7055ad82008-03-14 12:52:20 +010015790 HDA_CODEC_MUTE("Surround Playback Switch", 0x0d, 0x0, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015791 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x04, 1, 0x0, HDA_OUTPUT),
15792 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x04, 2, 0x0, HDA_OUTPUT),
Herton Ronaldo Krzesinski7055ad82008-03-14 12:52:20 +010015793 HDA_CODEC_MUTE_MONO("Center Playback Switch", 0x0e, 1, 0x0, HDA_INPUT),
15794 HDA_CODEC_MUTE_MONO("LFE Playback Switch", 0x0e, 2, 0x0, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015795 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
15796 HDA_CODEC_VOLUME("CD Playback Volume", 0x0b, 0x04, HDA_INPUT),
15797 HDA_CODEC_MUTE("CD Playback Switch", 0x0b, 0x04, HDA_INPUT),
15798 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
15799 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
15800 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
15801 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
15802 HDA_CODEC_VOLUME("Front Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
15803 HDA_CODEC_MUTE("Front Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015804 { } /* end */
15805};
15806
15807static struct snd_kcontrol_new alc662_lenovo_101e_mixer[] = {
15808 HDA_CODEC_VOLUME("Front Playback Volume", 0x02, 0x0, HDA_OUTPUT),
15809 HDA_BIND_MUTE("Front Playback Switch", 0x02, 2, HDA_INPUT),
Takashi Iwai86cd9292008-01-28 18:09:56 +010015810 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x03, 0x0, HDA_OUTPUT),
15811 HDA_BIND_MUTE("Speaker Playback Switch", 0x03, 2, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015812 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
15813 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
15814 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
15815 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
15816 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015817 { } /* end */
15818};
15819
Kailang Yang291702f2007-10-16 14:28:03 +020015820static struct snd_kcontrol_new alc662_eeepc_p701_mixer[] = {
Takashi Iwai42171c12009-05-08 14:11:43 +020015821 HDA_CODEC_VOLUME("Master Playback Volume", 0x02, 0x0, HDA_OUTPUT),
15822 ALC262_HIPPO_MASTER_SWITCH,
Kailang Yang291702f2007-10-16 14:28:03 +020015823
15824 HDA_CODEC_VOLUME("e-Mic Boost", 0x18, 0, HDA_INPUT),
15825 HDA_CODEC_VOLUME("e-Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
15826 HDA_CODEC_MUTE("e-Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
15827
15828 HDA_CODEC_VOLUME("i-Mic Boost", 0x19, 0, HDA_INPUT),
15829 HDA_CODEC_VOLUME("i-Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
15830 HDA_CODEC_MUTE("i-Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
15831 { } /* end */
15832};
15833
Kailang Yang8c427222008-01-10 13:03:59 +010015834static struct snd_kcontrol_new alc662_eeepc_ep20_mixer[] = {
Takashi Iwai42171c12009-05-08 14:11:43 +020015835 ALC262_HIPPO_MASTER_SWITCH,
15836 HDA_CODEC_VOLUME("Front Playback Volume", 0x02, 0x0, HDA_OUTPUT),
Kailang Yang8c427222008-01-10 13:03:59 +010015837 HDA_CODEC_VOLUME("Surround Playback Volume", 0x03, 0x0, HDA_OUTPUT),
Kailang Yang8c427222008-01-10 13:03:59 +010015838 HDA_CODEC_VOLUME_MONO("Center Playback Volume", 0x04, 1, 0x0, HDA_OUTPUT),
15839 HDA_CODEC_VOLUME_MONO("LFE Playback Volume", 0x04, 2, 0x0, HDA_OUTPUT),
Kailang Yang8c427222008-01-10 13:03:59 +010015840 HDA_BIND_MUTE("MuteCtrl Playback Switch", 0x0c, 2, HDA_INPUT),
15841 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
15842 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
15843 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
15844 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
15845 { } /* end */
15846};
15847
Kailang Yangf1d4e282008-08-26 14:03:29 +020015848static struct hda_bind_ctls alc663_asus_bind_master_vol = {
15849 .ops = &snd_hda_bind_vol,
15850 .values = {
15851 HDA_COMPOSE_AMP_VAL(0x02, 3, 0, HDA_OUTPUT),
15852 HDA_COMPOSE_AMP_VAL(0x03, 3, 0, HDA_OUTPUT),
15853 0
15854 },
15855};
15856
15857static struct hda_bind_ctls alc663_asus_one_bind_switch = {
15858 .ops = &snd_hda_bind_sw,
15859 .values = {
15860 HDA_COMPOSE_AMP_VAL(0x14, 3, 0, HDA_OUTPUT),
15861 HDA_COMPOSE_AMP_VAL(0x21, 3, 0, HDA_OUTPUT),
15862 0
15863 },
15864};
15865
Kailang Yang6dda9f42008-05-27 12:05:31 +020015866static struct snd_kcontrol_new alc663_m51va_mixer[] = {
Kailang Yangf1d4e282008-08-26 14:03:29 +020015867 HDA_BIND_VOL("Master Playback Volume", &alc663_asus_bind_master_vol),
15868 HDA_BIND_SW("Master Playback Switch", &alc663_asus_one_bind_switch),
15869 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
15870 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
15871 { } /* end */
15872};
15873
15874static struct hda_bind_ctls alc663_asus_tree_bind_switch = {
15875 .ops = &snd_hda_bind_sw,
15876 .values = {
15877 HDA_COMPOSE_AMP_VAL(0x14, 3, 0, HDA_OUTPUT),
15878 HDA_COMPOSE_AMP_VAL(0x15, 3, 0, HDA_OUTPUT),
15879 HDA_COMPOSE_AMP_VAL(0x21, 3, 0, HDA_OUTPUT),
15880 0
15881 },
15882};
15883
15884static struct snd_kcontrol_new alc663_two_hp_m1_mixer[] = {
15885 HDA_BIND_VOL("Master Playback Volume", &alc663_asus_bind_master_vol),
15886 HDA_BIND_SW("Master Playback Switch", &alc663_asus_tree_bind_switch),
15887 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
15888 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
15889 HDA_CODEC_VOLUME("F-Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
15890 HDA_CODEC_MUTE("F-Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
15891
15892 { } /* end */
15893};
15894
15895static struct hda_bind_ctls alc663_asus_four_bind_switch = {
15896 .ops = &snd_hda_bind_sw,
15897 .values = {
15898 HDA_COMPOSE_AMP_VAL(0x14, 3, 0, HDA_OUTPUT),
15899 HDA_COMPOSE_AMP_VAL(0x15, 3, 0, HDA_OUTPUT),
15900 HDA_COMPOSE_AMP_VAL(0x1b, 3, 0, HDA_OUTPUT),
15901 0
15902 },
15903};
15904
15905static struct snd_kcontrol_new alc663_two_hp_m2_mixer[] = {
15906 HDA_BIND_VOL("Master Playback Volume", &alc663_asus_bind_master_vol),
15907 HDA_BIND_SW("Master Playback Switch", &alc663_asus_four_bind_switch),
15908 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
15909 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
15910 HDA_CODEC_VOLUME("F-Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
15911 HDA_CODEC_MUTE("F-Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
15912 { } /* end */
15913};
15914
15915static struct snd_kcontrol_new alc662_1bjd_mixer[] = {
Kailang Yang6dda9f42008-05-27 12:05:31 +020015916 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x02, 0x0, HDA_OUTPUT),
15917 HDA_CODEC_MUTE("Speaker Playback Switch", 0x14, 0x0, HDA_OUTPUT),
Kailang Yangf1d4e282008-08-26 14:03:29 +020015918 HDA_CODEC_MUTE("Headphone Playback Switch", 0x1b, 0x0, HDA_OUTPUT),
15919 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
15920 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
15921 HDA_CODEC_VOLUME("F-Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
15922 HDA_CODEC_MUTE("F-Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
15923 { } /* end */
15924};
15925
15926static struct hda_bind_ctls alc663_asus_two_bind_master_vol = {
15927 .ops = &snd_hda_bind_vol,
15928 .values = {
15929 HDA_COMPOSE_AMP_VAL(0x02, 3, 0, HDA_OUTPUT),
15930 HDA_COMPOSE_AMP_VAL(0x04, 3, 0, HDA_OUTPUT),
15931 0
15932 },
15933};
15934
15935static struct hda_bind_ctls alc663_asus_two_bind_switch = {
15936 .ops = &snd_hda_bind_sw,
15937 .values = {
15938 HDA_COMPOSE_AMP_VAL(0x14, 3, 0, HDA_OUTPUT),
15939 HDA_COMPOSE_AMP_VAL(0x16, 3, 0, HDA_OUTPUT),
15940 0
15941 },
15942};
15943
15944static struct snd_kcontrol_new alc663_asus_21jd_clfe_mixer[] = {
15945 HDA_BIND_VOL("Master Playback Volume",
15946 &alc663_asus_two_bind_master_vol),
15947 HDA_BIND_SW("Master Playback Switch", &alc663_asus_two_bind_switch),
15948 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x03, 0x0, HDA_OUTPUT),
Kailang Yang6dda9f42008-05-27 12:05:31 +020015949 HDA_CODEC_MUTE("Headphone Playback Switch", 0x21, 0x0, HDA_OUTPUT),
15950 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
15951 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Kailang Yangf1d4e282008-08-26 14:03:29 +020015952 { } /* end */
15953};
15954
15955static struct snd_kcontrol_new alc663_asus_15jd_clfe_mixer[] = {
15956 HDA_BIND_VOL("Master Playback Volume", &alc663_asus_bind_master_vol),
15957 HDA_BIND_SW("Master Playback Switch", &alc663_asus_two_bind_switch),
15958 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x03, 0x0, HDA_OUTPUT),
15959 HDA_CODEC_MUTE("Headphone Playback Switch", 0x15, 0x0, HDA_OUTPUT),
15960 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
15961 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
Kailang Yang6dda9f42008-05-27 12:05:31 +020015962 { } /* end */
15963};
15964
15965static struct snd_kcontrol_new alc663_g71v_mixer[] = {
15966 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x02, 0x0, HDA_OUTPUT),
15967 HDA_CODEC_MUTE("Speaker Playback Switch", 0x14, 0x0, HDA_OUTPUT),
15968 HDA_CODEC_VOLUME("Front Playback Volume", 0x03, 0x0, HDA_OUTPUT),
15969 HDA_CODEC_MUTE("Front Playback Switch", 0x15, 0x0, HDA_OUTPUT),
15970 HDA_CODEC_MUTE("Headphone Playback Switch", 0x21, 0x0, HDA_OUTPUT),
15971
15972 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
15973 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
15974 HDA_CODEC_VOLUME("i-Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
15975 HDA_CODEC_MUTE("i-Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
15976 { } /* end */
15977};
15978
15979static struct snd_kcontrol_new alc663_g50v_mixer[] = {
15980 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x02, 0x0, HDA_OUTPUT),
15981 HDA_CODEC_MUTE("Speaker Playback Switch", 0x14, 0x0, HDA_OUTPUT),
15982 HDA_CODEC_MUTE("Headphone Playback Switch", 0x21, 0x0, HDA_OUTPUT),
15983
15984 HDA_CODEC_VOLUME("Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
15985 HDA_CODEC_MUTE("Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
15986 HDA_CODEC_VOLUME("i-Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
15987 HDA_CODEC_MUTE("i-Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
15988 HDA_CODEC_VOLUME("Line Playback Volume", 0x0b, 0x02, HDA_INPUT),
15989 HDA_CODEC_MUTE("Line Playback Switch", 0x0b, 0x02, HDA_INPUT),
15990 { } /* end */
15991};
15992
Kailang Yangbc9f98a2007-04-12 13:06:07 +020015993static struct snd_kcontrol_new alc662_chmode_mixer[] = {
15994 {
15995 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
15996 .name = "Channel Mode",
15997 .info = alc_ch_mode_info,
15998 .get = alc_ch_mode_get,
15999 .put = alc_ch_mode_put,
16000 },
16001 { } /* end */
16002};
16003
16004static struct hda_verb alc662_init_verbs[] = {
16005 /* ADC: mute amp left and right */
16006 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
16007 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
16008 /* Front mixer: unmute input/output amp left and right (volume = 0) */
16009
Takashi Iwaicb53c622007-08-10 17:21:45 +020016010 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
16011 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
16012 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
16013 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
16014 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016015
Kailang Yangb60dd392007-09-20 12:50:29 +020016016 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
16017 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
16018 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
16019 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
16020 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
16021 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016022
16023 /* Front Pin: output 0 (0x0c) */
16024 {0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
16025 {0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
16026
16027 /* Rear Pin: output 1 (0x0d) */
16028 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
16029 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
16030
16031 /* CLFE Pin: output 2 (0x0e) */
16032 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
16033 {0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
16034
16035 /* Mic (rear) pin: input vref at 80% */
16036 {0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
16037 {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
16038 /* Front Mic pin: input vref at 80% */
16039 {0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
16040 {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
16041 /* Line In pin: input */
16042 {0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
16043 {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
16044 /* Line-2 In: Headphone output (output 0 - 0x0c) */
16045 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
16046 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
16047 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
16048 /* CD pin widget for input */
16049 {0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
16050
16051 /* FIXME: use matrix-type input source selection */
16052 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
16053 /* Input mixer */
16054 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Kailang Yang291702f2007-10-16 14:28:03 +020016055 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Kailang Yang6dda9f42008-05-27 12:05:31 +020016056
16057 /* always trun on EAPD */
16058 {0x14, AC_VERB_SET_EAPD_BTLENABLE, 2},
16059 {0x15, AC_VERB_SET_EAPD_BTLENABLE, 2},
16060
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016061 { }
16062};
16063
16064static struct hda_verb alc662_sue_init_verbs[] = {
16065 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|ALC880_FRONT_EVENT},
16066 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|ALC880_HP_EVENT},
Kailang Yang291702f2007-10-16 14:28:03 +020016067 {}
16068};
16069
16070static struct hda_verb alc662_eeepc_sue_init_verbs[] = {
16071 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
16072 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
16073 {}
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016074};
16075
Kailang Yang8c427222008-01-10 13:03:59 +010016076/* Set Unsolicited Event*/
16077static struct hda_verb alc662_eeepc_ep20_sue_init_verbs[] = {
16078 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
16079 {0x14, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
16080 {}
16081};
16082
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016083/*
16084 * generic initialization of ADC, input mixers and output mixers
16085 */
16086static struct hda_verb alc662_auto_init_verbs[] = {
16087 /*
16088 * Unmute ADC and set the default input to mic-in
16089 */
16090 {0x09, AC_VERB_SET_CONNECT_SEL, 0x00},
16091 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
16092
16093 /* Unmute input amps (CD, Line In, Mic 1 & Mic 2) of the analog-loopback
16094 * mixer widget
16095 * Note: PASD motherboards uses the Line In 2 as the input for front
16096 * panel mic (mic 2)
16097 */
16098 /* Amp Indices: Mic1 = 0, Mic2 = 1, Line1 = 2, Line2 = 3, CD = 4 */
Takashi Iwaicb53c622007-08-10 17:21:45 +020016099 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
16100 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
16101 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
16102 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
16103 {0x0b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016104
16105 /*
16106 * Set up output mixers (0x0c - 0x0f)
16107 */
16108 /* set vol=0 to output mixers */
16109 {0x02, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
16110 {0x03, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
16111 {0x04, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_ZERO},
16112
16113 /* set up input amps for analog loopback */
16114 /* Amp Indices: DAC = 0, mixer = 1 */
Kailang Yangb60dd392007-09-20 12:50:29 +020016115 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
16116 {0x0c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
16117 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
16118 {0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
16119 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
16120 {0x0e, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016121
16122
16123 /* FIXME: use matrix-type input source selection */
16124 /* Mixer elements: 0x18, 19, 1a, 1b, 1c, 1d, 14, 15, 16, 17, 0b */
16125 /* Input mixer */
16126 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Kailang Yangd1a991a2007-08-15 16:21:59 +020016127 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016128 { }
16129};
16130
Takashi Iwai24fb9172008-09-02 14:48:20 +020016131/* additional verbs for ALC663 */
16132static struct hda_verb alc663_auto_init_verbs[] = {
16133 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
16134 {0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
16135 { }
16136};
16137
Kailang Yang6dda9f42008-05-27 12:05:31 +020016138static struct hda_verb alc663_m51va_init_verbs[] = {
Kailang Yangf1d4e282008-08-26 14:03:29 +020016139 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
16140 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
Kailang Yang6dda9f42008-05-27 12:05:31 +020016141 {0x21, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
16142 {0x21, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
Kailang Yangf1d4e282008-08-26 14:03:29 +020016143 {0x21, AC_VERB_SET_CONNECT_SEL, 0x01}, /* Headphone */
16144 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
16145 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(9)},
Kailang Yang6dda9f42008-05-27 12:05:31 +020016146 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
16147 {0x21, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
16148 {}
16149};
16150
Kailang Yangf1d4e282008-08-26 14:03:29 +020016151static struct hda_verb alc663_21jd_amic_init_verbs[] = {
16152 {0x21, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
16153 {0x21, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
16154 {0x21, AC_VERB_SET_CONNECT_SEL, 0x01}, /* Headphone */
16155 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
16156 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
16157 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
16158 {0x21, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
16159 {}
16160};
16161
16162static struct hda_verb alc662_1bjd_amic_init_verbs[] = {
16163 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
16164 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
16165 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
16166 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x00}, /* Headphone */
16167 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
16168 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
16169 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
16170 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
16171 {}
16172};
16173
16174static struct hda_verb alc663_15jd_amic_init_verbs[] = {
16175 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
16176 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
16177 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01}, /* Headphone */
16178 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
16179 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
16180 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
16181 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
16182 {}
16183};
16184
16185static struct hda_verb alc663_two_hp_amic_m1_init_verbs[] = {
16186 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
16187 {0x21, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
16188 {0x21, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
16189 {0x21, AC_VERB_SET_CONNECT_SEL, 0x0}, /* Headphone */
16190 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
16191 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
16192 {0x15, AC_VERB_SET_CONNECT_SEL, 0x0}, /* Headphone */
16193 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
16194 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
16195 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
16196 {0x21, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
16197 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
16198 {}
16199};
16200
16201static struct hda_verb alc663_two_hp_amic_m2_init_verbs[] = {
16202 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
16203 {0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
16204 {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
16205 {0x1b, AC_VERB_SET_CONNECT_SEL, 0x01}, /* Headphone */
16206 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
16207 {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
16208 {0x15, AC_VERB_SET_CONNECT_SEL, 0x01}, /* Headphone */
16209 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
16210 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)},
16211 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
16212 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
16213 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
16214 {}
16215};
16216
Kailang Yang6dda9f42008-05-27 12:05:31 +020016217static struct hda_verb alc663_g71v_init_verbs[] = {
16218 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
16219 /* {0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE}, */
16220 /* {0x15, AC_VERB_SET_CONNECT_SEL, 0x01}, */ /* Headphone */
16221
16222 {0x21, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
16223 {0x21, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
16224 {0x21, AC_VERB_SET_CONNECT_SEL, 0x00}, /* Headphone */
16225
16226 {0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|ALC880_FRONT_EVENT},
16227 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|ALC880_MIC_EVENT},
16228 {0x21, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|ALC880_HP_EVENT},
16229 {}
16230};
16231
16232static struct hda_verb alc663_g50v_init_verbs[] = {
16233 {0x21, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
16234 {0x21, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
16235 {0x21, AC_VERB_SET_CONNECT_SEL, 0x00}, /* Headphone */
16236
16237 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
16238 {0x21, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
16239 {}
16240};
16241
Kailang Yangf1d4e282008-08-26 14:03:29 +020016242static struct hda_verb alc662_ecs_init_verbs[] = {
16243 {0x09, AC_VERB_SET_AMP_GAIN_MUTE, 0x701f},
16244 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
16245 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
16246 {0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
16247 {}
16248};
16249
Kailang Yang622e84c2009-04-21 07:39:04 +020016250static struct hda_verb alc272_dell_zm1_init_verbs[] = {
16251 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
16252 {0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
16253 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
16254 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
16255 {0x21, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
16256 {0x21, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
16257 {0x21, AC_VERB_SET_CONNECT_SEL, 0x01}, /* Headphone */
16258 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
16259 {0x22, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(9)},
16260 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
16261 {0x21, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
16262 {}
16263};
16264
16265static struct hda_verb alc272_dell_init_verbs[] = {
16266 {0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
16267 {0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
16268 {0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
16269 {0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
16270 {0x21, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
16271 {0x21, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
16272 {0x21, AC_VERB_SET_CONNECT_SEL, 0x01}, /* Headphone */
16273 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
16274 {0x23, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(9)},
16275 {0x18, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_MIC_EVENT},
16276 {0x21, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | ALC880_HP_EVENT},
16277 {}
16278};
16279
Kailang Yangf1d4e282008-08-26 14:03:29 +020016280static struct snd_kcontrol_new alc662_auto_capture_mixer[] = {
16281 HDA_CODEC_VOLUME("Capture Volume", 0x09, 0x0, HDA_INPUT),
16282 HDA_CODEC_MUTE("Capture Switch", 0x09, 0x0, HDA_INPUT),
16283 { } /* end */
16284};
16285
Kailang Yang622e84c2009-04-21 07:39:04 +020016286static struct snd_kcontrol_new alc272_auto_capture_mixer[] = {
16287 HDA_CODEC_VOLUME("Capture Volume", 0x08, 0x0, HDA_INPUT),
16288 HDA_CODEC_MUTE("Capture Switch", 0x08, 0x0, HDA_INPUT),
16289 { } /* end */
16290};
16291
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016292static void alc662_lenovo_101e_ispeaker_automute(struct hda_codec *codec)
16293{
16294 unsigned int present;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020016295 unsigned char bits;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016296
Wu Fengguang864f92b2009-11-18 12:38:02 +080016297 present = snd_hda_jack_detect(codec, 0x14);
Takashi Iwai47fd8302007-08-10 17:11:07 +020016298 bits = present ? HDA_AMP_MUTE : 0;
Wu Fengguang864f92b2009-11-18 12:38:02 +080016299
Takashi Iwai47fd8302007-08-10 17:11:07 +020016300 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
16301 HDA_AMP_MUTE, bits);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016302}
16303
16304static void alc662_lenovo_101e_all_automute(struct hda_codec *codec)
16305{
16306 unsigned int present;
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020016307 unsigned char bits;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016308
Wu Fengguang864f92b2009-11-18 12:38:02 +080016309 present = snd_hda_jack_detect(codec, 0x1b);
Takashi Iwai47fd8302007-08-10 17:11:07 +020016310 bits = present ? HDA_AMP_MUTE : 0;
Wu Fengguang864f92b2009-11-18 12:38:02 +080016311
Takashi Iwai47fd8302007-08-10 17:11:07 +020016312 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
16313 HDA_AMP_MUTE, bits);
16314 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
16315 HDA_AMP_MUTE, bits);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016316}
16317
16318static void alc662_lenovo_101e_unsol_event(struct hda_codec *codec,
16319 unsigned int res)
16320{
16321 if ((res >> 26) == ALC880_HP_EVENT)
16322 alc662_lenovo_101e_all_automute(codec);
16323 if ((res >> 26) == ALC880_FRONT_EVENT)
16324 alc662_lenovo_101e_ispeaker_automute(codec);
16325}
16326
Kailang Yang291702f2007-10-16 14:28:03 +020016327/* unsolicited event for HP jack sensing */
16328static void alc662_eeepc_unsol_event(struct hda_codec *codec,
16329 unsigned int res)
16330{
Kailang Yang291702f2007-10-16 14:28:03 +020016331 if ((res >> 26) == ALC880_MIC_EVENT)
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016332 alc_mic_automute(codec);
Takashi Iwai42171c12009-05-08 14:11:43 +020016333 else
16334 alc262_hippo_unsol_event(codec, res);
Kailang Yang291702f2007-10-16 14:28:03 +020016335}
16336
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016337static void alc662_eeepc_setup(struct hda_codec *codec)
Kailang Yang291702f2007-10-16 14:28:03 +020016338{
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016339 struct alc_spec *spec = codec->spec;
16340
16341 alc262_hippo1_setup(codec);
16342 spec->ext_mic.pin = 0x18;
16343 spec->ext_mic.mux_idx = 0;
16344 spec->int_mic.pin = 0x19;
16345 spec->int_mic.mux_idx = 1;
16346 spec->auto_mic = 1;
Kailang Yang291702f2007-10-16 14:28:03 +020016347}
16348
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016349static void alc662_eeepc_inithook(struct hda_codec *codec)
16350{
16351 alc262_hippo_automute(codec);
16352 alc_mic_automute(codec);
16353}
16354
16355static void alc662_eeepc_ep20_setup(struct hda_codec *codec)
Kailang Yang8c427222008-01-10 13:03:59 +010016356{
Takashi Iwai42171c12009-05-08 14:11:43 +020016357 struct alc_spec *spec = codec->spec;
16358
16359 spec->autocfg.hp_pins[0] = 0x14;
16360 spec->autocfg.speaker_pins[0] = 0x1b;
Kailang Yang8c427222008-01-10 13:03:59 +010016361}
16362
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016363#define alc662_eeepc_ep20_inithook alc262_hippo_master_update
16364
Kailang Yang6dda9f42008-05-27 12:05:31 +020016365static void alc663_m51va_speaker_automute(struct hda_codec *codec)
16366{
16367 unsigned int present;
16368 unsigned char bits;
16369
Wu Fengguang864f92b2009-11-18 12:38:02 +080016370 present = snd_hda_jack_detect(codec, 0x21);
Kailang Yang6dda9f42008-05-27 12:05:31 +020016371 bits = present ? HDA_AMP_MUTE : 0;
Kailang Yangf1d4e282008-08-26 14:03:29 +020016372 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_INPUT, 0,
16373 AMP_IN_MUTE(0), bits);
16374 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_INPUT, 1,
16375 AMP_IN_MUTE(0), bits);
16376}
16377
16378static void alc663_21jd_two_speaker_automute(struct hda_codec *codec)
16379{
16380 unsigned int present;
16381 unsigned char bits;
16382
Wu Fengguang864f92b2009-11-18 12:38:02 +080016383 present = snd_hda_jack_detect(codec, 0x21);
Kailang Yangf1d4e282008-08-26 14:03:29 +020016384 bits = present ? HDA_AMP_MUTE : 0;
16385 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_INPUT, 0,
16386 AMP_IN_MUTE(0), bits);
16387 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_INPUT, 1,
16388 AMP_IN_MUTE(0), bits);
16389 snd_hda_codec_amp_stereo(codec, 0x0e, HDA_INPUT, 0,
16390 AMP_IN_MUTE(0), bits);
16391 snd_hda_codec_amp_stereo(codec, 0x0e, HDA_INPUT, 1,
16392 AMP_IN_MUTE(0), bits);
16393}
16394
16395static void alc663_15jd_two_speaker_automute(struct hda_codec *codec)
16396{
16397 unsigned int present;
16398 unsigned char bits;
16399
Wu Fengguang864f92b2009-11-18 12:38:02 +080016400 present = snd_hda_jack_detect(codec, 0x15);
Kailang Yangf1d4e282008-08-26 14:03:29 +020016401 bits = present ? HDA_AMP_MUTE : 0;
16402 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_INPUT, 0,
16403 AMP_IN_MUTE(0), bits);
16404 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_INPUT, 1,
16405 AMP_IN_MUTE(0), bits);
16406 snd_hda_codec_amp_stereo(codec, 0x0e, HDA_INPUT, 0,
16407 AMP_IN_MUTE(0), bits);
16408 snd_hda_codec_amp_stereo(codec, 0x0e, HDA_INPUT, 1,
16409 AMP_IN_MUTE(0), bits);
16410}
16411
16412static void alc662_f5z_speaker_automute(struct hda_codec *codec)
16413{
16414 unsigned int present;
16415 unsigned char bits;
16416
Wu Fengguang864f92b2009-11-18 12:38:02 +080016417 present = snd_hda_jack_detect(codec, 0x1b);
Kailang Yangf1d4e282008-08-26 14:03:29 +020016418 bits = present ? 0 : PIN_OUT;
16419 snd_hda_codec_write(codec, 0x14, 0,
16420 AC_VERB_SET_PIN_WIDGET_CONTROL, bits);
16421}
16422
16423static void alc663_two_hp_m1_speaker_automute(struct hda_codec *codec)
16424{
16425 unsigned int present1, present2;
16426
Wu Fengguang864f92b2009-11-18 12:38:02 +080016427 present1 = snd_hda_jack_detect(codec, 0x21);
16428 present2 = snd_hda_jack_detect(codec, 0x15);
Kailang Yangf1d4e282008-08-26 14:03:29 +020016429
16430 if (present1 || present2) {
16431 snd_hda_codec_write_cache(codec, 0x14, 0,
16432 AC_VERB_SET_PIN_WIDGET_CONTROL, 0);
16433 } else {
16434 snd_hda_codec_write_cache(codec, 0x14, 0,
16435 AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
16436 }
16437}
16438
16439static void alc663_two_hp_m2_speaker_automute(struct hda_codec *codec)
16440{
16441 unsigned int present1, present2;
16442
Wu Fengguang864f92b2009-11-18 12:38:02 +080016443 present1 = snd_hda_jack_detect(codec, 0x1b);
16444 present2 = snd_hda_jack_detect(codec, 0x15);
Kailang Yangf1d4e282008-08-26 14:03:29 +020016445
16446 if (present1 || present2) {
16447 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_INPUT, 0,
16448 AMP_IN_MUTE(0), AMP_IN_MUTE(0));
16449 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_INPUT, 1,
16450 AMP_IN_MUTE(0), AMP_IN_MUTE(0));
16451 } else {
16452 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_INPUT, 0,
16453 AMP_IN_MUTE(0), 0);
16454 snd_hda_codec_amp_stereo(codec, 0x0c, HDA_INPUT, 1,
16455 AMP_IN_MUTE(0), 0);
16456 }
Kailang Yang6dda9f42008-05-27 12:05:31 +020016457}
16458
Kailang Yang6dda9f42008-05-27 12:05:31 +020016459static void alc663_m51va_unsol_event(struct hda_codec *codec,
16460 unsigned int res)
16461{
16462 switch (res >> 26) {
16463 case ALC880_HP_EVENT:
16464 alc663_m51va_speaker_automute(codec);
16465 break;
16466 case ALC880_MIC_EVENT:
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016467 alc_mic_automute(codec);
Kailang Yang6dda9f42008-05-27 12:05:31 +020016468 break;
16469 }
16470}
16471
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016472static void alc663_m51va_setup(struct hda_codec *codec)
16473{
16474 struct alc_spec *spec = codec->spec;
16475 spec->ext_mic.pin = 0x18;
16476 spec->ext_mic.mux_idx = 0;
16477 spec->int_mic.pin = 0x12;
16478 spec->int_mic.mux_idx = 1;
16479 spec->auto_mic = 1;
16480}
16481
Kailang Yang6dda9f42008-05-27 12:05:31 +020016482static void alc663_m51va_inithook(struct hda_codec *codec)
16483{
16484 alc663_m51va_speaker_automute(codec);
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016485 alc_mic_automute(codec);
Kailang Yang6dda9f42008-05-27 12:05:31 +020016486}
16487
Kailang Yangf1d4e282008-08-26 14:03:29 +020016488/* ***************** Mode1 ******************************/
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016489#define alc663_mode1_unsol_event alc663_m51va_unsol_event
16490#define alc663_mode1_setup alc663_m51va_setup
16491#define alc663_mode1_inithook alc663_m51va_inithook
Kailang Yangf1d4e282008-08-26 14:03:29 +020016492
Kailang Yangf1d4e282008-08-26 14:03:29 +020016493/* ***************** Mode2 ******************************/
16494static void alc662_mode2_unsol_event(struct hda_codec *codec,
16495 unsigned int res)
16496{
16497 switch (res >> 26) {
16498 case ALC880_HP_EVENT:
16499 alc662_f5z_speaker_automute(codec);
16500 break;
16501 case ALC880_MIC_EVENT:
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016502 alc_mic_automute(codec);
Kailang Yangf1d4e282008-08-26 14:03:29 +020016503 break;
16504 }
16505}
16506
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016507#define alc662_mode2_setup alc663_m51va_setup
16508
Kailang Yangf1d4e282008-08-26 14:03:29 +020016509static void alc662_mode2_inithook(struct hda_codec *codec)
16510{
16511 alc662_f5z_speaker_automute(codec);
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016512 alc_mic_automute(codec);
Kailang Yangf1d4e282008-08-26 14:03:29 +020016513}
16514/* ***************** Mode3 ******************************/
16515static void alc663_mode3_unsol_event(struct hda_codec *codec,
16516 unsigned int res)
16517{
16518 switch (res >> 26) {
16519 case ALC880_HP_EVENT:
16520 alc663_two_hp_m1_speaker_automute(codec);
16521 break;
16522 case ALC880_MIC_EVENT:
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016523 alc_mic_automute(codec);
Kailang Yangf1d4e282008-08-26 14:03:29 +020016524 break;
16525 }
16526}
16527
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016528#define alc663_mode3_setup alc663_m51va_setup
16529
Kailang Yangf1d4e282008-08-26 14:03:29 +020016530static void alc663_mode3_inithook(struct hda_codec *codec)
16531{
16532 alc663_two_hp_m1_speaker_automute(codec);
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016533 alc_mic_automute(codec);
Kailang Yangf1d4e282008-08-26 14:03:29 +020016534}
16535/* ***************** Mode4 ******************************/
16536static void alc663_mode4_unsol_event(struct hda_codec *codec,
16537 unsigned int res)
16538{
16539 switch (res >> 26) {
16540 case ALC880_HP_EVENT:
16541 alc663_21jd_two_speaker_automute(codec);
16542 break;
16543 case ALC880_MIC_EVENT:
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016544 alc_mic_automute(codec);
Kailang Yangf1d4e282008-08-26 14:03:29 +020016545 break;
16546 }
16547}
16548
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016549#define alc663_mode4_setup alc663_m51va_setup
16550
Kailang Yangf1d4e282008-08-26 14:03:29 +020016551static void alc663_mode4_inithook(struct hda_codec *codec)
16552{
16553 alc663_21jd_two_speaker_automute(codec);
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016554 alc_mic_automute(codec);
Kailang Yangf1d4e282008-08-26 14:03:29 +020016555}
16556/* ***************** Mode5 ******************************/
16557static void alc663_mode5_unsol_event(struct hda_codec *codec,
16558 unsigned int res)
16559{
16560 switch (res >> 26) {
16561 case ALC880_HP_EVENT:
16562 alc663_15jd_two_speaker_automute(codec);
16563 break;
16564 case ALC880_MIC_EVENT:
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016565 alc_mic_automute(codec);
Kailang Yangf1d4e282008-08-26 14:03:29 +020016566 break;
16567 }
16568}
16569
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016570#define alc663_mode5_setup alc663_m51va_setup
16571
Kailang Yangf1d4e282008-08-26 14:03:29 +020016572static void alc663_mode5_inithook(struct hda_codec *codec)
16573{
16574 alc663_15jd_two_speaker_automute(codec);
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016575 alc_mic_automute(codec);
Kailang Yangf1d4e282008-08-26 14:03:29 +020016576}
16577/* ***************** Mode6 ******************************/
16578static void alc663_mode6_unsol_event(struct hda_codec *codec,
16579 unsigned int res)
16580{
16581 switch (res >> 26) {
16582 case ALC880_HP_EVENT:
16583 alc663_two_hp_m2_speaker_automute(codec);
16584 break;
16585 case ALC880_MIC_EVENT:
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016586 alc_mic_automute(codec);
Kailang Yangf1d4e282008-08-26 14:03:29 +020016587 break;
16588 }
16589}
16590
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016591#define alc663_mode6_setup alc663_m51va_setup
16592
Kailang Yangf1d4e282008-08-26 14:03:29 +020016593static void alc663_mode6_inithook(struct hda_codec *codec)
16594{
16595 alc663_two_hp_m2_speaker_automute(codec);
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016596 alc_mic_automute(codec);
Kailang Yangf1d4e282008-08-26 14:03:29 +020016597}
16598
Kailang Yang6dda9f42008-05-27 12:05:31 +020016599static void alc663_g71v_hp_automute(struct hda_codec *codec)
16600{
16601 unsigned int present;
16602 unsigned char bits;
16603
Wu Fengguang864f92b2009-11-18 12:38:02 +080016604 present = snd_hda_jack_detect(codec, 0x21);
Kailang Yang6dda9f42008-05-27 12:05:31 +020016605 bits = present ? HDA_AMP_MUTE : 0;
16606 snd_hda_codec_amp_stereo(codec, 0x15, HDA_OUTPUT, 0,
16607 HDA_AMP_MUTE, bits);
16608 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
16609 HDA_AMP_MUTE, bits);
16610}
16611
16612static void alc663_g71v_front_automute(struct hda_codec *codec)
16613{
16614 unsigned int present;
16615 unsigned char bits;
16616
Wu Fengguang864f92b2009-11-18 12:38:02 +080016617 present = snd_hda_jack_detect(codec, 0x15);
Kailang Yang6dda9f42008-05-27 12:05:31 +020016618 bits = present ? HDA_AMP_MUTE : 0;
16619 snd_hda_codec_amp_stereo(codec, 0x14, HDA_OUTPUT, 0,
16620 HDA_AMP_MUTE, bits);
16621}
16622
16623static void alc663_g71v_unsol_event(struct hda_codec *codec,
16624 unsigned int res)
16625{
16626 switch (res >> 26) {
16627 case ALC880_HP_EVENT:
16628 alc663_g71v_hp_automute(codec);
16629 break;
16630 case ALC880_FRONT_EVENT:
16631 alc663_g71v_front_automute(codec);
16632 break;
16633 case ALC880_MIC_EVENT:
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016634 alc_mic_automute(codec);
Kailang Yang6dda9f42008-05-27 12:05:31 +020016635 break;
16636 }
16637}
16638
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016639#define alc663_g71v_setup alc663_m51va_setup
16640
Kailang Yang6dda9f42008-05-27 12:05:31 +020016641static void alc663_g71v_inithook(struct hda_codec *codec)
16642{
16643 alc663_g71v_front_automute(codec);
16644 alc663_g71v_hp_automute(codec);
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016645 alc_mic_automute(codec);
Kailang Yang6dda9f42008-05-27 12:05:31 +020016646}
16647
16648static void alc663_g50v_unsol_event(struct hda_codec *codec,
16649 unsigned int res)
16650{
16651 switch (res >> 26) {
16652 case ALC880_HP_EVENT:
16653 alc663_m51va_speaker_automute(codec);
16654 break;
16655 case ALC880_MIC_EVENT:
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016656 alc_mic_automute(codec);
Kailang Yang6dda9f42008-05-27 12:05:31 +020016657 break;
16658 }
16659}
16660
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016661#define alc663_g50v_setup alc663_m51va_setup
16662
Kailang Yang6dda9f42008-05-27 12:05:31 +020016663static void alc663_g50v_inithook(struct hda_codec *codec)
16664{
16665 alc663_m51va_speaker_automute(codec);
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016666 alc_mic_automute(codec);
Kailang Yang6dda9f42008-05-27 12:05:31 +020016667}
16668
Kailang Yangf1d4e282008-08-26 14:03:29 +020016669static struct snd_kcontrol_new alc662_ecs_mixer[] = {
16670 HDA_CODEC_VOLUME("Master Playback Volume", 0x02, 0x0, HDA_OUTPUT),
Takashi Iwai42171c12009-05-08 14:11:43 +020016671 ALC262_HIPPO_MASTER_SWITCH,
Kailang Yangf1d4e282008-08-26 14:03:29 +020016672
16673 HDA_CODEC_VOLUME("e-Mic/LineIn Boost", 0x18, 0, HDA_INPUT),
16674 HDA_CODEC_VOLUME("e-Mic/LineIn Playback Volume", 0x0b, 0x0, HDA_INPUT),
16675 HDA_CODEC_MUTE("e-Mic/LineIn Playback Switch", 0x0b, 0x0, HDA_INPUT),
16676
16677 HDA_CODEC_VOLUME("i-Mic Boost", 0x19, 0, HDA_INPUT),
16678 HDA_CODEC_VOLUME("i-Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
16679 HDA_CODEC_MUTE("i-Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
16680 { } /* end */
16681};
16682
Chris Pockelé9541ba12009-05-12 08:08:53 +020016683static struct snd_kcontrol_new alc272_nc10_mixer[] = {
16684 /* Master Playback automatically created from Speaker and Headphone */
16685 HDA_CODEC_VOLUME("Speaker Playback Volume", 0x02, 0x0, HDA_OUTPUT),
16686 HDA_CODEC_MUTE("Speaker Playback Switch", 0x14, 0x0, HDA_OUTPUT),
16687 HDA_CODEC_VOLUME("Headphone Playback Volume", 0x03, 0x0, HDA_OUTPUT),
16688 HDA_CODEC_MUTE("Headphone Playback Switch", 0x21, 0x0, HDA_OUTPUT),
16689
16690 HDA_CODEC_VOLUME("Ext Mic Playback Volume", 0x0b, 0x0, HDA_INPUT),
16691 HDA_CODEC_MUTE("Ext Mic Playback Switch", 0x0b, 0x0, HDA_INPUT),
16692 HDA_CODEC_VOLUME("Ext Mic Boost", 0x18, 0, HDA_INPUT),
16693
16694 HDA_CODEC_VOLUME("Int Mic Playback Volume", 0x0b, 0x1, HDA_INPUT),
16695 HDA_CODEC_MUTE("Int Mic Playback Switch", 0x0b, 0x1, HDA_INPUT),
16696 HDA_CODEC_VOLUME("Int Mic Boost", 0x19, 0, HDA_INPUT),
16697 { } /* end */
16698};
16699
Takashi Iwaicb53c622007-08-10 17:21:45 +020016700#ifdef CONFIG_SND_HDA_POWER_SAVE
16701#define alc662_loopbacks alc880_loopbacks
16702#endif
16703
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016704
Sasha Alexandrdef319f2009-06-16 16:00:15 -040016705/* pcm configuration: identical with ALC880 */
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016706#define alc662_pcm_analog_playback alc880_pcm_analog_playback
16707#define alc662_pcm_analog_capture alc880_pcm_analog_capture
16708#define alc662_pcm_digital_playback alc880_pcm_digital_playback
16709#define alc662_pcm_digital_capture alc880_pcm_digital_capture
16710
16711/*
16712 * configuration and preset
16713 */
16714static const char *alc662_models[ALC662_MODEL_LAST] = {
16715 [ALC662_3ST_2ch_DIG] = "3stack-dig",
16716 [ALC662_3ST_6ch_DIG] = "3stack-6ch-dig",
16717 [ALC662_3ST_6ch] = "3stack-6ch",
16718 [ALC662_5ST_DIG] = "6stack-dig",
16719 [ALC662_LENOVO_101E] = "lenovo-101e",
Takashi Iwaib995d762007-10-17 10:41:06 +020016720 [ALC662_ASUS_EEEPC_P701] = "eeepc-p701",
Kailang Yang8c427222008-01-10 13:03:59 +010016721 [ALC662_ASUS_EEEPC_EP20] = "eeepc-ep20",
Kailang Yangf1d4e282008-08-26 14:03:29 +020016722 [ALC662_ECS] = "ecs",
Kailang Yang6dda9f42008-05-27 12:05:31 +020016723 [ALC663_ASUS_M51VA] = "m51va",
16724 [ALC663_ASUS_G71V] = "g71v",
16725 [ALC663_ASUS_H13] = "h13",
16726 [ALC663_ASUS_G50V] = "g50v",
Kailang Yangf1d4e282008-08-26 14:03:29 +020016727 [ALC663_ASUS_MODE1] = "asus-mode1",
16728 [ALC662_ASUS_MODE2] = "asus-mode2",
16729 [ALC663_ASUS_MODE3] = "asus-mode3",
16730 [ALC663_ASUS_MODE4] = "asus-mode4",
16731 [ALC663_ASUS_MODE5] = "asus-mode5",
16732 [ALC663_ASUS_MODE6] = "asus-mode6",
Takashi Iwai01f2bd42009-05-11 08:12:43 +020016733 [ALC272_DELL] = "dell",
16734 [ALC272_DELL_ZM1] = "dell-zm1",
Chris Pockelé9541ba12009-05-12 08:08:53 +020016735 [ALC272_SAMSUNG_NC10] = "samsung-nc10",
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016736 [ALC662_AUTO] = "auto",
16737};
16738
16739static struct snd_pci_quirk alc662_cfg_tbl[] = {
Takashi Iwaidea0a502009-02-09 17:14:52 +010016740 SND_PCI_QUIRK(0x1019, 0x9087, "ECS", ALC662_ECS),
Kailang Yang622e84c2009-04-21 07:39:04 +020016741 SND_PCI_QUIRK(0x1028, 0x02d6, "DELL", ALC272_DELL),
16742 SND_PCI_QUIRK(0x1028, 0x02f4, "DELL ZM1", ALC272_DELL_ZM1),
Takashi Iwaidea0a502009-02-09 17:14:52 +010016743 SND_PCI_QUIRK(0x1043, 0x1000, "ASUS N50Vm", ALC663_ASUS_MODE1),
16744 SND_PCI_QUIRK(0x1043, 0x1092, "ASUS NB", ALC663_ASUS_MODE3),
16745 SND_PCI_QUIRK(0x1043, 0x11c3, "ASUS M70V", ALC663_ASUS_MODE3),
16746 SND_PCI_QUIRK(0x1043, 0x11d3, "ASUS NB", ALC663_ASUS_MODE1),
16747 SND_PCI_QUIRK(0x1043, 0x11f3, "ASUS NB", ALC662_ASUS_MODE2),
16748 SND_PCI_QUIRK(0x1043, 0x1203, "ASUS NB", ALC663_ASUS_MODE1),
16749 SND_PCI_QUIRK(0x1043, 0x1339, "ASUS NB", ALC662_ASUS_MODE2),
16750 SND_PCI_QUIRK(0x1043, 0x16c3, "ASUS NB", ALC662_ASUS_MODE2),
16751 SND_PCI_QUIRK(0x1043, 0x1753, "ASUS NB", ALC662_ASUS_MODE2),
16752 SND_PCI_QUIRK(0x1043, 0x1763, "ASUS NB", ALC663_ASUS_MODE6),
16753 SND_PCI_QUIRK(0x1043, 0x1765, "ASUS NB", ALC663_ASUS_MODE6),
16754 SND_PCI_QUIRK(0x1043, 0x1783, "ASUS NB", ALC662_ASUS_MODE2),
Kailang Yang622e84c2009-04-21 07:39:04 +020016755 SND_PCI_QUIRK(0x1043, 0x17b3, "ASUS F70SL", ALC663_ASUS_MODE3),
16756 SND_PCI_QUIRK(0x1043, 0x17c3, "ASUS UX20", ALC663_ASUS_M51VA),
16757 SND_PCI_QUIRK(0x1043, 0x17f3, "ASUS X58LE", ALC662_ASUS_MODE2),
Takashi Iwaidea0a502009-02-09 17:14:52 +010016758 SND_PCI_QUIRK(0x1043, 0x1813, "ASUS NB", ALC662_ASUS_MODE2),
16759 SND_PCI_QUIRK(0x1043, 0x1823, "ASUS NB", ALC663_ASUS_MODE5),
16760 SND_PCI_QUIRK(0x1043, 0x1833, "ASUS NB", ALC663_ASUS_MODE6),
16761 SND_PCI_QUIRK(0x1043, 0x1843, "ASUS NB", ALC662_ASUS_MODE2),
Kailang Yang622e84c2009-04-21 07:39:04 +020016762 SND_PCI_QUIRK(0x1043, 0x1853, "ASUS F50Z", ALC663_ASUS_MODE1),
Takashi Iwaidea0a502009-02-09 17:14:52 +010016763 SND_PCI_QUIRK(0x1043, 0x1864, "ASUS NB", ALC662_ASUS_MODE2),
16764 SND_PCI_QUIRK(0x1043, 0x1876, "ASUS NB", ALC662_ASUS_MODE2),
Kailang Yang6dda9f42008-05-27 12:05:31 +020016765 SND_PCI_QUIRK(0x1043, 0x1878, "ASUS M51VA", ALC663_ASUS_M51VA),
Takashi Iwaidea0a502009-02-09 17:14:52 +010016766 /*SND_PCI_QUIRK(0x1043, 0x1878, "ASUS M50Vr", ALC663_ASUS_MODE1),*/
16767 SND_PCI_QUIRK(0x1043, 0x1893, "ASUS M50Vm", ALC663_ASUS_MODE3),
16768 SND_PCI_QUIRK(0x1043, 0x1894, "ASUS X55", ALC663_ASUS_MODE3),
Kailang Yang622e84c2009-04-21 07:39:04 +020016769 SND_PCI_QUIRK(0x1043, 0x18b3, "ASUS N80Vc", ALC663_ASUS_MODE1),
16770 SND_PCI_QUIRK(0x1043, 0x18d3, "ASUS N81Te", ALC663_ASUS_MODE1),
16771 SND_PCI_QUIRK(0x1043, 0x18f3, "ASUS N505Tp", ALC663_ASUS_MODE1),
Takashi Iwaidea0a502009-02-09 17:14:52 +010016772 SND_PCI_QUIRK(0x1043, 0x1903, "ASUS F5GL", ALC663_ASUS_MODE1),
16773 SND_PCI_QUIRK(0x1043, 0x1913, "ASUS NB", ALC662_ASUS_MODE2),
16774 SND_PCI_QUIRK(0x1043, 0x1933, "ASUS F80Q", ALC662_ASUS_MODE2),
Kailang Yang622e84c2009-04-21 07:39:04 +020016775 SND_PCI_QUIRK(0x1043, 0x1943, "ASUS Vx3V", ALC663_ASUS_MODE1),
Takashi Iwaidea0a502009-02-09 17:14:52 +010016776 SND_PCI_QUIRK(0x1043, 0x1953, "ASUS NB", ALC663_ASUS_MODE1),
16777 SND_PCI_QUIRK(0x1043, 0x1963, "ASUS X71C", ALC663_ASUS_MODE3),
Kailang Yang622e84c2009-04-21 07:39:04 +020016778 SND_PCI_QUIRK(0x1043, 0x1983, "ASUS N5051A", ALC663_ASUS_MODE1),
Takashi Iwaidea0a502009-02-09 17:14:52 +010016779 SND_PCI_QUIRK(0x1043, 0x1993, "ASUS N20", ALC663_ASUS_MODE1),
Kailang Yang80ffe862008-10-15 11:23:27 +020016780 SND_PCI_QUIRK(0x1043, 0x19a3, "ASUS G50V", ALC663_ASUS_G50V),
Takashi Iwaidea0a502009-02-09 17:14:52 +010016781 /*SND_PCI_QUIRK(0x1043, 0x19a3, "ASUS NB", ALC663_ASUS_MODE1),*/
16782 SND_PCI_QUIRK(0x1043, 0x19b3, "ASUS F7Z", ALC663_ASUS_MODE1),
16783 SND_PCI_QUIRK(0x1043, 0x19c3, "ASUS F5Z/F6x", ALC662_ASUS_MODE2),
Kailang Yang622e84c2009-04-21 07:39:04 +020016784 SND_PCI_QUIRK(0x1043, 0x19d3, "ASUS NB", ALC663_ASUS_M51VA),
Takashi Iwaidea0a502009-02-09 17:14:52 +010016785 SND_PCI_QUIRK(0x1043, 0x19e3, "ASUS NB", ALC663_ASUS_MODE1),
16786 SND_PCI_QUIRK(0x1043, 0x19f3, "ASUS NB", ALC663_ASUS_MODE4),
Herton Ronaldo Krzesinski3da23ca2008-03-14 12:52:59 +010016787 SND_PCI_QUIRK(0x1043, 0x8290, "ASUS P5GC-MX", ALC662_3ST_6ch_DIG),
Kailang Yang291702f2007-10-16 14:28:03 +020016788 SND_PCI_QUIRK(0x1043, 0x82a1, "ASUS Eeepc", ALC662_ASUS_EEEPC_P701),
Kailang Yang8c427222008-01-10 13:03:59 +010016789 SND_PCI_QUIRK(0x1043, 0x82d1, "ASUS Eeepc EP20", ALC662_ASUS_EEEPC_EP20),
Takashi Iwaidea0a502009-02-09 17:14:52 +010016790 SND_PCI_QUIRK(0x105b, 0x0cd6, "Foxconn", ALC662_ECS),
Herton Ronaldo Krzesinski95fe5f22008-09-26 23:48:45 -030016791 SND_PCI_QUIRK(0x105b, 0x0d47, "Foxconn 45CMX/45GMX/45CMX-K",
16792 ALC662_3ST_6ch_DIG),
Manoj Iyer3db6c032009-09-22 18:33:29 -050016793 SND_PCI_QUIRK(0x1179, 0xff6e, "Toshiba NB200", ALC663_ASUS_MODE4),
Chris Pockelé9541ba12009-05-12 08:08:53 +020016794 SND_PCI_QUIRK(0x144d, 0xca00, "Samsung NC10", ALC272_SAMSUNG_NC10),
Herton Ronaldo Krzesinskicb559742008-09-26 23:47:45 -030016795 SND_PCI_QUIRK(0x1458, 0xa002, "Gigabyte 945GCM-S2L",
16796 ALC662_3ST_6ch_DIG),
Vedran Miletic19c009a2008-09-29 20:29:25 +020016797 SND_PCI_QUIRK(0x1565, 0x820f, "Biostar TA780G M2+", ALC662_3ST_6ch_DIG),
Takashi Iwai5bd37292009-04-21 18:36:30 +020016798 SND_PCI_QUIRK(0x1631, 0xc10c, "PB RS65", ALC663_ASUS_M51VA),
Takashi Iwaidea0a502009-02-09 17:14:52 +010016799 SND_PCI_QUIRK(0x17aa, 0x101e, "Lenovo", ALC662_LENOVO_101E),
Takashi Iwai238713d2008-10-05 10:57:39 +020016800 SND_PCI_QUIRK(0x1849, 0x3662, "ASROCK K10N78FullHD-hSLI R3.0",
Vedran Miletic19c009a2008-09-29 20:29:25 +020016801 ALC662_3ST_6ch_DIG),
Takashi Iwaidea0a502009-02-09 17:14:52 +010016802 SND_PCI_QUIRK_MASK(0x1854, 0xf000, 0x2000, "ASUS H13-200x",
16803 ALC663_ASUS_H13),
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016804 {}
16805};
16806
16807static struct alc_config_preset alc662_presets[] = {
16808 [ALC662_3ST_2ch_DIG] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010016809 .mixers = { alc662_3ST_2ch_mixer },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016810 .init_verbs = { alc662_init_verbs },
16811 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
16812 .dac_nids = alc662_dac_nids,
16813 .dig_out_nid = ALC662_DIGOUT_NID,
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016814 .dig_in_nid = ALC662_DIGIN_NID,
16815 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
16816 .channel_mode = alc662_3ST_2ch_modes,
16817 .input_mux = &alc662_capture_source,
16818 },
16819 [ALC662_3ST_6ch_DIG] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010016820 .mixers = { alc662_3ST_6ch_mixer, alc662_chmode_mixer },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016821 .init_verbs = { alc662_init_verbs },
16822 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
16823 .dac_nids = alc662_dac_nids,
16824 .dig_out_nid = ALC662_DIGOUT_NID,
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016825 .dig_in_nid = ALC662_DIGIN_NID,
16826 .num_channel_mode = ARRAY_SIZE(alc662_3ST_6ch_modes),
16827 .channel_mode = alc662_3ST_6ch_modes,
16828 .need_dac_fix = 1,
16829 .input_mux = &alc662_capture_source,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020016830 },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016831 [ALC662_3ST_6ch] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010016832 .mixers = { alc662_3ST_6ch_mixer, alc662_chmode_mixer },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016833 .init_verbs = { alc662_init_verbs },
16834 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
16835 .dac_nids = alc662_dac_nids,
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016836 .num_channel_mode = ARRAY_SIZE(alc662_3ST_6ch_modes),
16837 .channel_mode = alc662_3ST_6ch_modes,
16838 .need_dac_fix = 1,
16839 .input_mux = &alc662_capture_source,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020016840 },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016841 [ALC662_5ST_DIG] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010016842 .mixers = { alc662_base_mixer, alc662_chmode_mixer },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016843 .init_verbs = { alc662_init_verbs },
16844 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
16845 .dac_nids = alc662_dac_nids,
16846 .dig_out_nid = ALC662_DIGOUT_NID,
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016847 .dig_in_nid = ALC662_DIGIN_NID,
16848 .num_channel_mode = ARRAY_SIZE(alc662_5stack_modes),
16849 .channel_mode = alc662_5stack_modes,
16850 .input_mux = &alc662_capture_source,
16851 },
16852 [ALC662_LENOVO_101E] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010016853 .mixers = { alc662_lenovo_101e_mixer },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016854 .init_verbs = { alc662_init_verbs, alc662_sue_init_verbs },
16855 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
16856 .dac_nids = alc662_dac_nids,
Kailang Yangbc9f98a2007-04-12 13:06:07 +020016857 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
16858 .channel_mode = alc662_3ST_2ch_modes,
16859 .input_mux = &alc662_lenovo_101e_capture_source,
16860 .unsol_event = alc662_lenovo_101e_unsol_event,
16861 .init_hook = alc662_lenovo_101e_all_automute,
16862 },
Kailang Yang291702f2007-10-16 14:28:03 +020016863 [ALC662_ASUS_EEEPC_P701] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010016864 .mixers = { alc662_eeepc_p701_mixer },
Kailang Yang291702f2007-10-16 14:28:03 +020016865 .init_verbs = { alc662_init_verbs,
16866 alc662_eeepc_sue_init_verbs },
16867 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
16868 .dac_nids = alc662_dac_nids,
Kailang Yang291702f2007-10-16 14:28:03 +020016869 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
16870 .channel_mode = alc662_3ST_2ch_modes,
Kailang Yang291702f2007-10-16 14:28:03 +020016871 .unsol_event = alc662_eeepc_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016872 .setup = alc662_eeepc_setup,
Kailang Yang291702f2007-10-16 14:28:03 +020016873 .init_hook = alc662_eeepc_inithook,
16874 },
Kailang Yang8c427222008-01-10 13:03:59 +010016875 [ALC662_ASUS_EEEPC_EP20] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010016876 .mixers = { alc662_eeepc_ep20_mixer,
Kailang Yang8c427222008-01-10 13:03:59 +010016877 alc662_chmode_mixer },
16878 .init_verbs = { alc662_init_verbs,
16879 alc662_eeepc_ep20_sue_init_verbs },
16880 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
16881 .dac_nids = alc662_dac_nids,
Kailang Yang8c427222008-01-10 13:03:59 +010016882 .num_channel_mode = ARRAY_SIZE(alc662_3ST_6ch_modes),
16883 .channel_mode = alc662_3ST_6ch_modes,
16884 .input_mux = &alc662_lenovo_101e_capture_source,
Takashi Iwai42171c12009-05-08 14:11:43 +020016885 .unsol_event = alc662_eeepc_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016886 .setup = alc662_eeepc_ep20_setup,
Kailang Yang8c427222008-01-10 13:03:59 +010016887 .init_hook = alc662_eeepc_ep20_inithook,
16888 },
Kailang Yangf1d4e282008-08-26 14:03:29 +020016889 [ALC662_ECS] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010016890 .mixers = { alc662_ecs_mixer },
Kailang Yangf1d4e282008-08-26 14:03:29 +020016891 .init_verbs = { alc662_init_verbs,
16892 alc662_ecs_init_verbs },
16893 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
16894 .dac_nids = alc662_dac_nids,
16895 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
16896 .channel_mode = alc662_3ST_2ch_modes,
Kailang Yangf1d4e282008-08-26 14:03:29 +020016897 .unsol_event = alc662_eeepc_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016898 .setup = alc662_eeepc_setup,
Kailang Yangf1d4e282008-08-26 14:03:29 +020016899 .init_hook = alc662_eeepc_inithook,
16900 },
Kailang Yang6dda9f42008-05-27 12:05:31 +020016901 [ALC663_ASUS_M51VA] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010016902 .mixers = { alc663_m51va_mixer },
Kailang Yang6dda9f42008-05-27 12:05:31 +020016903 .init_verbs = { alc662_init_verbs, alc663_m51va_init_verbs },
16904 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
16905 .dac_nids = alc662_dac_nids,
16906 .dig_out_nid = ALC662_DIGOUT_NID,
16907 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
16908 .channel_mode = alc662_3ST_2ch_modes,
Kailang Yang6dda9f42008-05-27 12:05:31 +020016909 .unsol_event = alc663_m51va_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016910 .setup = alc663_m51va_setup,
Kailang Yang6dda9f42008-05-27 12:05:31 +020016911 .init_hook = alc663_m51va_inithook,
16912 },
16913 [ALC663_ASUS_G71V] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010016914 .mixers = { alc663_g71v_mixer },
Kailang Yang6dda9f42008-05-27 12:05:31 +020016915 .init_verbs = { alc662_init_verbs, alc663_g71v_init_verbs },
16916 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
16917 .dac_nids = alc662_dac_nids,
16918 .dig_out_nid = ALC662_DIGOUT_NID,
16919 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
16920 .channel_mode = alc662_3ST_2ch_modes,
Kailang Yang6dda9f42008-05-27 12:05:31 +020016921 .unsol_event = alc663_g71v_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016922 .setup = alc663_g71v_setup,
Kailang Yang6dda9f42008-05-27 12:05:31 +020016923 .init_hook = alc663_g71v_inithook,
16924 },
16925 [ALC663_ASUS_H13] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010016926 .mixers = { alc663_m51va_mixer },
Kailang Yang6dda9f42008-05-27 12:05:31 +020016927 .init_verbs = { alc662_init_verbs, alc663_m51va_init_verbs },
16928 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
16929 .dac_nids = alc662_dac_nids,
16930 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
16931 .channel_mode = alc662_3ST_2ch_modes,
Kailang Yang6dda9f42008-05-27 12:05:31 +020016932 .unsol_event = alc663_m51va_unsol_event,
16933 .init_hook = alc663_m51va_inithook,
16934 },
16935 [ALC663_ASUS_G50V] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010016936 .mixers = { alc663_g50v_mixer },
Kailang Yang6dda9f42008-05-27 12:05:31 +020016937 .init_verbs = { alc662_init_verbs, alc663_g50v_init_verbs },
16938 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
16939 .dac_nids = alc662_dac_nids,
16940 .dig_out_nid = ALC662_DIGOUT_NID,
16941 .num_channel_mode = ARRAY_SIZE(alc662_3ST_6ch_modes),
16942 .channel_mode = alc662_3ST_6ch_modes,
16943 .input_mux = &alc663_capture_source,
16944 .unsol_event = alc663_g50v_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016945 .setup = alc663_g50v_setup,
Kailang Yang6dda9f42008-05-27 12:05:31 +020016946 .init_hook = alc663_g50v_inithook,
16947 },
Kailang Yangf1d4e282008-08-26 14:03:29 +020016948 [ALC663_ASUS_MODE1] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010016949 .mixers = { alc663_m51va_mixer },
16950 .cap_mixer = alc662_auto_capture_mixer,
Kailang Yangf1d4e282008-08-26 14:03:29 +020016951 .init_verbs = { alc662_init_verbs,
16952 alc663_21jd_amic_init_verbs },
16953 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
16954 .hp_nid = 0x03,
16955 .dac_nids = alc662_dac_nids,
16956 .dig_out_nid = ALC662_DIGOUT_NID,
16957 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
16958 .channel_mode = alc662_3ST_2ch_modes,
Kailang Yangf1d4e282008-08-26 14:03:29 +020016959 .unsol_event = alc663_mode1_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016960 .setup = alc663_mode1_setup,
Kailang Yangf1d4e282008-08-26 14:03:29 +020016961 .init_hook = alc663_mode1_inithook,
16962 },
16963 [ALC662_ASUS_MODE2] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010016964 .mixers = { alc662_1bjd_mixer },
16965 .cap_mixer = alc662_auto_capture_mixer,
Kailang Yangf1d4e282008-08-26 14:03:29 +020016966 .init_verbs = { alc662_init_verbs,
16967 alc662_1bjd_amic_init_verbs },
16968 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
16969 .dac_nids = alc662_dac_nids,
16970 .dig_out_nid = ALC662_DIGOUT_NID,
16971 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
16972 .channel_mode = alc662_3ST_2ch_modes,
Kailang Yangf1d4e282008-08-26 14:03:29 +020016973 .unsol_event = alc662_mode2_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016974 .setup = alc662_mode2_setup,
Kailang Yangf1d4e282008-08-26 14:03:29 +020016975 .init_hook = alc662_mode2_inithook,
16976 },
16977 [ALC663_ASUS_MODE3] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010016978 .mixers = { alc663_two_hp_m1_mixer },
16979 .cap_mixer = alc662_auto_capture_mixer,
Kailang Yangf1d4e282008-08-26 14:03:29 +020016980 .init_verbs = { alc662_init_verbs,
16981 alc663_two_hp_amic_m1_init_verbs },
16982 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
16983 .hp_nid = 0x03,
16984 .dac_nids = alc662_dac_nids,
16985 .dig_out_nid = ALC662_DIGOUT_NID,
16986 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
16987 .channel_mode = alc662_3ST_2ch_modes,
Kailang Yangf1d4e282008-08-26 14:03:29 +020016988 .unsol_event = alc663_mode3_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020016989 .setup = alc663_mode3_setup,
Kailang Yangf1d4e282008-08-26 14:03:29 +020016990 .init_hook = alc663_mode3_inithook,
16991 },
16992 [ALC663_ASUS_MODE4] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010016993 .mixers = { alc663_asus_21jd_clfe_mixer },
16994 .cap_mixer = alc662_auto_capture_mixer,
Kailang Yangf1d4e282008-08-26 14:03:29 +020016995 .init_verbs = { alc662_init_verbs,
16996 alc663_21jd_amic_init_verbs},
16997 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
16998 .hp_nid = 0x03,
16999 .dac_nids = alc662_dac_nids,
17000 .dig_out_nid = ALC662_DIGOUT_NID,
17001 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
17002 .channel_mode = alc662_3ST_2ch_modes,
Kailang Yangf1d4e282008-08-26 14:03:29 +020017003 .unsol_event = alc663_mode4_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020017004 .setup = alc663_mode4_setup,
Kailang Yangf1d4e282008-08-26 14:03:29 +020017005 .init_hook = alc663_mode4_inithook,
17006 },
17007 [ALC663_ASUS_MODE5] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010017008 .mixers = { alc663_asus_15jd_clfe_mixer },
17009 .cap_mixer = alc662_auto_capture_mixer,
Kailang Yangf1d4e282008-08-26 14:03:29 +020017010 .init_verbs = { alc662_init_verbs,
17011 alc663_15jd_amic_init_verbs },
17012 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
17013 .hp_nid = 0x03,
17014 .dac_nids = alc662_dac_nids,
17015 .dig_out_nid = ALC662_DIGOUT_NID,
17016 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
17017 .channel_mode = alc662_3ST_2ch_modes,
Kailang Yangf1d4e282008-08-26 14:03:29 +020017018 .unsol_event = alc663_mode5_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020017019 .setup = alc663_mode5_setup,
Kailang Yangf1d4e282008-08-26 14:03:29 +020017020 .init_hook = alc663_mode5_inithook,
17021 },
17022 [ALC663_ASUS_MODE6] = {
Takashi Iwaif9e336f2008-10-31 16:37:07 +010017023 .mixers = { alc663_two_hp_m2_mixer },
17024 .cap_mixer = alc662_auto_capture_mixer,
Kailang Yangf1d4e282008-08-26 14:03:29 +020017025 .init_verbs = { alc662_init_verbs,
17026 alc663_two_hp_amic_m2_init_verbs },
17027 .num_dacs = ARRAY_SIZE(alc662_dac_nids),
17028 .hp_nid = 0x03,
17029 .dac_nids = alc662_dac_nids,
17030 .dig_out_nid = ALC662_DIGOUT_NID,
17031 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
17032 .channel_mode = alc662_3ST_2ch_modes,
Kailang Yangf1d4e282008-08-26 14:03:29 +020017033 .unsol_event = alc663_mode6_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020017034 .setup = alc663_mode6_setup,
Kailang Yangf1d4e282008-08-26 14:03:29 +020017035 .init_hook = alc663_mode6_inithook,
17036 },
Kailang Yang622e84c2009-04-21 07:39:04 +020017037 [ALC272_DELL] = {
17038 .mixers = { alc663_m51va_mixer },
17039 .cap_mixer = alc272_auto_capture_mixer,
17040 .init_verbs = { alc662_init_verbs, alc272_dell_init_verbs },
17041 .num_dacs = ARRAY_SIZE(alc272_dac_nids),
17042 .dac_nids = alc662_dac_nids,
17043 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
17044 .adc_nids = alc272_adc_nids,
17045 .num_adc_nids = ARRAY_SIZE(alc272_adc_nids),
17046 .capsrc_nids = alc272_capsrc_nids,
17047 .channel_mode = alc662_3ST_2ch_modes,
Kailang Yang622e84c2009-04-21 07:39:04 +020017048 .unsol_event = alc663_m51va_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020017049 .setup = alc663_m51va_setup,
Kailang Yang622e84c2009-04-21 07:39:04 +020017050 .init_hook = alc663_m51va_inithook,
17051 },
17052 [ALC272_DELL_ZM1] = {
17053 .mixers = { alc663_m51va_mixer },
17054 .cap_mixer = alc662_auto_capture_mixer,
17055 .init_verbs = { alc662_init_verbs, alc272_dell_zm1_init_verbs },
17056 .num_dacs = ARRAY_SIZE(alc272_dac_nids),
17057 .dac_nids = alc662_dac_nids,
17058 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
17059 .adc_nids = alc662_adc_nids,
Takashi Iwaib59bdf32009-08-11 09:47:30 +020017060 .num_adc_nids = 1,
Kailang Yang622e84c2009-04-21 07:39:04 +020017061 .capsrc_nids = alc662_capsrc_nids,
17062 .channel_mode = alc662_3ST_2ch_modes,
Kailang Yang622e84c2009-04-21 07:39:04 +020017063 .unsol_event = alc663_m51va_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020017064 .setup = alc663_m51va_setup,
Kailang Yang622e84c2009-04-21 07:39:04 +020017065 .init_hook = alc663_m51va_inithook,
17066 },
Chris Pockelé9541ba12009-05-12 08:08:53 +020017067 [ALC272_SAMSUNG_NC10] = {
17068 .mixers = { alc272_nc10_mixer },
17069 .init_verbs = { alc662_init_verbs,
17070 alc663_21jd_amic_init_verbs },
17071 .num_dacs = ARRAY_SIZE(alc272_dac_nids),
17072 .dac_nids = alc272_dac_nids,
17073 .num_channel_mode = ARRAY_SIZE(alc662_3ST_2ch_modes),
17074 .channel_mode = alc662_3ST_2ch_modes,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020017075 /*.input_mux = &alc272_nc10_capture_source,*/
Chris Pockelé9541ba12009-05-12 08:08:53 +020017076 .unsol_event = alc663_mode4_unsol_event,
Takashi Iwai4f5d17062009-08-11 18:17:46 +020017077 .setup = alc663_mode4_setup,
Chris Pockelé9541ba12009-05-12 08:08:53 +020017078 .init_hook = alc663_mode4_inithook,
17079 },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017080};
17081
17082
17083/*
17084 * BIOS auto configuration
17085 */
17086
Takashi Iwai7085ec12009-10-02 09:03:58 +020017087/* convert from MIX nid to DAC */
17088static inline hda_nid_t alc662_mix_to_dac(hda_nid_t nid)
17089{
17090 if (nid == 0x0f)
17091 return 0x02;
17092 else if (nid >= 0x0c && nid <= 0x0e)
17093 return nid - 0x0c + 0x02;
17094 else
17095 return 0;
17096}
17097
17098/* get MIX nid connected to the given pin targeted to DAC */
17099static hda_nid_t alc662_dac_to_mix(struct hda_codec *codec, hda_nid_t pin,
17100 hda_nid_t dac)
17101{
17102 hda_nid_t mix[4];
17103 int i, num;
17104
17105 num = snd_hda_get_connections(codec, pin, mix, ARRAY_SIZE(mix));
17106 for (i = 0; i < num; i++) {
17107 if (alc662_mix_to_dac(mix[i]) == dac)
17108 return mix[i];
17109 }
17110 return 0;
17111}
17112
17113/* look for an empty DAC slot */
17114static hda_nid_t alc662_look_for_dac(struct hda_codec *codec, hda_nid_t pin)
17115{
17116 struct alc_spec *spec = codec->spec;
17117 hda_nid_t srcs[5];
17118 int i, j, num;
17119
17120 num = snd_hda_get_connections(codec, pin, srcs, ARRAY_SIZE(srcs));
17121 if (num < 0)
17122 return 0;
17123 for (i = 0; i < num; i++) {
17124 hda_nid_t nid = alc662_mix_to_dac(srcs[i]);
17125 if (!nid)
17126 continue;
17127 for (j = 0; j < spec->multiout.num_dacs; j++)
17128 if (spec->multiout.dac_nids[j] == nid)
17129 break;
17130 if (j >= spec->multiout.num_dacs)
17131 return nid;
17132 }
17133 return 0;
17134}
17135
17136/* fill in the dac_nids table from the parsed pin configuration */
17137static int alc662_auto_fill_dac_nids(struct hda_codec *codec,
17138 const struct auto_pin_cfg *cfg)
17139{
17140 struct alc_spec *spec = codec->spec;
17141 int i;
17142 hda_nid_t dac;
17143
17144 spec->multiout.dac_nids = spec->private_dac_nids;
17145 for (i = 0; i < cfg->line_outs; i++) {
17146 dac = alc662_look_for_dac(codec, cfg->line_out_pins[i]);
17147 if (!dac)
17148 continue;
17149 spec->multiout.dac_nids[spec->multiout.num_dacs++] = dac;
17150 }
17151 return 0;
17152}
17153
Takashi Iwai0afe5f82009-10-02 09:20:00 +020017154static inline int alc662_add_vol_ctl(struct alc_spec *spec, const char *pfx,
Takashi Iwai7085ec12009-10-02 09:03:58 +020017155 hda_nid_t nid, unsigned int chs)
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017156{
Takashi Iwai0afe5f82009-10-02 09:20:00 +020017157 return add_pb_vol_ctrl(spec, ALC_CTL_WIDGET_VOL, pfx,
Takashi Iwai7085ec12009-10-02 09:03:58 +020017158 HDA_COMPOSE_AMP_VAL(nid, chs, 0, HDA_OUTPUT));
17159}
17160
Takashi Iwai0afe5f82009-10-02 09:20:00 +020017161static inline int alc662_add_sw_ctl(struct alc_spec *spec, const char *pfx,
Takashi Iwai7085ec12009-10-02 09:03:58 +020017162 hda_nid_t nid, unsigned int chs)
17163{
Takashi Iwai0afe5f82009-10-02 09:20:00 +020017164 return add_pb_sw_ctrl(spec, ALC_CTL_WIDGET_MUTE, pfx,
Takashi Iwai7085ec12009-10-02 09:03:58 +020017165 HDA_COMPOSE_AMP_VAL(nid, chs, 0, HDA_INPUT));
17166}
17167
17168#define alc662_add_stereo_vol(spec, pfx, nid) \
17169 alc662_add_vol_ctl(spec, pfx, nid, 3)
17170#define alc662_add_stereo_sw(spec, pfx, nid) \
17171 alc662_add_sw_ctl(spec, pfx, nid, 3)
17172
17173/* add playback controls from the parsed DAC table */
17174static int alc662_auto_create_multi_out_ctls(struct hda_codec *codec,
17175 const struct auto_pin_cfg *cfg)
17176{
17177 struct alc_spec *spec = codec->spec;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017178 static const char *chname[4] = {
17179 "Front", "Surround", NULL /*CLFE*/, "Side"
17180 };
Takashi Iwai7085ec12009-10-02 09:03:58 +020017181 hda_nid_t nid, mix;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017182 int i, err;
17183
17184 for (i = 0; i < cfg->line_outs; i++) {
Takashi Iwai7085ec12009-10-02 09:03:58 +020017185 nid = spec->multiout.dac_nids[i];
17186 if (!nid)
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017187 continue;
Takashi Iwai7085ec12009-10-02 09:03:58 +020017188 mix = alc662_dac_to_mix(codec, cfg->line_out_pins[i], nid);
17189 if (!mix)
17190 continue;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017191 if (i == 2) {
17192 /* Center/LFE */
Takashi Iwai7085ec12009-10-02 09:03:58 +020017193 err = alc662_add_vol_ctl(spec, "Center", nid, 1);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017194 if (err < 0)
17195 return err;
Takashi Iwai7085ec12009-10-02 09:03:58 +020017196 err = alc662_add_vol_ctl(spec, "LFE", nid, 2);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017197 if (err < 0)
17198 return err;
Takashi Iwai7085ec12009-10-02 09:03:58 +020017199 err = alc662_add_sw_ctl(spec, "Center", mix, 1);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017200 if (err < 0)
17201 return err;
Takashi Iwai7085ec12009-10-02 09:03:58 +020017202 err = alc662_add_sw_ctl(spec, "LFE", mix, 2);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017203 if (err < 0)
17204 return err;
17205 } else {
Takashi Iwai0d884cb2009-08-25 16:14:35 +020017206 const char *pfx;
17207 if (cfg->line_outs == 1 &&
17208 cfg->line_out_type == AUTO_PIN_SPEAKER_OUT) {
Takashi Iwai7085ec12009-10-02 09:03:58 +020017209 if (cfg->hp_outs)
Takashi Iwai0d884cb2009-08-25 16:14:35 +020017210 pfx = "Speaker";
17211 else
17212 pfx = "PCM";
17213 } else
17214 pfx = chname[i];
Takashi Iwai7085ec12009-10-02 09:03:58 +020017215 err = alc662_add_vol_ctl(spec, pfx, nid, 3);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017216 if (err < 0)
17217 return err;
Takashi Iwai0d884cb2009-08-25 16:14:35 +020017218 if (cfg->line_outs == 1 &&
17219 cfg->line_out_type == AUTO_PIN_SPEAKER_OUT)
17220 pfx = "Speaker";
Takashi Iwai7085ec12009-10-02 09:03:58 +020017221 err = alc662_add_sw_ctl(spec, pfx, mix, 3);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017222 if (err < 0)
17223 return err;
17224 }
17225 }
17226 return 0;
17227}
17228
17229/* add playback controls for speaker and HP outputs */
Takashi Iwai7085ec12009-10-02 09:03:58 +020017230/* return DAC nid if any new DAC is assigned */
17231static int alc662_auto_create_extra_out(struct hda_codec *codec, hda_nid_t pin,
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017232 const char *pfx)
17233{
Takashi Iwai7085ec12009-10-02 09:03:58 +020017234 struct alc_spec *spec = codec->spec;
17235 hda_nid_t nid, mix;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017236 int err;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017237
17238 if (!pin)
17239 return 0;
Takashi Iwai7085ec12009-10-02 09:03:58 +020017240 nid = alc662_look_for_dac(codec, pin);
17241 if (!nid) {
Takashi Iwai7085ec12009-10-02 09:03:58 +020017242 /* the corresponding DAC is already occupied */
17243 if (!(get_wcaps(codec, pin) & AC_WCAP_OUT_AMP))
17244 return 0; /* no way */
17245 /* create a switch only */
Takashi Iwai0afe5f82009-10-02 09:20:00 +020017246 return add_pb_sw_ctrl(spec, ALC_CTL_WIDGET_MUTE, pfx,
Takashi Iwai7085ec12009-10-02 09:03:58 +020017247 HDA_COMPOSE_AMP_VAL(pin, 3, 0, HDA_OUTPUT));
17248 }
17249
17250 mix = alc662_dac_to_mix(codec, pin, nid);
17251 if (!mix)
17252 return 0;
17253 err = alc662_add_vol_ctl(spec, pfx, nid, 3);
17254 if (err < 0)
Takashi Iwai24fb9172008-09-02 14:48:20 +020017255 return err;
Takashi Iwai7085ec12009-10-02 09:03:58 +020017256 err = alc662_add_sw_ctl(spec, pfx, mix, 3);
17257 if (err < 0)
17258 return err;
17259 return nid;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017260}
17261
17262/* create playback/capture controls for input pins */
Takashi Iwai05f5f472009-08-25 13:10:18 +020017263#define alc662_auto_create_input_ctls \
Takashi Iwai4b7348a2009-10-14 18:25:23 +020017264 alc882_auto_create_input_ctls
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017265
17266static void alc662_auto_set_output_and_unmute(struct hda_codec *codec,
17267 hda_nid_t nid, int pin_type,
Takashi Iwai7085ec12009-10-02 09:03:58 +020017268 hda_nid_t dac)
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017269{
Takashi Iwai7085ec12009-10-02 09:03:58 +020017270 int i, num;
17271 hda_nid_t srcs[4];
17272
Takashi Iwaif6c7e542008-02-12 18:32:23 +010017273 alc_set_pin_output(codec, nid, pin_type);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017274 /* need the manual connection? */
Takashi Iwai7085ec12009-10-02 09:03:58 +020017275 num = snd_hda_get_connections(codec, nid, srcs, ARRAY_SIZE(srcs));
17276 if (num <= 1)
17277 return;
17278 for (i = 0; i < num; i++) {
17279 if (alc662_mix_to_dac(srcs[i]) != dac)
17280 continue;
17281 snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_CONNECT_SEL, i);
17282 return;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017283 }
17284}
17285
17286static void alc662_auto_init_multi_out(struct hda_codec *codec)
17287{
17288 struct alc_spec *spec = codec->spec;
Takashi Iwai7085ec12009-10-02 09:03:58 +020017289 int pin_type = get_pin_type(spec->autocfg.line_out_type);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017290 int i;
17291
17292 for (i = 0; i <= HDA_SIDE; i++) {
17293 hda_nid_t nid = spec->autocfg.line_out_pins[i];
17294 if (nid)
Takashi Iwaibaba8ee2007-04-23 17:17:48 +020017295 alc662_auto_set_output_and_unmute(codec, nid, pin_type,
Takashi Iwai7085ec12009-10-02 09:03:58 +020017296 spec->multiout.dac_nids[i]);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017297 }
17298}
17299
17300static void alc662_auto_init_hp_out(struct hda_codec *codec)
17301{
17302 struct alc_spec *spec = codec->spec;
17303 hda_nid_t pin;
17304
17305 pin = spec->autocfg.hp_pins[0];
Takashi Iwai7085ec12009-10-02 09:03:58 +020017306 if (pin)
17307 alc662_auto_set_output_and_unmute(codec, pin, PIN_HP,
17308 spec->multiout.hp_nid);
Takashi Iwaif6c7e542008-02-12 18:32:23 +010017309 pin = spec->autocfg.speaker_pins[0];
17310 if (pin)
Takashi Iwai7085ec12009-10-02 09:03:58 +020017311 alc662_auto_set_output_and_unmute(codec, pin, PIN_OUT,
17312 spec->multiout.extra_out_nid[0]);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017313}
17314
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017315#define ALC662_PIN_CD_NID ALC880_PIN_CD_NID
17316
17317static void alc662_auto_init_analog_input(struct hda_codec *codec)
17318{
17319 struct alc_spec *spec = codec->spec;
17320 int i;
17321
17322 for (i = 0; i < AUTO_PIN_LAST; i++) {
17323 hda_nid_t nid = spec->autocfg.input_pins[i];
Takashi Iwai05f5f472009-08-25 13:10:18 +020017324 if (alc_is_input_pin(codec, nid)) {
Takashi Iwai23f0c042009-02-26 13:03:58 +010017325 alc_set_input_pin(codec, nid, i);
Takashi Iwai52ca15b2009-03-23 12:51:55 +010017326 if (nid != ALC662_PIN_CD_NID &&
Takashi Iwaie82c0252009-03-23 15:17:38 +010017327 (get_wcaps(codec, nid) & AC_WCAP_OUT_AMP))
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017328 snd_hda_codec_write(codec, nid, 0,
17329 AC_VERB_SET_AMP_GAIN_MUTE,
17330 AMP_OUT_MUTE);
17331 }
17332 }
17333}
17334
Takashi Iwaif511b012008-08-15 16:46:42 +020017335#define alc662_auto_init_input_src alc882_auto_init_input_src
17336
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017337static int alc662_parse_auto_config(struct hda_codec *codec)
17338{
17339 struct alc_spec *spec = codec->spec;
17340 int err;
17341 static hda_nid_t alc662_ignore[] = { 0x1d, 0 };
17342
17343 err = snd_hda_parse_pin_def_config(codec, &spec->autocfg,
17344 alc662_ignore);
17345 if (err < 0)
17346 return err;
17347 if (!spec->autocfg.line_outs)
17348 return 0; /* can't find valid BIOS pin config */
17349
Takashi Iwai7085ec12009-10-02 09:03:58 +020017350 err = alc662_auto_fill_dac_nids(codec, &spec->autocfg);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020017351 if (err < 0)
17352 return err;
Takashi Iwai7085ec12009-10-02 09:03:58 +020017353 err = alc662_auto_create_multi_out_ctls(codec, &spec->autocfg);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020017354 if (err < 0)
17355 return err;
Takashi Iwai7085ec12009-10-02 09:03:58 +020017356 err = alc662_auto_create_extra_out(codec,
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020017357 spec->autocfg.speaker_pins[0],
17358 "Speaker");
17359 if (err < 0)
17360 return err;
Takashi Iwai7085ec12009-10-02 09:03:58 +020017361 if (err)
17362 spec->multiout.extra_out_nid[0] = err;
17363 err = alc662_auto_create_extra_out(codec, spec->autocfg.hp_pins[0],
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020017364 "Headphone");
17365 if (err < 0)
17366 return err;
Takashi Iwai7085ec12009-10-02 09:03:58 +020017367 if (err)
17368 spec->multiout.hp_nid = err;
Takashi Iwai05f5f472009-08-25 13:10:18 +020017369 err = alc662_auto_create_input_ctls(codec, &spec->autocfg);
Takashi Iwaif12ab1e2007-04-12 15:51:47 +020017370 if (err < 0)
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017371 return err;
17372
17373 spec->multiout.max_channels = spec->multiout.num_dacs * 2;
17374
Takashi Iwai0852d7a2009-02-11 11:35:15 +010017375 if (spec->autocfg.dig_outs)
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017376 spec->multiout.dig_out_nid = ALC880_DIGOUT_NID;
17377
Takashi Iwai603c4012008-07-30 15:01:44 +020017378 if (spec->kctls.list)
Takashi Iwaid88897e2008-10-31 15:01:37 +010017379 add_mixer(spec, spec->kctls.list);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017380
17381 spec->num_mux_defs = 1;
Herton Ronaldo Krzesinski61b9b9b2009-01-28 09:16:33 -020017382 spec->input_mux = &spec->private_imux[0];
Kailang Yangea1fb292008-08-26 12:58:38 +020017383
Takashi Iwaid88897e2008-10-31 15:01:37 +010017384 add_verb(spec, alc662_auto_init_verbs);
Takashi Iwai24fb9172008-09-02 14:48:20 +020017385 if (codec->vendor_id == 0x10ec0663)
Takashi Iwaid88897e2008-10-31 15:01:37 +010017386 add_verb(spec, alc663_auto_init_verbs);
Takashi Iwaiee979a142008-09-02 15:42:20 +020017387
17388 err = alc_auto_add_mic_boost(codec);
17389 if (err < 0)
17390 return err;
17391
Takashi Iwai4a79ba32009-04-22 16:31:35 +020017392 alc_ssid_check(codec, 0x15, 0x1b, 0x14);
17393
Takashi Iwai8c872862007-06-19 12:11:16 +020017394 return 1;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017395}
17396
17397/* additional initialization for auto-configuration model */
17398static void alc662_auto_init(struct hda_codec *codec)
17399{
Takashi Iwaif6c7e542008-02-12 18:32:23 +010017400 struct alc_spec *spec = codec->spec;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017401 alc662_auto_init_multi_out(codec);
17402 alc662_auto_init_hp_out(codec);
17403 alc662_auto_init_analog_input(codec);
Takashi Iwaif511b012008-08-15 16:46:42 +020017404 alc662_auto_init_input_src(codec);
Takashi Iwaif6c7e542008-02-12 18:32:23 +010017405 if (spec->unsol_event)
Kailang Yang7fb0d782008-10-15 11:12:35 +020017406 alc_inithook(codec);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017407}
17408
17409static int patch_alc662(struct hda_codec *codec)
17410{
17411 struct alc_spec *spec;
17412 int err, board_config;
17413
17414 spec = kzalloc(sizeof(*spec), GFP_KERNEL);
17415 if (!spec)
17416 return -ENOMEM;
17417
17418 codec->spec = spec;
17419
Takashi Iwai2c3bf9a2008-06-04 12:39:38 +020017420 alc_fix_pll_init(codec, 0x20, 0x04, 15);
17421
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017422 board_config = snd_hda_check_board_config(codec, ALC662_MODEL_LAST,
17423 alc662_models,
17424 alc662_cfg_tbl);
17425 if (board_config < 0) {
Takashi Iwai9a11f1a2009-07-28 16:01:20 +020017426 printk(KERN_INFO "hda_codec: %s: BIOS auto-probing.\n",
17427 codec->chip_name);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017428 board_config = ALC662_AUTO;
17429 }
17430
17431 if (board_config == ALC662_AUTO) {
17432 /* automatic parse from the BIOS config */
17433 err = alc662_parse_auto_config(codec);
17434 if (err < 0) {
17435 alc_free(codec);
17436 return err;
Takashi Iwai8c872862007-06-19 12:11:16 +020017437 } else if (!err) {
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017438 printk(KERN_INFO
17439 "hda_codec: Cannot set up configuration "
17440 "from BIOS. Using base mode...\n");
17441 board_config = ALC662_3ST_2ch_DIG;
17442 }
17443 }
17444
Kusanagi Kouichi680cd532009-02-05 00:00:58 +090017445 err = snd_hda_attach_beep_device(codec, 0x1);
17446 if (err < 0) {
17447 alc_free(codec);
17448 return err;
17449 }
17450
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017451 if (board_config != ALC662_AUTO)
Takashi Iwaie9c364c2009-08-11 17:16:13 +020017452 setup_preset(codec, &alc662_presets[board_config]);
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017453
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017454 spec->stream_analog_playback = &alc662_pcm_analog_playback;
17455 spec->stream_analog_capture = &alc662_pcm_analog_capture;
17456
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017457 spec->stream_digital_playback = &alc662_pcm_digital_playback;
17458 spec->stream_digital_capture = &alc662_pcm_digital_capture;
17459
Takashi Iwaidd704692009-08-11 08:45:11 +020017460 if (!spec->adc_nids) {
17461 spec->adc_nids = alc662_adc_nids;
17462 spec->num_adc_nids = ARRAY_SIZE(alc662_adc_nids);
17463 }
17464 if (!spec->capsrc_nids)
17465 spec->capsrc_nids = alc662_capsrc_nids;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017466
Takashi Iwaif9e336f2008-10-31 16:37:07 +010017467 if (!spec->cap_mixer)
Takashi Iwaib59bdf32009-08-11 09:47:30 +020017468 set_capture_mixer(codec);
Takashi Iwaib9591442009-03-16 15:25:00 +010017469 if (codec->vendor_id == 0x10ec0662)
17470 set_beep_amp(spec, 0x0b, 0x05, HDA_INPUT);
17471 else
17472 set_beep_amp(spec, 0x0b, 0x04, HDA_INPUT);
Takashi Iwaif9e336f2008-10-31 16:37:07 +010017473
Takashi Iwai2134ea42008-01-10 16:53:55 +010017474 spec->vmaster_nid = 0x02;
17475
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017476 codec->patch_ops = alc_patch_ops;
17477 if (board_config == ALC662_AUTO)
17478 spec->init_hook = alc662_auto_init;
Takashi Iwaicb53c622007-08-10 17:21:45 +020017479#ifdef CONFIG_SND_HDA_POWER_SAVE
17480 if (!spec->loopback.amplist)
17481 spec->loopback.amplist = alc662_loopbacks;
17482#endif
Takashi Iwaidaead532008-11-28 12:55:36 +010017483 codec->proc_widget_hook = print_realtek_coef;
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017484
17485 return 0;
17486}
17487
17488/*
Linus Torvalds1da177e2005-04-16 15:20:36 -070017489 * patch entries
17490 */
Takashi Iwai1289e9e2008-11-27 15:47:11 +010017491static struct hda_codec_preset snd_hda_preset_realtek[] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -070017492 { .id = 0x10ec0260, .name = "ALC260", .patch = patch_alc260 },
Kailang Yangdf694da2005-12-05 19:42:22 +010017493 { .id = 0x10ec0262, .name = "ALC262", .patch = patch_alc262 },
Kailang Yangf6a92242007-12-13 16:52:54 +010017494 { .id = 0x10ec0267, .name = "ALC267", .patch = patch_alc268 },
Kailang Yanga361d842007-06-05 12:30:55 +020017495 { .id = 0x10ec0268, .name = "ALC268", .patch = patch_alc268 },
Kailang Yangf6a92242007-12-13 16:52:54 +010017496 { .id = 0x10ec0269, .name = "ALC269", .patch = patch_alc269 },
Kailang Yang01afd412008-10-15 11:22:09 +020017497 { .id = 0x10ec0272, .name = "ALC272", .patch = patch_alc662 },
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010017498 { .id = 0x10ec0861, .rev = 0x100340, .name = "ALC660",
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017499 .patch = patch_alc861 },
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010017500 { .id = 0x10ec0660, .name = "ALC660-VD", .patch = patch_alc861vd },
17501 { .id = 0x10ec0861, .name = "ALC861", .patch = patch_alc861 },
17502 { .id = 0x10ec0862, .name = "ALC861-VD", .patch = patch_alc861vd },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017503 { .id = 0x10ec0662, .rev = 0x100002, .name = "ALC662 rev2",
Takashi Iwai49535502009-06-30 15:28:30 +020017504 .patch = patch_alc882 },
Kailang Yangbc9f98a2007-04-12 13:06:07 +020017505 { .id = 0x10ec0662, .rev = 0x100101, .name = "ALC662 rev1",
17506 .patch = patch_alc662 },
Kailang Yang6dda9f42008-05-27 12:05:31 +020017507 { .id = 0x10ec0663, .name = "ALC663", .patch = patch_alc662 },
Jakub Schmidtkef32610e2007-02-02 18:17:27 +010017508 { .id = 0x10ec0880, .name = "ALC880", .patch = patch_alc880 },
Linus Torvalds1da177e2005-04-16 15:20:36 -070017509 { .id = 0x10ec0882, .name = "ALC882", .patch = patch_alc882 },
Takashi Iwai49535502009-06-30 15:28:30 +020017510 { .id = 0x10ec0883, .name = "ALC883", .patch = patch_alc882 },
Clive Messer669faba2008-09-30 15:49:13 +020017511 { .id = 0x10ec0885, .rev = 0x100101, .name = "ALC889A",
Takashi Iwai49535502009-06-30 15:28:30 +020017512 .patch = patch_alc882 },
Takashi Iwaicb308f92008-04-16 14:13:29 +020017513 { .id = 0x10ec0885, .rev = 0x100103, .name = "ALC889A",
Takashi Iwai49535502009-06-30 15:28:30 +020017514 .patch = patch_alc882 },
Kailang Yangdf694da2005-12-05 19:42:22 +010017515 { .id = 0x10ec0885, .name = "ALC885", .patch = patch_alc882 },
Takashi Iwai49535502009-06-30 15:28:30 +020017516 { .id = 0x10ec0887, .name = "ALC887", .patch = patch_alc882 },
Kailang Yang44426082008-10-15 11:18:05 +020017517 { .id = 0x10ec0888, .rev = 0x100101, .name = "ALC1200",
Takashi Iwai49535502009-06-30 15:28:30 +020017518 .patch = patch_alc882 },
17519 { .id = 0x10ec0888, .name = "ALC888", .patch = patch_alc882 },
17520 { .id = 0x10ec0889, .name = "ALC889", .patch = patch_alc882 },
Linus Torvalds1da177e2005-04-16 15:20:36 -070017521 {} /* terminator */
17522};
Takashi Iwai1289e9e2008-11-27 15:47:11 +010017523
17524MODULE_ALIAS("snd-hda-codec-id:10ec*");
17525
17526MODULE_LICENSE("GPL");
17527MODULE_DESCRIPTION("Realtek HD-audio codec");
17528
17529static struct hda_codec_preset_list realtek_list = {
17530 .preset = snd_hda_preset_realtek,
17531 .owner = THIS_MODULE,
17532};
17533
17534static int __init patch_realtek_init(void)
17535{
17536 return snd_hda_add_codec_preset(&realtek_list);
17537}
17538
17539static void __exit patch_realtek_exit(void)
17540{
17541 snd_hda_delete_codec_preset(&realtek_list);
17542}
17543
17544module_init(patch_realtek_init)
17545module_exit(patch_realtek_exit)