/* * Driver for the 8 user LEDs found on the RealViews and Versatiles * Based on DaVinci's DM365 board code * * License terms: GNU General Public License (GPL) version 2 * Author: Linus Walleij <triad@df.lth.se> */ #include <linux/kernel.h> #include <linux/init.h> #include <linux/module.h> #include <linux/io.h> #include <linux/slab.h> #include <linux/leds.h> #include <linux/platform_device.h> struct versatile_led { void __iomem *base; struct led_classdev cdev; u8 mask; }; /* * The triggers lines up below will only be used if the * LED triggers are compiled in. */ static const struct { const char *name; const char *trigger; } versatile_leds[] = { { "versatile:0", "heartbeat", }, { "versatile:1", "mmc0", }, { "versatile:2", "cpu0" }, { "versatile:3", "cpu1" }, { "versatile:4", "cpu2" }, { "versatile:5", "cpu3" }, { "versatile:6", }, { "versatile:7", }, }; static void versatile_led_set(struct led_classdev *cdev, enum led_brightness b) { struct versatile_led *led = container_of(cdev, struct versatile_led, cdev); u32 reg = readl(led->base); if (b != LED_OFF) reg |= led->mask; else reg &= ~led->mask; writel(reg, led->base); } static enum led_brightness versatile_led_get(struct led_classdev *cdev) { struct versatile_led *led = container_of(cdev, struct versatile_led, cdev); u32 reg = readl(led->base); return (reg & led->mask) ? LED_FULL : LED_OFF; } static int versatile_leds_probe(struct platform_device *dev) { int i; struct resource *res; void __iomem *base; res = platform_get_resource(dev, IORESOURCE_MEM, 0); base = devm_ioremap_resource(&dev->dev, res); if (IS_ERR(base)) return PTR_ERR(base); /* All off */ writel(0, base); for (i = 0; i < ARRAY_SIZE(versatile_leds); i++) { struct versatile_led *led; led = kzalloc(sizeof(*led), GFP_KERNEL); if (!led) break; led->base = base; led->cdev.name = versatile_leds[i].name; led->cdev.brightness_set = versatile_led_set; led->cdev.brightness_get = versatile_led_get; led->cdev.default_trigger = versatile_leds[i].trigger; led->mask = BIT(i); if (led_classdev_register(NULL, &led->cdev) < 0) { kfree(led); break; } } return 0; } static struct platform_driver versatile_leds_driver = { .driver = { .name = "versatile-leds", }, .probe = versatile_leds_probe, }; module_platform_driver(versatile_leds_driver); MODULE_AUTHOR("Linus Walleij <linus.walleij@linaro.org>"); MODULE_DESCRIPTION("ARM Versatile LED driver"); MODULE_LICENSE("GPL v2");