]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
leds: Add driver for LEDs from qnap-mcu devices
authorHeiko Stuebner <heiko@sntech.de>
Thu, 7 Nov 2024 11:47:08 +0000 (12:47 +0100)
committerLee Jones <lee@kernel.org>
Tue, 17 Dec 2024 13:14:38 +0000 (13:14 +0000)
This adds a driver that connects to the qnap-mcu mfd driver and provides
access to the LEDs on it.

Signed-off-by: Heiko Stuebner <heiko@sntech.de>
Link: https://lore.kernel.org/r/20241107114712.538976-6-heiko@sntech.de
Signed-off-by: Lee Jones <lee@kernel.org>
MAINTAINERS
drivers/leds/Kconfig
drivers/leds/Makefile
drivers/leds/leds-qnap-mcu.c [new file with mode: 0644]

index ccb9e7badb19d98580aeede704300689144d11a7..3391026448e731d48955838ffdcad72e02e4d8d4 100644 (file)
@@ -19108,6 +19108,7 @@ F:      drivers/media/tuners/qm1d1c0042*
 QNAP MCU DRIVER
 M:     Heiko Stuebner <heiko@sntech.de>
 S:     Maintained
+F:     drivers/leds/leds-qnap-mcu.c
 F:     drivers/mfd/qnap-mcu.c
 F:     include/linux/qnap-mcu.h
 
index b784bb74a8378a18735d0878f614680371c59064..28a208fa893ea66817fc1926319d35a757a8f739 100644 (file)
@@ -580,6 +580,17 @@ config LEDS_PCA995X
          LED driver chips accessed via the I2C bus. Supported
          devices include PCA9955BTW, PCA9952TW and PCA9955TW.
 
+config LEDS_QNAP_MCU
+       tristate "LED Support for QNAP MCU controllers"
+       depends on LEDS_CLASS
+       depends on MFD_QNAP_MCU
+       help
+         This option enables support for LEDs available on embedded
+         controllers used in QNAP NAS devices.
+
+         This driver can also be built as a module. If so, the module
+         will be called qnap-mcu-leds.
+
 config LEDS_WM831X_STATUS
        tristate "LED support for status LEDs on WM831x PMICs"
        depends on LEDS_CLASS
index 18afbb5a23ee5ac62308471a6c50eb0edd82418a..c6f74865d7299ed7655455d2ff077d23c1c25368 100644 (file)
@@ -79,6 +79,7 @@ obj-$(CONFIG_LEDS_PCA995X)            += leds-pca995x.o
 obj-$(CONFIG_LEDS_PM8058)              += leds-pm8058.o
 obj-$(CONFIG_LEDS_POWERNV)             += leds-powernv.o
 obj-$(CONFIG_LEDS_PWM)                 += leds-pwm.o
+obj-$(CONFIG_LEDS_QNAP_MCU)            += leds-qnap-mcu.o
 obj-$(CONFIG_LEDS_REGULATOR)           += leds-regulator.o
 obj-$(CONFIG_LEDS_SC27XX_BLTC)         += leds-sc27xx-bltc.o
 obj-$(CONFIG_LEDS_SUN50I_A100)         += leds-sun50i-a100.o
diff --git a/drivers/leds/leds-qnap-mcu.c b/drivers/leds/leds-qnap-mcu.c
new file mode 100644 (file)
index 0000000..4e47094
--- /dev/null
@@ -0,0 +1,227 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * Driver for LEDs found on QNAP MCU devices
+ *
+ * Copyright (C) 2024 Heiko Stuebner <heiko@sntech.de>
+ */
+
+#include <linux/leds.h>
+#include <linux/mfd/qnap-mcu.h>
+#include <linux/module.h>
+#include <linux/platform_device.h>
+#include <linux/slab.h>
+#include <uapi/linux/uleds.h>
+
+enum qnap_mcu_err_led_mode {
+       QNAP_MCU_ERR_LED_ON = 0,
+       QNAP_MCU_ERR_LED_OFF = 1,
+       QNAP_MCU_ERR_LED_BLINK_FAST = 2,
+       QNAP_MCU_ERR_LED_BLINK_SLOW = 3,
+};
+
+struct qnap_mcu_err_led {
+       struct qnap_mcu *mcu;
+       struct led_classdev cdev;
+       char name[LED_MAX_NAME_SIZE];
+       u8 num;
+       u8 mode;
+};
+
+static inline struct qnap_mcu_err_led *
+               cdev_to_qnap_mcu_err_led(struct led_classdev *led_cdev)
+{
+       return container_of(led_cdev, struct qnap_mcu_err_led, cdev);
+}
+
+static int qnap_mcu_err_led_set(struct led_classdev *led_cdev,
+                               enum led_brightness brightness)
+{
+       struct qnap_mcu_err_led *err_led = cdev_to_qnap_mcu_err_led(led_cdev);
+       u8 cmd[] = { '@', 'R', '0' + err_led->num, '0' };
+
+       /* Don't disturb a possible set blink-mode if LED stays on */
+       if (brightness != 0 && err_led->mode >= QNAP_MCU_ERR_LED_BLINK_FAST)
+               return 0;
+
+       err_led->mode = brightness ? QNAP_MCU_ERR_LED_ON : QNAP_MCU_ERR_LED_OFF;
+       cmd[3] = '0' + err_led->mode;
+
+       return qnap_mcu_exec_with_ack(err_led->mcu, cmd, sizeof(cmd));
+}
+
+static int qnap_mcu_err_led_blink_set(struct led_classdev *led_cdev,
+                                     unsigned long *delay_on,
+                                     unsigned long *delay_off)
+{
+       struct qnap_mcu_err_led *err_led = cdev_to_qnap_mcu_err_led(led_cdev);
+       u8 cmd[] = { '@', 'R', '0' + err_led->num, '0' };
+
+       /* LED is off, nothing to do */
+       if (err_led->mode == QNAP_MCU_ERR_LED_OFF)
+               return 0;
+
+       if (*delay_on < 500) {
+               *delay_on = 100;
+               *delay_off = 100;
+               err_led->mode = QNAP_MCU_ERR_LED_BLINK_FAST;
+       } else {
+               *delay_on = 500;
+               *delay_off = 500;
+               err_led->mode = QNAP_MCU_ERR_LED_BLINK_SLOW;
+       }
+
+       cmd[3] = '0' + err_led->mode;
+
+       return qnap_mcu_exec_with_ack(err_led->mcu, cmd, sizeof(cmd));
+}
+
+static int qnap_mcu_register_err_led(struct device *dev, struct qnap_mcu *mcu, int num_err_led)
+{
+       struct qnap_mcu_err_led *err_led;
+       int ret;
+
+       err_led = devm_kzalloc(dev, sizeof(*err_led), GFP_KERNEL);
+       if (!err_led)
+               return -ENOMEM;
+
+       err_led->mcu = mcu;
+       err_led->num = num_err_led;
+       err_led->mode = QNAP_MCU_ERR_LED_OFF;
+
+       scnprintf(err_led->name, LED_MAX_NAME_SIZE, "hdd%d:red:status", num_err_led + 1);
+       err_led->cdev.name = err_led->name;
+
+       err_led->cdev.brightness_set_blocking = qnap_mcu_err_led_set;
+       err_led->cdev.blink_set = qnap_mcu_err_led_blink_set;
+       err_led->cdev.brightness = 0;
+       err_led->cdev.max_brightness = 1;
+
+       ret = devm_led_classdev_register(dev, &err_led->cdev);
+       if (ret)
+               return ret;
+
+       return qnap_mcu_err_led_set(&err_led->cdev, 0);
+}
+
+enum qnap_mcu_usb_led_mode {
+       QNAP_MCU_USB_LED_ON = 1,
+       QNAP_MCU_USB_LED_OFF = 3,
+       QNAP_MCU_USB_LED_BLINK = 2,
+};
+
+struct qnap_mcu_usb_led {
+       struct qnap_mcu *mcu;
+       struct led_classdev cdev;
+       u8 mode;
+};
+
+static inline struct qnap_mcu_usb_led *
+               cdev_to_qnap_mcu_usb_led(struct led_classdev *led_cdev)
+{
+       return container_of(led_cdev, struct qnap_mcu_usb_led, cdev);
+}
+
+static int qnap_mcu_usb_led_set(struct led_classdev *led_cdev,
+                               enum led_brightness brightness)
+{
+       struct qnap_mcu_usb_led *usb_led = cdev_to_qnap_mcu_usb_led(led_cdev);
+       u8 cmd[] = { '@', 'C', 0 };
+
+       /* Don't disturb a possible set blink-mode if LED stays on */
+       if (brightness != 0 && usb_led->mode == QNAP_MCU_USB_LED_BLINK)
+               return 0;
+
+       usb_led->mode = brightness ? QNAP_MCU_USB_LED_ON : QNAP_MCU_USB_LED_OFF;
+
+       /*
+        * Byte 3 is shared between the usb led target on/off/blink
+        * and also the buzzer control (in the input driver)
+        */
+       cmd[2] = 'D' + usb_led->mode;
+
+       return qnap_mcu_exec_with_ack(usb_led->mcu, cmd, sizeof(cmd));
+}
+
+static int qnap_mcu_usb_led_blink_set(struct led_classdev *led_cdev,
+                                     unsigned long *delay_on,
+                                     unsigned long *delay_off)
+{
+       struct qnap_mcu_usb_led *usb_led = cdev_to_qnap_mcu_usb_led(led_cdev);
+       u8 cmd[] = { '@', 'C', 0 };
+
+       /* LED is off, nothing to do */
+       if (usb_led->mode == QNAP_MCU_USB_LED_OFF)
+               return 0;
+
+       *delay_on = 250;
+       *delay_off = 250;
+       usb_led->mode = QNAP_MCU_USB_LED_BLINK;
+
+       /*
+        * Byte 3 is shared between the USB LED target on/off/blink
+        * and also the buzzer control (in the input driver)
+        */
+       cmd[2] = 'D' + usb_led->mode;
+
+       return qnap_mcu_exec_with_ack(usb_led->mcu, cmd, sizeof(cmd));
+}
+
+static int qnap_mcu_register_usb_led(struct device *dev, struct qnap_mcu *mcu)
+{
+       struct qnap_mcu_usb_led *usb_led;
+       int ret;
+
+       usb_led = devm_kzalloc(dev, sizeof(*usb_led), GFP_KERNEL);
+       if (!usb_led)
+               return -ENOMEM;
+
+       usb_led->mcu = mcu;
+       usb_led->mode = QNAP_MCU_USB_LED_OFF;
+       usb_led->cdev.name = "usb:blue:disk";
+       usb_led->cdev.brightness_set_blocking = qnap_mcu_usb_led_set;
+       usb_led->cdev.blink_set = qnap_mcu_usb_led_blink_set;
+       usb_led->cdev.brightness = 0;
+       usb_led->cdev.max_brightness = 1;
+
+       ret = devm_led_classdev_register(dev, &usb_led->cdev);
+       if (ret)
+               return ret;
+
+       return qnap_mcu_usb_led_set(&usb_led->cdev, 0);
+}
+
+static int qnap_mcu_leds_probe(struct platform_device *pdev)
+{
+       struct qnap_mcu *mcu = dev_get_drvdata(pdev->dev.parent);
+       const struct qnap_mcu_variant *variant = pdev->dev.platform_data;
+       int ret;
+
+       for (int i = 0; i < variant->num_drives; i++) {
+               ret = qnap_mcu_register_err_led(&pdev->dev, mcu, i);
+               if (ret)
+                       return dev_err_probe(&pdev->dev, ret,
+                                       "failed to register error LED %d\n", i);
+       }
+
+       if (variant->usb_led) {
+               ret = qnap_mcu_register_usb_led(&pdev->dev, mcu);
+               if (ret)
+                       return dev_err_probe(&pdev->dev, ret,
+                                       "failed to register USB LED\n");
+       }
+
+       return 0;
+}
+
+static struct platform_driver qnap_mcu_leds_driver = {
+       .probe = qnap_mcu_leds_probe,
+       .driver = {
+               .name = "qnap-mcu-leds",
+       },
+};
+module_platform_driver(qnap_mcu_leds_driver);
+
+MODULE_ALIAS("platform:qnap-mcu-leds");
+MODULE_AUTHOR("Heiko Stuebner <heiko@sntech.de>");
+MODULE_DESCRIPTION("QNAP MCU LEDs driver");
+MODULE_LICENSE("GPL");