xref: /linux/drivers/net/wireless/intel/iwlwifi/mld/led.c (revision 1a9239bb4253f9076b5b4b2a1a4e8d7defd77a95)
1*d1e879ecSMiri Korenblit // SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause
2*d1e879ecSMiri Korenblit /*
3*d1e879ecSMiri Korenblit  * Copyright (C) 2024 Intel Corporation
4*d1e879ecSMiri Korenblit  */
5*d1e879ecSMiri Korenblit #include <linux/leds.h>
6*d1e879ecSMiri Korenblit #include <net/mac80211.h>
7*d1e879ecSMiri Korenblit 
8*d1e879ecSMiri Korenblit #include "fw/api/led.h"
9*d1e879ecSMiri Korenblit #include "mld.h"
10*d1e879ecSMiri Korenblit #include "led.h"
11*d1e879ecSMiri Korenblit #include "hcmd.h"
12*d1e879ecSMiri Korenblit 
iwl_mld_send_led_fw_cmd(struct iwl_mld * mld,bool on)13*d1e879ecSMiri Korenblit static void iwl_mld_send_led_fw_cmd(struct iwl_mld *mld, bool on)
14*d1e879ecSMiri Korenblit {
15*d1e879ecSMiri Korenblit 	struct iwl_led_cmd led_cmd = {
16*d1e879ecSMiri Korenblit 		.status = cpu_to_le32(on),
17*d1e879ecSMiri Korenblit 	};
18*d1e879ecSMiri Korenblit 	int err;
19*d1e879ecSMiri Korenblit 
20*d1e879ecSMiri Korenblit 	if (WARN_ON(!mld->fw_status.running))
21*d1e879ecSMiri Korenblit 		return;
22*d1e879ecSMiri Korenblit 
23*d1e879ecSMiri Korenblit 	err = iwl_mld_send_cmd_with_flags_pdu(mld, WIDE_ID(LONG_GROUP,
24*d1e879ecSMiri Korenblit 							   LEDS_CMD),
25*d1e879ecSMiri Korenblit 					      CMD_ASYNC, &led_cmd);
26*d1e879ecSMiri Korenblit 
27*d1e879ecSMiri Korenblit 	if (err)
28*d1e879ecSMiri Korenblit 		IWL_WARN(mld, "LED command failed: %d\n", err);
29*d1e879ecSMiri Korenblit }
30*d1e879ecSMiri Korenblit 
iwl_led_brightness_set(struct led_classdev * led_cdev,enum led_brightness brightness)31*d1e879ecSMiri Korenblit static void iwl_led_brightness_set(struct led_classdev *led_cdev,
32*d1e879ecSMiri Korenblit 				   enum led_brightness brightness)
33*d1e879ecSMiri Korenblit {
34*d1e879ecSMiri Korenblit 	struct iwl_mld *mld = container_of(led_cdev, struct iwl_mld, led);
35*d1e879ecSMiri Korenblit 
36*d1e879ecSMiri Korenblit 	if (!mld->fw_status.running)
37*d1e879ecSMiri Korenblit 		return;
38*d1e879ecSMiri Korenblit 
39*d1e879ecSMiri Korenblit 	iwl_mld_send_led_fw_cmd(mld, brightness > 0);
40*d1e879ecSMiri Korenblit }
41*d1e879ecSMiri Korenblit 
iwl_mld_leds_init(struct iwl_mld * mld)42*d1e879ecSMiri Korenblit int iwl_mld_leds_init(struct iwl_mld *mld)
43*d1e879ecSMiri Korenblit {
44*d1e879ecSMiri Korenblit 	int mode = iwlwifi_mod_params.led_mode;
45*d1e879ecSMiri Korenblit 	int ret;
46*d1e879ecSMiri Korenblit 
47*d1e879ecSMiri Korenblit 	switch (mode) {
48*d1e879ecSMiri Korenblit 	case IWL_LED_BLINK:
49*d1e879ecSMiri Korenblit 		IWL_ERR(mld, "Blink led mode not supported, used default\n");
50*d1e879ecSMiri Korenblit 		fallthrough;
51*d1e879ecSMiri Korenblit 	case IWL_LED_DEFAULT:
52*d1e879ecSMiri Korenblit 	case IWL_LED_RF_STATE:
53*d1e879ecSMiri Korenblit 		mode = IWL_LED_RF_STATE;
54*d1e879ecSMiri Korenblit 		break;
55*d1e879ecSMiri Korenblit 	case IWL_LED_DISABLE:
56*d1e879ecSMiri Korenblit 		IWL_INFO(mld, "Led disabled\n");
57*d1e879ecSMiri Korenblit 		return 0;
58*d1e879ecSMiri Korenblit 	default:
59*d1e879ecSMiri Korenblit 		return -EINVAL;
60*d1e879ecSMiri Korenblit 	}
61*d1e879ecSMiri Korenblit 
62*d1e879ecSMiri Korenblit 	mld->led.name = kasprintf(GFP_KERNEL, "%s-led",
63*d1e879ecSMiri Korenblit 				  wiphy_name(mld->hw->wiphy));
64*d1e879ecSMiri Korenblit 	if (!mld->led.name)
65*d1e879ecSMiri Korenblit 		return -ENOMEM;
66*d1e879ecSMiri Korenblit 
67*d1e879ecSMiri Korenblit 	mld->led.brightness_set = iwl_led_brightness_set;
68*d1e879ecSMiri Korenblit 	mld->led.max_brightness = 1;
69*d1e879ecSMiri Korenblit 
70*d1e879ecSMiri Korenblit 	if (mode == IWL_LED_RF_STATE)
71*d1e879ecSMiri Korenblit 		mld->led.default_trigger =
72*d1e879ecSMiri Korenblit 			ieee80211_get_radio_led_name(mld->hw);
73*d1e879ecSMiri Korenblit 
74*d1e879ecSMiri Korenblit 	ret = led_classdev_register(mld->trans->dev, &mld->led);
75*d1e879ecSMiri Korenblit 	if (ret) {
76*d1e879ecSMiri Korenblit 		kfree(mld->led.name);
77*d1e879ecSMiri Korenblit 		mld->led.name = NULL;
78*d1e879ecSMiri Korenblit 		IWL_INFO(mld, "Failed to enable led\n");
79*d1e879ecSMiri Korenblit 	}
80*d1e879ecSMiri Korenblit 
81*d1e879ecSMiri Korenblit 	return ret;
82*d1e879ecSMiri Korenblit }
83*d1e879ecSMiri Korenblit 
iwl_mld_led_config_fw(struct iwl_mld * mld)84*d1e879ecSMiri Korenblit void iwl_mld_led_config_fw(struct iwl_mld *mld)
85*d1e879ecSMiri Korenblit {
86*d1e879ecSMiri Korenblit 	if (!mld->led.name)
87*d1e879ecSMiri Korenblit 		return;
88*d1e879ecSMiri Korenblit 
89*d1e879ecSMiri Korenblit 	iwl_mld_send_led_fw_cmd(mld, mld->led.brightness > 0);
90*d1e879ecSMiri Korenblit }
91*d1e879ecSMiri Korenblit 
iwl_mld_leds_exit(struct iwl_mld * mld)92*d1e879ecSMiri Korenblit void iwl_mld_leds_exit(struct iwl_mld *mld)
93*d1e879ecSMiri Korenblit {
94*d1e879ecSMiri Korenblit 	if (!mld->led.name)
95*d1e879ecSMiri Korenblit 		return;
96*d1e879ecSMiri Korenblit 
97*d1e879ecSMiri Korenblit 	led_classdev_unregister(&mld->led);
98*d1e879ecSMiri Korenblit 	kfree(mld->led.name);
99*d1e879ecSMiri Korenblit 	mld->led.name = NULL;
100*d1e879ecSMiri Korenblit }
101