mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-23 00:20:52 -05:00
3192f14124
The DT of_device.h and of_platform.h date back to the separate of_platform_bus_type before it as merged into the regular platform bus. As part of that merge prepping Arm DT support 13 years ago, they "temporarily" include each other. They also include platform_device.h and of.h. As a result, there's a pretty much random mix of those include files used throughout the tree. In order to detangle these headers and replace the implicit includes with struct declarations, users need to explicitly include the correct includes. Signed-off-by: Rob Herring <robh@kernel.org> Link: https://lore.kernel.org/r/20230714174651.4058753-1-robh@kernel.org Signed-off-by: Lee Jones <lee@kernel.org>
133 lines
3 KiB
C
133 lines
3 KiB
C
// 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/platform_device.h>
|
|
#include <linux/regmap.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");
|