Commit 667067d8 authored by Takashi Iwai's avatar Takashi Iwai

ALSA: hda - Fix / clean up IDT92HD83xxx codec parser

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: default avatarTakashi Iwai <tiwai@suse.de>
parent a6cd7a71
...@@ -359,14 +359,9 @@ static unsigned long stac92hd73xx_capvols[] = { ...@@ -359,14 +359,9 @@ static unsigned long stac92hd73xx_capvols[] = {
}; };
#define stac92hd73xx_capsws stac92hd73xx_capvols #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 #define STAC92HD83_DAC_COUNT 3
static hda_nid_t stac92hd83xxx_dmux_nids[2] = { static hda_nid_t stac92hd83xxx_mux_nids[2] = {
0x17, 0x18, 0x17, 0x18,
}; };
...@@ -386,10 +381,6 @@ static unsigned int stac92hd83xxx_pwr_mapping[4] = { ...@@ -386,10 +381,6 @@ static unsigned int stac92hd83xxx_pwr_mapping[4] = {
0x03, 0x0c, 0x20, 0x40, 0x03, 0x0c, 0x20, 0x40,
}; };
static hda_nid_t stac92hd83xxx_amp_nids[1] = {
0xc,
};
#define STAC92HD83XXX_NUM_CAPS 2 #define STAC92HD83XXX_NUM_CAPS 2
static unsigned long stac92hd83xxx_capvols[] = { static unsigned long stac92hd83xxx_capvols[] = {
HDA_COMPOSE_AMP_VAL(0x17, 3, 0, HDA_OUTPUT), HDA_COMPOSE_AMP_VAL(0x17, 3, 0, HDA_OUTPUT),
...@@ -1212,26 +1203,6 @@ static struct snd_kcontrol_new stac92hd73xx_10ch_mixer[] = { ...@@ -1212,26 +1203,6 @@ static struct snd_kcontrol_new stac92hd73xx_10ch_mixer[] = {
}; };
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 stac92hd71bxx_loopback[] = { static struct snd_kcontrol_new stac92hd71bxx_loopback[] = {
STAC_ANALOG_LOOPBACK(0xFA0, 0x7A0, 2) STAC_ANALOG_LOOPBACK(0xFA0, 0x7A0, 2)
}; };
...@@ -3521,20 +3492,34 @@ static int stac92xx_beep_switch_ctl(struct hda_codec *codec) ...@@ -3521,20 +3492,34 @@ static int stac92xx_beep_switch_ctl(struct hda_codec *codec)
static int stac92xx_auto_create_mux_input_ctls(struct hda_codec *codec) static int stac92xx_auto_create_mux_input_ctls(struct hda_codec *codec)
{ {
struct sigmatel_spec *spec = codec->spec; 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++) { for (i = 0; i < spec->num_muxes; i++) {
hda_nid_t nid;
unsigned int wcaps;
unsigned long val;
nid = spec->mux_nids[i]; nid = spec->mux_nids[i];
wcaps = get_wcaps(codec, nid); wcaps = get_wcaps(codec, nid);
if (!(wcaps & AC_WCAP_OUT_AMP))
continue;
if (wcaps & AC_WCAP_OUT_AMP) { /* check whether already the same control was created as
err = stac92xx_add_control_idx(spec, * normal Capture Volume.
STAC_CTL_WIDGET_VOL, i, "Mux Capture Volume", */
HDA_COMPOSE_AMP_VAL(nid, 3, 0, HDA_OUTPUT)); 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) if (err < 0)
return err; return err;
} }
}
return 0; return 0;
}; };
...@@ -3588,6 +3573,24 @@ static int get_connection_index(struct hda_codec *codec, hda_nid_t mux, ...@@ -3588,6 +3573,24 @@ static int get_connection_index(struct hda_codec *codec, hda_nid_t mux,
return -1; 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 */ /* create playback/capture controls for input pins on dmic capable codecs */
static int stac92xx_auto_create_dmic_input_ctls(struct hda_codec *codec, static int stac92xx_auto_create_dmic_input_ctls(struct hda_codec *codec,
const struct auto_pin_cfg *cfg) const struct auto_pin_cfg *cfg)
...@@ -3597,7 +3600,6 @@ static int stac92xx_auto_create_dmic_input_ctls(struct hda_codec *codec, ...@@ -3597,7 +3600,6 @@ static int stac92xx_auto_create_dmic_input_ctls(struct hda_codec *codec,
struct hda_input_mux *dimux = &spec->private_dimux; struct hda_input_mux *dimux = &spec->private_dimux;
int err, i, active_mics; int err, i, active_mics;
unsigned int def_conf; unsigned int def_conf;
char name[32];
dimux->items[dimux->num_items].label = stac92xx_dmic_labels[0]; dimux->items[dimux->num_items].label = stac92xx_dmic_labels[0];
dimux->items[dimux->num_items].index = 0; dimux->items[dimux->num_items].index = 0;
...@@ -3605,6 +3607,10 @@ static int stac92xx_auto_create_dmic_input_ctls(struct hda_codec *codec, ...@@ -3605,6 +3607,10 @@ static int stac92xx_auto_create_dmic_input_ctls(struct hda_codec *codec,
active_mics = 0; active_mics = 0;
for (i = 0; i < spec->num_dmics; i++) { 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]); def_conf = snd_hda_codec_get_pincfg(codec, spec->dmic_nids[i]);
if (get_defcfg_connect(def_conf) != AC_JACK_PORT_NONE) if (get_defcfg_connect(def_conf) != AC_JACK_PORT_NONE)
active_mics++; active_mics++;
...@@ -3613,14 +3619,15 @@ static int stac92xx_auto_create_dmic_input_ctls(struct hda_codec *codec, ...@@ -3613,14 +3619,15 @@ static int stac92xx_auto_create_dmic_input_ctls(struct hda_codec *codec,
for (i = 0; i < spec->num_dmics; i++) { for (i = 0; i < spec->num_dmics; i++) {
hda_nid_t nid; hda_nid_t nid;
int index; int index;
unsigned int wcaps;
const char *label; 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) if (get_defcfg_connect(def_conf) == AC_JACK_PORT_NONE)
continue; continue;
nid = spec->dmic_nids[i];
index = get_connection_index(codec, spec->dmux_nids[0], nid); index = get_connection_index(codec, spec->dmux_nids[0], nid);
if (index < 0) if (index < 0)
continue; continue;
...@@ -3630,21 +3637,9 @@ static int stac92xx_auto_create_dmic_input_ctls(struct hda_codec *codec, ...@@ -3630,21 +3637,9 @@ static int stac92xx_auto_create_dmic_input_ctls(struct hda_codec *codec,
else else
label = stac92xx_dmic_labels[dimux->num_items]; label = stac92xx_dmic_labels[dimux->num_items];
wcaps = get_wcaps(codec, nid) & err = create_elem_capture_vol(codec, nid, label);
(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) if (err < 0)
return err; return err;
}
dimux->items[dimux->num_items].label = label; dimux->items[dimux->num_items].label = label;
dimux->items[dimux->num_items].index = index; dimux->items[dimux->num_items].index = index;
...@@ -3746,29 +3741,29 @@ static int stac92xx_auto_create_analog_input_ctls(struct hda_codec *codec, const ...@@ -3746,29 +3741,29 @@ static int stac92xx_auto_create_analog_input_ctls(struct hda_codec *codec, const
{ {
struct sigmatel_spec *spec = codec->spec; struct sigmatel_spec *spec = codec->spec;
struct hda_input_mux *imux = &spec->private_imux; struct hda_input_mux *imux = &spec->private_imux;
hda_nid_t con_lst[HDA_MAX_NUM_INPUTS]; int i, j;
int i, j, k;
for (i = 0; i < AUTO_PIN_LAST; i++) { 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; continue;
index = -1; index = -1;
for (j = 0; j < spec->num_muxes; j++) { for (j = 0; j < spec->num_muxes; j++) {
int num_cons; index = get_connection_index(codec, spec->mux_nids[j],
num_cons = snd_hda_get_connections(codec, nid);
spec->mux_nids[j], if (index >= 0)
con_lst, break;
HDA_MAX_NUM_INPUTS);
for (k = 0; k < num_cons; k++)
if (con_lst[k] == cfg->input_pins[i]) {
index = k;
goto found;
}
} }
if (index < 0)
continue; continue;
found:
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].label = auto_pin_cfg_labels[i];
imux->items[imux->num_items].index = index; imux->items[imux->num_items].index = index;
imux->num_items++; imux->num_items++;
...@@ -5299,22 +5294,16 @@ static int patch_stac92hd83xxx(struct hda_codec *codec) ...@@ -5299,22 +5294,16 @@ static int patch_stac92hd83xxx(struct hda_codec *codec)
codec->slave_dig_outs = stac92hd83xxx_slave_dig_outs; codec->slave_dig_outs = stac92hd83xxx_slave_dig_outs;
spec->mono_nid = 0x19; spec->mono_nid = 0x19;
spec->digbeep_nid = 0x21; spec->digbeep_nid = 0x21;
spec->dmic_nids = stac92hd83xxx_dmic_nids; spec->mux_nids = stac92hd83xxx_mux_nids;
spec->dmux_nids = stac92hd83xxx_dmux_nids; spec->num_muxes = ARRAY_SIZE(stac92hd83xxx_mux_nids);
spec->adc_nids = stac92hd83xxx_adc_nids; spec->adc_nids = stac92hd83xxx_adc_nids;
spec->pwr_nids = stac92hd83xxx_pwr_nids; spec->pwr_nids = stac92hd83xxx_pwr_nids;
spec->amp_nids = stac92hd83xxx_amp_nids;
spec->pwr_mapping = stac92hd83xxx_pwr_mapping; spec->pwr_mapping = stac92hd83xxx_pwr_mapping;
spec->num_pwrs = ARRAY_SIZE(stac92hd83xxx_pwr_nids); spec->num_pwrs = ARRAY_SIZE(stac92hd83xxx_pwr_nids);
spec->multiout.dac_nids = spec->dac_nids; spec->multiout.dac_nids = spec->dac_nids;
spec->init = stac92hd83xxx_core_init; spec->init = stac92hd83xxx_core_init;
spec->mixer = stac92hd83xxx_mixer;
spec->num_pins = ARRAY_SIZE(stac92hd83xxx_pin_nids); 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->pin_nids = stac92hd83xxx_pin_nids;
spec->num_caps = STAC92HD83XXX_NUM_CAPS; spec->num_caps = STAC92HD83XXX_NUM_CAPS;
spec->capvols = stac92hd83xxx_capvols; spec->capvols = stac92hd83xxx_capvols;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment