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