| /* |
| * 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"); |