153 lines
3.8 KiB
C
153 lines
3.8 KiB
C
|
// 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 <linux/dell-led.h>
|
||
|
|
||
|
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 */
|