]> git.karo-electronics.de Git - karo-tx-linux.git/blobdiff - sound/pci/hda/patch_conexant.c
Merge branch 'common/fbdev-meram' of git://git.kernel.org/pub/scm/linux/kernel/git...
[karo-tx-linux.git] / sound / pci / hda / patch_conexant.c
index fb759bff91a99345184a5861fc56e5aa5ba6a388..4f37477d3c71b5238b049275987cca2656df0107 100644 (file)
@@ -39,6 +39,7 @@
 
 #define CONEXANT_HP_EVENT      0x37
 #define CONEXANT_MIC_EVENT     0x38
+#define CONEXANT_LINE_EVENT    0x39
 
 /* Conexant 5051 specific */
 
@@ -55,6 +56,13 @@ struct pin_dac_pair {
        int type;
 };
 
+struct imux_info {
+       hda_nid_t pin;          /* input pin NID */
+       hda_nid_t adc;          /* connected ADC NID */ 
+       hda_nid_t boost;        /* optional boost volume NID */
+       int index;              /* corresponding to autocfg.input */
+};
+
 struct conexant_spec {
 
        const struct snd_kcontrol_new *mixers[5];
@@ -74,8 +82,11 @@ struct conexant_spec {
                                         */
        unsigned int cur_eapd;
        unsigned int hp_present;
+       unsigned int line_present;
        unsigned int auto_mic;
        int auto_mic_ext;               /* imux_pins[] index for ext mic */
+       int auto_mic_dock;              /* imux_pins[] index for dock mic */
+       int auto_mic_int;               /* imux_pins[] index for int mic */
        unsigned int need_dac_fix;
        hda_nid_t slave_dig_outs[2];
 
@@ -108,8 +119,7 @@ struct conexant_spec {
        /* dynamic controls, init_verbs and input_mux */
        struct auto_pin_cfg autocfg;
        struct hda_input_mux private_imux;
-       hda_nid_t imux_adcs[HDA_MAX_NUM_INPUTS];
-       hda_nid_t imux_pins[HDA_MAX_NUM_INPUTS];
+       struct imux_info imux_info[HDA_MAX_NUM_INPUTS];
        hda_nid_t private_adc_nids[HDA_MAX_NUM_INPUTS];
        hda_nid_t private_dac_nids[AUTO_CFG_MAX_OUTS];
        struct pin_dac_pair dac_info[8];
@@ -117,6 +127,9 @@ struct conexant_spec {
 
        unsigned int port_d_mode;
        unsigned int auto_mute:1;       /* used in auto-parser */
+       unsigned int detect_line:1;     /* Line-out detection enabled */
+       unsigned int automute_lines:1;  /* automute line-out as well */
+       unsigned int automute_hp_lo:1;  /* both HP and LO available */
        unsigned int dell_automute:1;
        unsigned int dell_vostro:1;
        unsigned int ideapad:1;
@@ -469,12 +482,18 @@ static const struct snd_kcontrol_new cxt_beep_mixer[] = {
 static const char * const slave_vols[] = {
        "Headphone Playback Volume",
        "Speaker Playback Volume",
+       "Front Playback Volume",
+       "Surround Playback Volume",
+       "CLFE Playback Volume",
        NULL
 };
 
 static const char * const slave_sws[] = {
        "Headphone Playback Switch",
        "Speaker Playback Switch",
+       "Front Playback Switch",
+       "Surround Playback Switch",
+       "CLFE Playback Switch",
        NULL
 };
 
@@ -1104,8 +1123,10 @@ static int patch_cxt5045(struct hda_codec *codec)
        board_config = snd_hda_check_board_config(codec, CXT5045_MODELS,
                                                  cxt5045_models,
                                                  cxt5045_cfg_tbl);
+#if 0 /* use the old method just for safety */
        if (board_config < 0)
                board_config = CXT5045_AUTO;
+#endif
        if (board_config == CXT5045_AUTO)
                return patch_conexant_auto(codec);
 
@@ -1972,8 +1993,10 @@ static int patch_cxt5051(struct hda_codec *codec)
        board_config = snd_hda_check_board_config(codec, CXT5051_MODELS,
                                                  cxt5051_models,
                                                  cxt5051_cfg_tbl);
+#if 0 /* use the old method just for safety */
        if (board_config < 0)
                board_config = CXT5051_AUTO;
+#endif
        if (board_config == CXT5051_AUTO)
                return patch_conexant_auto(codec);
 
@@ -3034,6 +3057,7 @@ enum {
        CXT5066_THINKPAD,       /* Lenovo ThinkPad T410s, others? */
        CXT5066_ASUS,           /* Asus K52JU, Lenovo G560 - Int mic at 0x1a and Ext mic at 0x1b */
        CXT5066_HP_LAPTOP,      /* HP Laptop */
+       CXT5066_AUTO,           /* BIOS auto-parser */
        CXT5066_MODELS
 };
 
@@ -3046,6 +3070,7 @@ static const char * const cxt5066_models[CXT5066_MODELS] = {
        [CXT5066_THINKPAD]      = "thinkpad",
        [CXT5066_ASUS]          = "asus",
        [CXT5066_HP_LAPTOP]     = "hp-laptop",
+       [CXT5066_AUTO]          = "auto",
 };
 
 static const struct snd_pci_quirk cxt5066_cfg_tbl[] = {
@@ -3083,6 +3108,15 @@ static int patch_cxt5066(struct hda_codec *codec)
        struct conexant_spec *spec;
        int board_config;
 
+       board_config = snd_hda_check_board_config(codec, CXT5066_MODELS,
+                                                 cxt5066_models, cxt5066_cfg_tbl);
+#if 0 /* use the old method just for safety */
+       if (board_config < 0)
+               board_config = CXT5066_AUTO;
+#endif
+       if (board_config == CXT5066_AUTO)
+               return patch_conexant_auto(codec);
+
        spec = kzalloc(sizeof(*spec), GFP_KERNEL);
        if (!spec)
                return -ENOMEM;
@@ -3113,8 +3147,6 @@ static int patch_cxt5066(struct hda_codec *codec)
 
        set_beep_amp(spec, 0x13, 0, HDA_OUTPUT);
 
-       board_config = snd_hda_check_board_config(codec, CXT5066_MODELS,
-                                                 cxt5066_models, cxt5066_cfg_tbl);
        switch (board_config) {
        default:
        case CXT5066_LAPTOP:
@@ -3239,7 +3271,7 @@ static int cx_auto_capture_pcm_prepare(struct hda_pcm_stream *hinfo,
                                       struct snd_pcm_substream *substream)
 {
        struct conexant_spec *spec = codec->spec;
-       hda_nid_t adc = spec->imux_adcs[spec->cur_mux[0]];
+       hda_nid_t adc = spec->imux_info[spec->cur_mux[0]].adc;
        if (spec->adc_switching) {
                spec->cur_adc = adc;
                spec->cur_adc_stream_tag = stream_tag;
@@ -3395,48 +3427,193 @@ static void cx_auto_parse_output(struct hda_codec *codec)
        spec->multiout.dac_nids = spec->private_dac_nids;
        spec->multiout.max_channels = spec->multiout.num_dacs * 2;
 
-       if (cfg->hp_outs > 0)
-               spec->auto_mute = 1;
+       for (i = 0; i < cfg->hp_outs; i++) {
+               if (is_jack_detectable(codec, cfg->hp_pins[i])) {
+                       spec->auto_mute = 1;
+                       break;
+               }
+       }
+       if (spec->auto_mute && cfg->line_out_pins[0] &&
+           cfg->line_out_pins[0] != cfg->hp_pins[0] &&
+           cfg->line_out_pins[0] != cfg->speaker_pins[0]) {
+               for (i = 0; i < cfg->line_outs; i++) {
+                       if (is_jack_detectable(codec, cfg->line_out_pins[i])) {
+                               spec->detect_line = 1;
+                               break;
+                       }
+               }
+               spec->automute_lines = spec->detect_line;
+       }
+
        spec->vmaster_nid = spec->private_dac_nids[0];
 }
 
 static void cx_auto_turn_eapd(struct hda_codec *codec, int num_pins,
                              hda_nid_t *pins, bool on);
 
+static void do_automute(struct hda_codec *codec, int num_pins,
+                       hda_nid_t *pins, bool on)
+{
+       int i;
+       for (i = 0; i < num_pins; i++)
+               snd_hda_codec_write(codec, pins[i], 0,
+                                   AC_VERB_SET_PIN_WIDGET_CONTROL,
+                                   on ? PIN_OUT : 0);
+       cx_auto_turn_eapd(codec, num_pins, pins, on);
+}
+
+static int detect_jacks(struct hda_codec *codec, int num_pins, hda_nid_t *pins)
+{
+       int i, present = 0;
+
+       for (i = 0; i < num_pins; i++) {
+               hda_nid_t nid = pins[i];
+               if (!nid || !is_jack_detectable(codec, nid))
+                       break;
+               snd_hda_input_jack_report(codec, nid);
+               present |= snd_hda_jack_detect(codec, nid);
+       }
+       return present;
+}
+
 /* auto-mute/unmute speaker and line outs according to headphone jack */
+static void cx_auto_update_speakers(struct hda_codec *codec)
+{
+       struct conexant_spec *spec = codec->spec;
+       struct auto_pin_cfg *cfg = &spec->autocfg;
+       int on;
+
+       if (!spec->auto_mute)
+               on = 0;
+       else
+               on = spec->hp_present | spec->line_present;
+       cx_auto_turn_eapd(codec, cfg->hp_outs, cfg->hp_pins, on);
+       do_automute(codec, cfg->speaker_outs, cfg->speaker_pins, !on);
+
+       /* toggle line-out mutes if needed, too */
+       /* if LO is a copy of either HP or Speaker, don't need to handle it */
+       if (cfg->line_out_pins[0] == cfg->hp_pins[0] ||
+           cfg->line_out_pins[0] == cfg->speaker_pins[0])
+               return;
+       if (!spec->automute_lines || !spec->auto_mute)
+               on = 0;
+       else
+               on = spec->hp_present;
+       do_automute(codec, cfg->line_outs, cfg->line_out_pins, !on);
+}
+
 static void cx_auto_hp_automute(struct hda_codec *codec)
 {
        struct conexant_spec *spec = codec->spec;
        struct auto_pin_cfg *cfg = &spec->autocfg;
-       int i, present;
 
        if (!spec->auto_mute)
                return;
-       present = 0;
-       for (i = 0; i < cfg->hp_outs; i++) {
-               if (snd_hda_jack_detect(codec, cfg->hp_pins[i])) {
-                       present = 1;
-                       break;
-               }
-       }
-       cx_auto_turn_eapd(codec, cfg->hp_outs, cfg->hp_pins, present);
-       for (i = 0; i < cfg->line_outs; i++) {
-               snd_hda_codec_write(codec, cfg->line_out_pins[i], 0,
-                                   AC_VERB_SET_PIN_WIDGET_CONTROL,
-                                   present ? 0 : PIN_OUT);
+       spec->hp_present = detect_jacks(codec, cfg->hp_outs, cfg->hp_pins);
+       cx_auto_update_speakers(codec);
+}
+
+static void cx_auto_line_automute(struct hda_codec *codec)
+{
+       struct conexant_spec *spec = codec->spec;
+       struct auto_pin_cfg *cfg = &spec->autocfg;
+
+       if (!spec->auto_mute || !spec->detect_line)
+               return;
+       spec->line_present = detect_jacks(codec, cfg->line_outs,
+                                         cfg->line_out_pins);
+       cx_auto_update_speakers(codec);
+}
+
+static int cx_automute_mode_info(struct snd_kcontrol *kcontrol,
+                                struct snd_ctl_elem_info *uinfo)
+{
+       struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
+       struct conexant_spec *spec = codec->spec;
+       static const char * const texts2[] = {
+               "Disabled", "Enabled"
+       };
+       static const char * const texts3[] = {
+               "Disabled", "Speaker Only", "Line-Out+Speaker"
+       };
+       const char * const *texts;
+
+       uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
+       uinfo->count = 1;
+       if (spec->automute_hp_lo) {
+               uinfo->value.enumerated.items = 3;
+               texts = texts3;
+       } else {
+               uinfo->value.enumerated.items = 2;
+               texts = texts2;
        }
-       cx_auto_turn_eapd(codec, cfg->line_outs, cfg->line_out_pins, !present);
-       for (i = 0; !present && i < cfg->line_outs; i++)
-               if (snd_hda_jack_detect(codec, cfg->line_out_pins[i]))
-                       present = 1;
-       for (i = 0; i < cfg->speaker_outs; i++) {
-               snd_hda_codec_write(codec, cfg->speaker_pins[i], 0,
-                                   AC_VERB_SET_PIN_WIDGET_CONTROL,
-                                   present ? 0 : PIN_OUT);
+       if (uinfo->value.enumerated.item >= uinfo->value.enumerated.items)
+               uinfo->value.enumerated.item = uinfo->value.enumerated.items - 1;
+       strcpy(uinfo->value.enumerated.name,
+              texts[uinfo->value.enumerated.item]);
+       return 0;
+}
+
+static int cx_automute_mode_get(struct snd_kcontrol *kcontrol,
+                               struct snd_ctl_elem_value *ucontrol)
+{
+       struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
+       struct conexant_spec *spec = codec->spec;
+       unsigned int val;
+       if (!spec->auto_mute)
+               val = 0;
+       else if (!spec->automute_lines)
+               val = 1;
+       else
+               val = 2;
+       ucontrol->value.enumerated.item[0] = val;
+       return 0;
+}
+
+static int cx_automute_mode_put(struct snd_kcontrol *kcontrol,
+                               struct snd_ctl_elem_value *ucontrol)
+{
+       struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
+       struct conexant_spec *spec = codec->spec;
+
+       switch (ucontrol->value.enumerated.item[0]) {
+       case 0:
+               if (!spec->auto_mute)
+                       return 0;
+               spec->auto_mute = 0;
+               break;
+       case 1:
+               if (spec->auto_mute && !spec->automute_lines)
+                       return 0;
+               spec->auto_mute = 1;
+               spec->automute_lines = 0;
+               break;
+       case 2:
+               if (!spec->automute_hp_lo)
+                       return -EINVAL;
+               if (spec->auto_mute && spec->automute_lines)
+                       return 0;
+               spec->auto_mute = 1;
+               spec->automute_lines = 1;
+               break;
+       default:
+               return -EINVAL;
        }
-       cx_auto_turn_eapd(codec, cfg->speaker_outs, cfg->speaker_pins, !present);
+       cx_auto_update_speakers(codec);
+       return 1;
 }
 
+static const struct snd_kcontrol_new cx_automute_mode_enum[] = {
+       {
+               .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
+               .name = "Auto-Mute Mode",
+               .info = cx_automute_mode_info,
+               .get = cx_automute_mode_get,
+               .put = cx_automute_mode_put,
+       },
+       { }
+};
+
 static int cx_auto_mux_enum_info(struct snd_kcontrol *kcontrol,
                                 struct snd_ctl_elem_info *uinfo)
 {
@@ -3460,30 +3637,42 @@ static int cx_auto_mux_enum_get(struct snd_kcontrol *kcontrol,
  * if do_select is set, actually select the route.
  */
 static int __select_input_connection(struct hda_codec *codec, hda_nid_t mux,
-                                    hda_nid_t pin, bool do_select, int depth)
+                                    hda_nid_t pin, hda_nid_t *srcp,
+                                    bool do_select, int depth)
 {
        hda_nid_t conn[HDA_MAX_NUM_INPUTS];
        int i, nums;
 
+       switch (get_wcaps_type(get_wcaps(codec, mux))) {
+       case AC_WID_AUD_IN:
+       case AC_WID_AUD_SEL:
+       case AC_WID_AUD_MIX:
+               break;
+       default:
+               return -1;
+       }
+
        nums = snd_hda_get_connections(codec, mux, conn, ARRAY_SIZE(conn));
        for (i = 0; i < nums; i++)
                if (conn[i] == pin) {
                        if (do_select)
                                snd_hda_codec_write(codec, mux, 0,
                                                    AC_VERB_SET_CONNECT_SEL, i);
+                       if (srcp)
+                               *srcp = mux;
                        return i;
                }
        depth++;
        if (depth == 2)
                return -1;
        for (i = 0; i < nums; i++) {
-               int ret  = __select_input_connection(codec, conn[i], pin,
+               int ret  = __select_input_connection(codec, conn[i], pin, srcp,
                                                     do_select, depth);
                if (ret >= 0) {
                        if (do_select)
                                snd_hda_codec_write(codec, mux, 0,
                                                    AC_VERB_SET_CONNECT_SEL, i);
-                       return ret;
+                       return i;
                }
        }
        return -1;
@@ -3492,13 +3681,13 @@ static int __select_input_connection(struct hda_codec *codec, hda_nid_t mux,
 static void select_input_connection(struct hda_codec *codec, hda_nid_t mux,
                                   hda_nid_t pin)
 {
-       __select_input_connection(codec, mux, pin, true, 0);
+       __select_input_connection(codec, mux, pin, NULL, true, 0);
 }
 
 static int get_input_connection(struct hda_codec *codec, hda_nid_t mux,
                                hda_nid_t pin)
 {
-       return __select_input_connection(codec, mux, pin, false, 0);
+       return __select_input_connection(codec, mux, pin, NULL, false, 0);
 }
 
 static int cx_auto_mux_enum_update(struct hda_codec *codec,
@@ -3514,9 +3703,9 @@ static int cx_auto_mux_enum_update(struct hda_codec *codec,
                idx = imux->num_items - 1;
        if (spec->cur_mux[0] == idx)
                return 0;
-       adc = spec->imux_adcs[idx];
-       select_input_connection(codec, spec->imux_adcs[idx],
-                               spec->imux_pins[idx]);
+       adc = spec->imux_info[idx].adc;
+       select_input_connection(codec, spec->imux_info[idx].adc,
+                               spec->imux_info[idx].pin);
        if (spec->cur_adc && spec->cur_adc != adc) {
                /* stream is running, let's swap the current ADC */
                __snd_hda_codec_cleanup_stream(codec, spec->cur_adc, 1);
@@ -3550,18 +3739,27 @@ static const struct snd_kcontrol_new cx_auto_capture_mixers[] = {
        {}
 };
 
+static bool select_automic(struct hda_codec *codec, int idx, bool detect)
+{
+       struct conexant_spec *spec = codec->spec;
+       if (idx < 0)
+               return false;
+       if (detect && !snd_hda_jack_detect(codec, spec->imux_info[idx].pin))
+               return false;
+       cx_auto_mux_enum_update(codec, &spec->private_imux, idx);
+       return true;
+}
+
 /* automatic switch internal and external mic */
 static void cx_auto_automic(struct hda_codec *codec)
 {
        struct conexant_spec *spec = codec->spec;
-       int ext_idx = spec->auto_mic_ext;
 
        if (!spec->auto_mic)
                return;
-       if (snd_hda_jack_detect(codec, spec->imux_pins[ext_idx]))
-               cx_auto_mux_enum_update(codec, &spec->private_imux, ext_idx);
-       else
-               cx_auto_mux_enum_update(codec, &spec->private_imux, !ext_idx);
+       if (!select_automic(codec, spec->auto_mic_ext, true))
+               if (!select_automic(codec, spec->auto_mic_dock, true))
+                       select_automic(codec, spec->auto_mic_int, false);
 }
 
 static void cx_auto_unsol_event(struct hda_codec *codec, unsigned int res)
@@ -3570,7 +3768,9 @@ static void cx_auto_unsol_event(struct hda_codec *codec, unsigned int res)
        switch (res >> 26) {
        case CONEXANT_HP_EVENT:
                cx_auto_hp_automute(codec);
-               snd_hda_input_jack_report(codec, nid);
+               break;
+       case CONEXANT_LINE_EVENT:
+               cx_auto_line_automute(codec);
                break;
        case CONEXANT_MIC_EVENT:
                cx_auto_automic(codec);
@@ -3579,42 +3779,45 @@ static void cx_auto_unsol_event(struct hda_codec *codec, unsigned int res)
        }
 }
 
-/* return true if it's an internal-mic pin */
-static int is_int_mic(struct hda_codec *codec, hda_nid_t pin)
-{
-       unsigned int def_conf = snd_hda_codec_get_pincfg(codec, pin);
-       return get_defcfg_device(def_conf) == AC_JACK_MIC_IN &&
-               snd_hda_get_input_pin_attr(def_conf) == INPUT_PIN_ATTR_INT;
-}
-
-/* return true if it's an external-mic pin */
-static int is_ext_mic(struct hda_codec *codec, hda_nid_t pin)
-{
-       unsigned int def_conf = snd_hda_codec_get_pincfg(codec, pin);
-       return get_defcfg_device(def_conf) == AC_JACK_MIC_IN &&
-               snd_hda_get_input_pin_attr(def_conf) >= INPUT_PIN_ATTR_NORMAL &&
-               (snd_hda_query_pin_caps(codec, pin) & AC_PINCAP_PRES_DETECT);
-}
-
 /* check whether the pin config is suitable for auto-mic switching;
- * auto-mic is enabled only when one int-mic and one-ext mic exist
+ * auto-mic is enabled only when one int-mic and one ext- and/or
+ * one dock-mic exist
  */
 static void cx_auto_check_auto_mic(struct hda_codec *codec)
 {
        struct conexant_spec *spec = codec->spec;
+       int pset[INPUT_PIN_ATTR_NORMAL + 1];
+       int i;
 
-       if (is_ext_mic(codec, spec->imux_pins[0]) &&
-           is_int_mic(codec, spec->imux_pins[1])) {
-               spec->auto_mic = 1;
-               spec->auto_mic_ext = 0;
-               return;
-       }
-       if (is_int_mic(codec, spec->imux_pins[0]) &&
-           is_ext_mic(codec, spec->imux_pins[1])) {
-               spec->auto_mic = 1;
-               spec->auto_mic_ext = 1;
-               return;
+       for (i = 0; i < INPUT_PIN_ATTR_NORMAL; i++)
+               pset[i] = -1;
+       for (i = 0; i < spec->private_imux.num_items; i++) {
+               hda_nid_t pin = spec->imux_info[i].pin;
+               unsigned int def_conf = snd_hda_codec_get_pincfg(codec, pin);
+               int type, attr;
+               attr = snd_hda_get_input_pin_attr(def_conf);
+               if (attr == INPUT_PIN_ATTR_UNUSED)
+                       return; /* invalid entry */
+               if (attr > INPUT_PIN_ATTR_NORMAL)
+                       attr = INPUT_PIN_ATTR_NORMAL;
+               if (attr != INPUT_PIN_ATTR_INT &&
+                   !is_jack_detectable(codec, pin))
+                       return; /* non-detectable pin */
+               type = get_defcfg_device(def_conf);
+               if (type != AC_JACK_MIC_IN &&
+                   (attr != INPUT_PIN_ATTR_DOCK || type != AC_JACK_LINE_IN))
+                       return; /* no valid input type */
+               if (pset[attr] >= 0)
+                       return; /* already occupied */
+               pset[attr] = i;
        }
+       if (pset[INPUT_PIN_ATTR_INT] < 0 ||
+           (pset[INPUT_PIN_ATTR_NORMAL] < 0 && pset[INPUT_PIN_ATTR_DOCK]))
+               return; /* no input to switch*/
+       spec->auto_mic = 1;
+       spec->auto_mic_ext = pset[INPUT_PIN_ATTR_NORMAL];
+       spec->auto_mic_dock = pset[INPUT_PIN_ATTR_DOCK];
+       spec->auto_mic_int = pset[INPUT_PIN_ATTR_INT];
 }
 
 static void cx_auto_parse_input(struct hda_codec *codec)
@@ -3633,19 +3836,21 @@ static void cx_auto_parse_input(struct hda_codec *codec)
                        if (idx >= 0) {
                                const char *label;
                                label = hda_get_autocfg_input_label(codec, cfg, i);
-                               spec->imux_adcs[imux->num_items] = adc;
-                               spec->imux_pins[imux->num_items] =
+                               spec->imux_info[imux->num_items].index = i;
+                               spec->imux_info[imux->num_items].boost = 0;
+                               spec->imux_info[imux->num_items].adc = adc;
+                               spec->imux_info[imux->num_items].pin =
                                        cfg->inputs[i].pin;
                                snd_hda_add_imux_item(imux, label, idx, NULL);
                                break;
                        }
                }
        }
-       if (imux->num_items == 2 && cfg->num_inputs == 2)
+       if (imux->num_items >= 2 && cfg->num_inputs == imux->num_items)
                cx_auto_check_auto_mic(codec);
        if (imux->num_items > 1 && !spec->auto_mic) {
                for (i = 1; i < imux->num_items; i++) {
-                       if (spec->imux_adcs[i] != spec->imux_adcs[0]) {
+                       if (spec->imux_info[i].adc != spec->imux_info[0].adc) {
                                spec->adc_switching = 1;
                                break;
                        }
@@ -3755,6 +3960,16 @@ static void mute_outputs(struct hda_codec *codec, int num_nids,
        }
 }
 
+static void enable_unsol_pins(struct hda_codec *codec, int num_pins,
+                             hda_nid_t *pins, unsigned int tag)
+{
+       int i;
+       for (i = 0; i < num_pins; i++)
+               snd_hda_codec_write(codec, pins[i], 0,
+                                   AC_VERB_SET_UNSOLICITED_ENABLE,
+                                   AC_USRSP_EN | tag);
+}
+
 static void cx_auto_init_output(struct hda_codec *codec)
 {
        struct conexant_spec *spec = codec->spec;
@@ -3769,35 +3984,27 @@ static void cx_auto_init_output(struct hda_codec *codec)
        mute_outputs(codec, cfg->hp_outs, cfg->hp_pins);
        mute_outputs(codec, cfg->line_outs, cfg->line_out_pins);
        mute_outputs(codec, cfg->speaker_outs, cfg->speaker_pins);
-       if (spec->auto_mute) {
-               for (i = 0; i < cfg->hp_outs; i++) {
-                       snd_hda_codec_write(codec, cfg->hp_pins[i], 0,
-                                   AC_VERB_SET_UNSOLICITED_ENABLE,
-                                   AC_USRSP_EN | CONEXANT_HP_EVENT);
-               }
-               cx_auto_hp_automute(codec);
-       } else {
-               for (i = 0; i < cfg->line_outs; i++)
-                       snd_hda_codec_write(codec, cfg->line_out_pins[i], 0,
-                                   AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
-               for (i = 0; i < cfg->speaker_outs; i++)
-                       snd_hda_codec_write(codec, cfg->speaker_pins[i], 0,
-                                   AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
-               /* turn on EAPD */
-               cx_auto_turn_eapd(codec, cfg->line_outs, cfg->line_out_pins,
-                                 true);
-               cx_auto_turn_eapd(codec, cfg->hp_outs, cfg->hp_pins,
-                                 true);
-               cx_auto_turn_eapd(codec, cfg->speaker_outs, cfg->speaker_pins,
-                                 true);
-       }
-
        for (i = 0; i < spec->dac_info_filled; i++) {
                nid = spec->dac_info[i].dac;
                if (!nid)
                        nid = spec->multiout.dac_nids[0];
                select_connection(codec, spec->dac_info[i].pin, nid);
        }
+       if (spec->auto_mute) {
+               enable_unsol_pins(codec, cfg->hp_outs, cfg->hp_pins,
+                                 CONEXANT_HP_EVENT);
+               spec->hp_present = detect_jacks(codec, cfg->hp_outs,
+                                               cfg->hp_pins);
+               if (spec->detect_line) {
+                       enable_unsol_pins(codec, cfg->line_outs,
+                                         cfg->line_out_pins,
+                                         CONEXANT_LINE_EVENT);
+                       spec->line_present =
+                               detect_jacks(codec, cfg->line_outs,
+                                            cfg->line_out_pins);
+               }
+       }
+       cx_auto_update_speakers(codec);
 }
 
 static void cx_auto_init_input(struct hda_codec *codec)
@@ -3829,14 +4036,22 @@ static void cx_auto_init_input(struct hda_codec *codec)
        }
 
        if (spec->auto_mic) {
-               int ext_idx = spec->auto_mic_ext;
-               snd_hda_codec_write(codec, cfg->inputs[ext_idx].pin, 0,
-                                   AC_VERB_SET_UNSOLICITED_ENABLE,
-                                   AC_USRSP_EN | CONEXANT_MIC_EVENT);
+               if (spec->auto_mic_ext >= 0) {
+                       snd_hda_codec_write(codec,
+                               cfg->inputs[spec->auto_mic_ext].pin, 0,
+                               AC_VERB_SET_UNSOLICITED_ENABLE,
+                               AC_USRSP_EN | CONEXANT_MIC_EVENT);
+               }
+               if (spec->auto_mic_dock >= 0) {
+                       snd_hda_codec_write(codec,
+                               cfg->inputs[spec->auto_mic_dock].pin, 0,
+                               AC_VERB_SET_UNSOLICITED_ENABLE,
+                               AC_USRSP_EN | CONEXANT_MIC_EVENT);
+               }
                cx_auto_automic(codec);
        } else {
-               select_input_connection(codec, spec->imux_adcs[0],
-                                       spec->imux_pins[0]);
+               select_input_connection(codec, spec->imux_info[0].adc,
+                                       spec->imux_info[0].pin);
        }
 }
 
@@ -3953,6 +4168,13 @@ static int cx_auto_build_output_controls(struct hda_codec *codec)
                if (err < 0)
                        return err;
        }
+
+       if (spec->auto_mute) {
+               err = snd_hda_add_new_ctls(codec, cx_automute_mode_enum);
+               if (err < 0)
+                       return err;
+       }
+       
        return 0;
 }
 
@@ -3974,35 +4196,73 @@ static int cx_auto_add_capture_volume(struct hda_codec *codec, hda_nid_t nid,
        return 0;
 }
 
+static int cx_auto_add_boost_volume(struct hda_codec *codec, int idx,
+                                   const char *label, int cidx)
+{
+       struct conexant_spec *spec = codec->spec;
+       hda_nid_t mux, nid;
+       int i, con;
+
+       nid = spec->imux_info[idx].pin;
+       if (get_wcaps(codec, nid) & AC_WCAP_IN_AMP)
+               return cx_auto_add_volume(codec, label, " Boost", cidx,
+                                         nid, HDA_INPUT);
+       con = __select_input_connection(codec, spec->imux_info[idx].adc, nid,
+                                       &mux, false, 0);
+       if (con < 0)
+               return 0;
+       for (i = 0; i < idx; i++) {
+               if (spec->imux_info[i].boost == mux)
+                       return 0; /* already present */
+       }
+
+       if (get_wcaps(codec, mux) & AC_WCAP_OUT_AMP) {
+               spec->imux_info[idx].boost = mux;
+               return cx_auto_add_volume(codec, label, " Boost", 0,
+                                         mux, HDA_OUTPUT);
+       }
+       return 0;
+}
+
 static int cx_auto_build_input_controls(struct hda_codec *codec)
 {
        struct conexant_spec *spec = codec->spec;
-       struct auto_pin_cfg *cfg = &spec->autocfg;
-       static const char *prev_label;
+       struct hda_input_mux *imux = &spec->private_imux;
+       const char *prev_label;
+       int input_conn[HDA_MAX_NUM_INPUTS];
        int i, err, cidx;
+       int multi_connection;
+
+       multi_connection = 0;
+       for (i = 0; i < imux->num_items; i++) {
+               cidx = get_input_connection(codec, spec->imux_info[i].adc,
+                                           spec->imux_info[i].pin);
+               input_conn[i] = (spec->imux_info[i].adc << 8) | cidx;
+               if (i > 0 && input_conn[i] != input_conn[0])
+                       multi_connection = 1;
+       }
 
        prev_label = NULL;
        cidx = 0;
-       for (i = 0; i < cfg->num_inputs; i++) {
-               hda_nid_t nid = cfg->inputs[i].pin;
+       for (i = 0; i < imux->num_items; i++) {
+               hda_nid_t nid = spec->imux_info[i].pin;
                const char *label;
-               int pin_amp = get_wcaps(codec, nid) & AC_WCAP_IN_AMP;
 
-               label = hda_get_autocfg_input_label(codec, cfg, i);
+               label = hda_get_autocfg_input_label(codec, &spec->autocfg,
+                                                   spec->imux_info[i].index);
                if (label == prev_label)
                        cidx++;
                else
                        cidx = 0;
                prev_label = label;
 
-               if (pin_amp) {
-                       err = cx_auto_add_volume(codec, label, " Boost", cidx,
-                                                nid, HDA_INPUT);
-                       if (err < 0)
-                               return err;
-               }
+               err = cx_auto_add_boost_volume(codec, i, label, cidx);
+               if (err < 0)
+                       return err;
 
-               if (cfg->num_inputs == 1) {
+               if (!multi_connection) {
+                       if (i > 0)
+                               continue;
                        err = cx_auto_add_capture_volume(codec, nid,
                                                         "Capture", "", cidx);
                } else {