| // SPDX-License-Identifier: BSD-2-Clause OR GPL-2.0-or-later |
| /* |
| * Dell Wyse 3020 a.k.a. "Ariel" Embedded Controller LED Driver |
| * |
| * Copyright (C) 2020 Lubomir Rintel |
| */ |
| |
| #include <linux/module.h> |
| #include <linux/leds.h> |
| #include <linux/regmap.h> |
| #include <linux/of_platform.h> |
| |
| enum ec_index { |
| EC_BLUE_LED = 0x01, |
| EC_AMBER_LED = 0x02, |
| EC_GREEN_LED = 0x03, |
| }; |
| |
| enum { |
| EC_LED_OFF = 0x00, |
| EC_LED_STILL = 0x01, |
| EC_LED_FADE = 0x02, |
| EC_LED_BLINK = 0x03, |
| }; |
| |
| struct ariel_led { |
| struct regmap *ec_ram; |
| enum ec_index ec_index; |
| struct led_classdev led_cdev; |
| }; |
| |
| #define led_cdev_to_ariel_led(c) container_of(c, struct ariel_led, led_cdev) |
| |
| static enum led_brightness ariel_led_get(struct led_classdev *led_cdev) |
| { |
| struct ariel_led *led = led_cdev_to_ariel_led(led_cdev); |
| unsigned int led_status = 0; |
| |
| if (regmap_read(led->ec_ram, led->ec_index, &led_status)) |
| return LED_OFF; |
| |
| if (led_status == EC_LED_STILL) |
| return LED_FULL; |
| else |
| return LED_OFF; |
| } |
| |
| static void ariel_led_set(struct led_classdev *led_cdev, |
| enum led_brightness brightness) |
| { |
| struct ariel_led *led = led_cdev_to_ariel_led(led_cdev); |
| |
| if (brightness == LED_OFF) |
| regmap_write(led->ec_ram, led->ec_index, EC_LED_OFF); |
| else |
| regmap_write(led->ec_ram, led->ec_index, EC_LED_STILL); |
| } |
| |
| static int ariel_blink_set(struct led_classdev *led_cdev, |
| unsigned long *delay_on, unsigned long *delay_off) |
| { |
| struct ariel_led *led = led_cdev_to_ariel_led(led_cdev); |
| |
| if (*delay_on == 0 && *delay_off == 0) |
| return -EINVAL; |
| |
| if (*delay_on == 0) { |
| regmap_write(led->ec_ram, led->ec_index, EC_LED_OFF); |
| } else if (*delay_off == 0) { |
| regmap_write(led->ec_ram, led->ec_index, EC_LED_STILL); |
| } else { |
| *delay_on = 500; |
| *delay_off = 500; |
| regmap_write(led->ec_ram, led->ec_index, EC_LED_BLINK); |
| } |
| |
| return 0; |
| } |
| |
| #define NLEDS 3 |
| |
| static int ariel_led_probe(struct platform_device *pdev) |
| { |
| struct device *dev = &pdev->dev; |
| struct ariel_led *leds; |
| struct regmap *ec_ram; |
| int ret; |
| int i; |
| |
| ec_ram = dev_get_regmap(dev->parent, "ec_ram"); |
| if (!ec_ram) |
| return -ENODEV; |
| |
| leds = devm_kcalloc(dev, NLEDS, sizeof(*leds), GFP_KERNEL); |
| if (!leds) |
| return -ENOMEM; |
| |
| leds[0].ec_index = EC_BLUE_LED; |
| leds[0].led_cdev.name = "blue:power"; |
| leds[0].led_cdev.default_trigger = "default-on"; |
| |
| leds[1].ec_index = EC_AMBER_LED; |
| leds[1].led_cdev.name = "amber:status"; |
| |
| leds[2].ec_index = EC_GREEN_LED; |
| leds[2].led_cdev.name = "green:status"; |
| leds[2].led_cdev.default_trigger = "default-on"; |
| |
| for (i = 0; i < NLEDS; i++) { |
| leds[i].ec_ram = ec_ram; |
| leds[i].led_cdev.brightness_get = ariel_led_get; |
| leds[i].led_cdev.brightness_set = ariel_led_set; |
| leds[i].led_cdev.blink_set = ariel_blink_set; |
| |
| ret = devm_led_classdev_register(dev, &leds[i].led_cdev); |
| if (ret) |
| return ret; |
| } |
| |
| return 0; |
| } |
| |
| static struct platform_driver ariel_led_driver = { |
| .probe = ariel_led_probe, |
| .driver = { |
| .name = "dell-wyse-ariel-led", |
| }, |
| }; |
| module_platform_driver(ariel_led_driver); |
| |
| MODULE_AUTHOR("Lubomir Rintel <lkundrak@v3.sk>"); |
| MODULE_DESCRIPTION("Dell Wyse 3020 Status LEDs Driver"); |
| MODULE_LICENSE("Dual BSD/GPL"); |