// SPDX-License-Identifier: GPL-2.0 /* Helper functions for Dell Mic Mute LED control; * to be included from codec driver */ #if IS_ENABLED(CONFIG_DELL_LAPTOP) #include enum { MICMUTE_LED_ON, MICMUTE_LED_OFF, MICMUTE_LED_FOLLOW_CAPTURE, MICMUTE_LED_FOLLOW_MUTE, }; static int dell_led_mode = MICMUTE_LED_FOLLOW_MUTE; static int dell_capture; static int dell_led_value; static int (*dell_micmute_led_set_func)(int); static void (*dell_old_cap_hook)(struct hda_codec *, struct snd_kcontrol *, struct snd_ctl_elem_value *); static void call_micmute_led_update(void) { int val; switch (dell_led_mode) { case MICMUTE_LED_ON: val = 1; break; case MICMUTE_LED_OFF: val = 0; break; case MICMUTE_LED_FOLLOW_CAPTURE: val = dell_capture; break; case MICMUTE_LED_FOLLOW_MUTE: default: val = !dell_capture; break; } if (val == dell_led_value) return; dell_led_value = val; dell_micmute_led_set_func(dell_led_value); } static void update_dell_wmi_micmute_led(struct hda_codec *codec, struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) { if (dell_old_cap_hook) dell_old_cap_hook(codec, kcontrol, ucontrol); if (!ucontrol || !dell_micmute_led_set_func) return; if (strcmp("Capture Switch", ucontrol->id.name) == 0 && ucontrol->id.index == 0) { /* TODO: How do I verify if it's a mono or stereo here? */ dell_capture = (ucontrol->value.integer.value[0] || ucontrol->value.integer.value[1]); call_micmute_led_update(); } } static int dell_mic_mute_led_mode_info(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_info *uinfo) { static const char * const texts[] = { "On", "Off", "Follow Capture", "Follow Mute", }; return snd_ctl_enum_info(uinfo, 1, ARRAY_SIZE(texts), texts); } static int dell_mic_mute_led_mode_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) { ucontrol->value.enumerated.item[0] = dell_led_mode; return 0; } static int dell_mic_mute_led_mode_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) { unsigned int mode; mode = ucontrol->value.enumerated.item[0]; if (mode > MICMUTE_LED_FOLLOW_MUTE) mode = MICMUTE_LED_FOLLOW_MUTE; if (mode == dell_led_mode) return 0; dell_led_mode = mode; call_micmute_led_update(); return 1; } static const struct snd_kcontrol_new dell_mic_mute_mode_ctls[] = { { .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = "Mic Mute-LED Mode", .info = dell_mic_mute_led_mode_info, .get = dell_mic_mute_led_mode_get, .put = dell_mic_mute_led_mode_put, }, {} }; static void alc_fixup_dell_wmi(struct hda_codec *codec, const struct hda_fixup *fix, int action) { struct alc_spec *spec = codec->spec; bool removefunc = false; if (action == HDA_FIXUP_ACT_PROBE) { if (!dell_micmute_led_set_func) dell_micmute_led_set_func = symbol_request(dell_micmute_led_set); if (!dell_micmute_led_set_func) { codec_warn(codec, "Failed to find dell wmi symbol dell_micmute_led_set\n"); return; } removefunc = true; if (dell_micmute_led_set_func(false) >= 0) { dell_led_value = 0; if (spec->gen.num_adc_nids > 1 && !spec->gen.dyn_adc_switch) codec_dbg(codec, "Skipping micmute LED control due to several ADCs"); else { dell_old_cap_hook = spec->gen.cap_sync_hook; spec->gen.cap_sync_hook = update_dell_wmi_micmute_led; removefunc = false; add_mixer(spec, dell_mic_mute_mode_ctls); } } } if (dell_micmute_led_set_func && (action == HDA_FIXUP_ACT_FREE || removefunc)) { symbol_put(dell_micmute_led_set); dell_micmute_led_set_func = NULL; dell_old_cap_hook = NULL; } } #else /* CONFIG_DELL_LAPTOP */ static void alc_fixup_dell_wmi(struct hda_codec *codec, const struct hda_fixup *fix, int action) { } #endif /* CONFIG_DELL_LAPTOP */