| // 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 */ |