--- /dev/null
+From 667067d8980249a71ccf82a55202fff2cd1cd54f Mon Sep 17 00:00:00 2001
+From: Takashi Iwai <tiwai@suse.de>
+Date: Thu, 13 Aug 2009 18:14:42 +0200
+Subject: ALSA: hda - Fix / clean up IDT92HD83xxx codec parser
+Patch-mainline:
+References: bnc#531533
+
+A few improvements for IDT 92HD83xxx codec pareser:
+- Remove unused / deprecated mixer-amp controls
+- Handle d-mics as normal inputs since this codec has no separate
+ MUXes for analog and digital
+- Don't create duplicated controls for capture volumes with Mux
+ capture volumes
+
+Signed-off-by: Takashi Iwai <tiwai@suse.de>
+
+---
+ sound/pci/hda/patch_sigmatel.c | 150 +++++++++++++++++++----------------------
+ 1 file changed, 70 insertions(+), 80 deletions(-)
+
+--- a/sound/pci/hda/patch_sigmatel.c
++++ b/sound/pci/hda/patch_sigmatel.c
+@@ -349,14 +349,9 @@
+ };
+ #define stac92hd73xx_capsws stac92hd73xx_capvols
+
+-#define STAC92HD83XXX_NUM_DMICS 2
+-static hda_nid_t stac92hd83xxx_dmic_nids[STAC92HD83XXX_NUM_DMICS + 1] = {
+- 0x11, 0x12, 0
+-};
+-
+ #define STAC92HD83_DAC_COUNT 3
+
+-static hda_nid_t stac92hd83xxx_dmux_nids[2] = {
++static hda_nid_t stac92hd83xxx_mux_nids[2] = {
+ 0x17, 0x18,
+ };
+
+@@ -376,10 +371,6 @@
+ 0x03, 0x0c, 0x20, 0x40,
+ };
+
+-static hda_nid_t stac92hd83xxx_amp_nids[1] = {
+- 0xc,
+-};
+-
+ #define STAC92HD83XXX_NUM_CAPS 2
+ static unsigned long stac92hd83xxx_capvols[] = {
+ HDA_COMPOSE_AMP_VAL(0x17, 3, 0, HDA_OUTPUT),
+@@ -1092,26 +1083,6 @@
+ };
+
+
+-static struct snd_kcontrol_new stac92hd83xxx_mixer[] = {
+- HDA_CODEC_VOLUME("DAC0 Capture Volume", 0x1b, 0x3, HDA_INPUT),
+- HDA_CODEC_MUTE("DAC0 Capture Switch", 0x1b, 0x3, HDA_INPUT),
+-
+- HDA_CODEC_VOLUME("DAC1 Capture Volume", 0x1b, 0x4, HDA_INPUT),
+- HDA_CODEC_MUTE("DAC1 Capture Switch", 0x1b, 0x4, HDA_INPUT),
+-
+- HDA_CODEC_VOLUME("Front Mic Capture Volume", 0x1b, 0x0, HDA_INPUT),
+- HDA_CODEC_MUTE("Front Mic Capture Switch", 0x1b, 0x0, HDA_INPUT),
+-
+- HDA_CODEC_VOLUME("Line In Capture Volume", 0x1b, 0x2, HDA_INPUT),
+- HDA_CODEC_MUTE("Line In Capture Switch", 0x1b, 0x2, HDA_INPUT),
+-
+- /*
+- HDA_CODEC_VOLUME("Mic Capture Volume", 0x1b, 0x1, HDA_INPUT),
+- HDA_CODEC_MUTE("Mic Capture Switch", 0x1b 0x1, HDA_INPUT),
+- */
+- { } /* end */
+-};
+-
+ static struct snd_kcontrol_new stac925x_mixer[] = {
+ HDA_CODEC_VOLUME("Master Playback Volume", 0x0e, 0, HDA_OUTPUT),
+ HDA_CODEC_MUTE("Master Playback Switch", 0x0e, 0, HDA_OUTPUT),
+@@ -3380,19 +3351,33 @@
+ static int stac92xx_auto_create_mux_input_ctls(struct hda_codec *codec)
+ {
+ struct sigmatel_spec *spec = codec->spec;
+- int wcaps, nid, i, err = 0;
++ int i, j, err = 0;
+
+ for (i = 0; i < spec->num_muxes; i++) {
++ hda_nid_t nid;
++ unsigned int wcaps;
++ unsigned long val;
++
+ nid = spec->mux_nids[i];
+ wcaps = get_wcaps(codec, nid);
++ if (!(wcaps & AC_WCAP_OUT_AMP))
++ continue;
+
+- if (wcaps & AC_WCAP_OUT_AMP) {
+- err = stac92xx_add_control_idx(spec,
+- STAC_CTL_WIDGET_VOL, i, "Mux Capture Volume",
+- HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_OUTPUT));
+- if (err < 0)
+- return err;
++ /* check whether already the same control was created as
++ * normal Capture Volume.
++ */
++ val = HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_OUTPUT);
++ for (j = 0; j < spec->num_caps; j++) {
++ if (spec->capvols[j] == val)
++ break;
+ }
++ if (j < spec->num_caps)
++ continue;
++
++ err = stac92xx_add_control_idx(spec, STAC_CTL_WIDGET_VOL, i,
++ "Mux Capture Volume", val);
++ if (err < 0)
++ return err;
+ }
+ return 0;
+ };
+@@ -3447,6 +3432,24 @@
+ return -1;
+ }
+
++/* create a volume assigned to the given pin (only if supported) */
++static int create_elem_capture_vol(struct hda_codec *codec, hda_nid_t nid,
++ const char *label)
++{
++ unsigned int caps, nums;
++ char name[32];
++
++ if (!(get_wcaps(codec, nid) & AC_WCAP_IN_AMP))
++ return 0;
++ caps = query_amp_caps(codec, nid, HDA_OUTPUT);
++ nums = (caps & AC_AMPCAP_NUM_STEPS) >> AC_AMPCAP_NUM_STEPS_SHIFT;
++ if (!nums)
++ return 0;
++ snprintf(name, sizeof(name), "%s Capture Volume", label);
++ return stac92xx_add_control(codec->spec, STAC_CTL_WIDGET_VOL, name,
++ HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_INPUT));
++}
++
+ /* create playback/capture controls for input pins on dmic capable codecs */
+ static int stac92xx_auto_create_dmic_input_ctls(struct hda_codec *codec,
+ const struct auto_pin_cfg *cfg)
+@@ -3456,7 +3459,6 @@
+ struct hda_input_mux *dimux = &spec->private_dimux;
+ int err, i, active_mics;
+ unsigned int def_conf;
+- char name[32];
+
+ dimux->items[dimux->num_items].label = stac92xx_dmic_labels[0];
+ dimux->items[dimux->num_items].index = 0;
+@@ -3464,6 +3466,10 @@
+
+ active_mics = 0;
+ for (i = 0; i < spec->num_dmics; i++) {
++ /* check the validity: sometimes it's a dead vendor-spec node */
++ if (get_wcaps_type(get_wcaps(codec, spec->dmic_nids[i]))
++ != AC_WID_PIN)
++ continue;
+ def_conf = snd_hda_codec_get_pincfg(codec, spec->dmic_nids[i]);
+ if (get_defcfg_connect(def_conf) != AC_JACK_PORT_NONE)
+ active_mics++;
+@@ -3472,14 +3478,15 @@
+ for (i = 0; i < spec->num_dmics; i++) {
+ hda_nid_t nid;
+ int index;
+- unsigned int wcaps;
+ const char *label;
+
+- def_conf = snd_hda_codec_get_pincfg(codec, spec->dmic_nids[i]);
++ nid = spec->dmic_nids[i];
++ if (get_wcaps_type(get_wcaps(codec, nid)) != AC_WID_PIN)
++ continue;
++ def_conf = snd_hda_codec_get_pincfg(codec, nid);
+ if (get_defcfg_connect(def_conf) == AC_JACK_PORT_NONE)
+ continue;
+
+- nid = spec->dmic_nids[i];
+ index = get_connection_index(codec, spec->dmux_nids[0], nid);
+ if (index < 0)
+ continue;
+@@ -3489,21 +3496,9 @@
+ else
+ label = stac92xx_dmic_labels[dimux->num_items];
+
+- wcaps = get_wcaps(codec, nid) &
+- (AC_WCAP_OUT_AMP | AC_WCAP_IN_AMP);
+-
+- if (wcaps) {
+- sprintf(name, "%s Capture Volume", label);
+-
+- err = stac92xx_add_control(spec,
+- STAC_CTL_WIDGET_VOL,
+- name,
+- HDA_COMPOSE_AMP_VAL(nid, 3, 0,
+- (wcaps & AC_WCAP_OUT_AMP) ?
+- HDA_OUTPUT : HDA_INPUT));
+- if (err < 0)
+- return err;
+- }
++ err = create_elem_capture_vol(codec, nid, label);
++ if (err < 0)
++ return err;
+
+ dimux->items[dimux->num_items].label = label;
+ dimux->items[dimux->num_items].index = index;
+@@ -3604,29 +3599,29 @@
+ {
+ struct sigmatel_spec *spec = codec->spec;
+ struct hda_input_mux *imux = &spec->private_imux;
+- hda_nid_t con_lst[HDA_MAX_NUM_INPUTS];
+- int i, j, k;
++ int i, j;
+
+ for (i = 0; i < AUTO_PIN_LAST; i++) {
+- int index;
++ hda_nid_t nid = cfg->input_pins[i];
++ int index, err;
+
+- if (!cfg->input_pins[i])
++ if (!nid)
+ continue;
+ index = -1;
+ for (j = 0; j < spec->num_muxes; j++) {
+- int num_cons;
+- num_cons = snd_hda_get_connections(codec,
+- spec->mux_nids[j],
+- con_lst,
+- HDA_MAX_NUM_INPUTS);
+- for (k = 0; k < num_cons; k++)
+- if (con_lst[k] == cfg->input_pins[i]) {
+- index = k;
+- goto found;
+- }
++ index = get_connection_index(codec, spec->mux_nids[j],
++ nid);
++ if (index >= 0)
++ break;
+ }
+- continue;
+- found:
++ if (index < 0)
++ continue;
++
++ err = create_elem_capture_vol(codec, nid,
++ auto_pin_cfg_labels[i]);
++ if (err < 0)
++ return err;
++
+ imux->items[imux->num_items].label = auto_pin_cfg_labels[i];
+ imux->items[imux->num_items].index = index;
+ imux->num_items++;
+@@ -4998,22 +4993,17 @@
+ codec->slave_dig_outs = stac92hd83xxx_slave_dig_outs;
+ spec->mono_nid = 0x19;
+ spec->digbeep_nid = 0x21;
+- spec->dmic_nids = stac92hd83xxx_dmic_nids;
+- spec->dmux_nids = stac92hd83xxx_dmux_nids;
++ spec->mux_nids = stac92hd83xxx_mux_nids;
++ spec->num_muxes = ARRAY_SIZE(stac92hd83xxx_mux_nids);
+ spec->adc_nids = stac92hd83xxx_adc_nids;
++ spec->num_adcs = ARRAY_SIZE(stac92hd83xxx_adc_nids);
+ spec->pwr_nids = stac92hd83xxx_pwr_nids;
+- spec->amp_nids = stac92hd83xxx_amp_nids;
+ spec->pwr_mapping = stac92hd83xxx_pwr_mapping;
+ spec->num_pwrs = ARRAY_SIZE(stac92hd83xxx_pwr_nids);
+ spec->multiout.dac_nids = spec->dac_nids;
+
+ spec->init = stac92hd83xxx_core_init;
+- spec->mixer = stac92hd83xxx_mixer;
+ spec->num_pins = ARRAY_SIZE(stac92hd83xxx_pin_nids);
+- spec->num_dmuxes = ARRAY_SIZE(stac92hd83xxx_dmux_nids);
+- spec->num_adcs = ARRAY_SIZE(stac92hd83xxx_adc_nids);
+- spec->num_amps = ARRAY_SIZE(stac92hd83xxx_amp_nids);
+- spec->num_dmics = STAC92HD83XXX_NUM_DMICS;
+ spec->pin_nids = stac92hd83xxx_pin_nids;
+ spec->num_caps = STAC92HD83XXX_NUM_CAPS;
+ spec->capvols = stac92hd83xxx_capvols;