2 * Driver for BCM6358 memory-mapped LEDs, based on leds-syscon.c
4 * Copyright 2015 Álvaro Fernández Rojas <noltari@gmail.com>
6 * This program is free software; you can redistribute it and/or modify it
7 * under the terms of the GNU General Public License as published by the
8 * Free Software Foundation; either version 2 of the License, or (at your
9 * option) any later version.
11 #include <linux/delay.h>
13 #include <linux/leds.h>
14 #include <linux/module.h>
16 #include <linux/platform_device.h>
17 #include <linux/spinlock.h>
19 #define BCM6358_REG_MODE 0x0
20 #define BCM6358_REG_CTRL 0x4
22 #define BCM6358_SLED_CLKDIV_MASK 3
23 #define BCM6358_SLED_CLKDIV_1 0
24 #define BCM6358_SLED_CLKDIV_2 1
25 #define BCM6358_SLED_CLKDIV_4 2
26 #define BCM6358_SLED_CLKDIV_8 3
28 #define BCM6358_SLED_POLARITY BIT(2)
29 #define BCM6358_SLED_BUSY BIT(3)
31 #define BCM6358_SLED_MAX_COUNT 32
32 #define BCM6358_SLED_WAIT 100
35 * struct bcm6358_led - state container for bcm6358 based LEDs
36 * @cdev: LED class device for this LED
37 * @mem: memory resource
39 * @pin: LED pin number
40 * @active_low: LED is active low
43 struct led_classdev cdev
;
50 static void bcm6358_led_write(void __iomem
*reg
, unsigned long data
)
52 iowrite32be(data
, reg
);
55 static unsigned long bcm6358_led_read(void __iomem
*reg
)
57 return ioread32be(reg
);
60 static unsigned long bcm6358_led_busy(void __iomem
*mem
)
64 while ((val
= bcm6358_led_read(mem
+ BCM6358_REG_CTRL
)) &
66 udelay(BCM6358_SLED_WAIT
);
71 static void bcm6358_led_mode(struct bcm6358_led
*led
, unsigned long value
)
75 bcm6358_led_busy(led
->mem
);
77 val
= bcm6358_led_read(led
->mem
+ BCM6358_REG_MODE
);
78 if ((led
->active_low
&& value
== LED_OFF
) ||
79 (!led
->active_low
&& value
!= LED_OFF
))
82 val
&= ~(BIT(led
->pin
));
83 bcm6358_led_write(led
->mem
+ BCM6358_REG_MODE
, val
);
86 static void bcm6358_led_set(struct led_classdev
*led_cdev
,
87 enum led_brightness value
)
89 struct bcm6358_led
*led
=
90 container_of(led_cdev
, struct bcm6358_led
, cdev
);
93 spin_lock_irqsave(led
->lock
, flags
);
94 bcm6358_led_mode(led
, value
);
95 spin_unlock_irqrestore(led
->lock
, flags
);
98 static int bcm6358_led(struct device
*dev
, struct device_node
*nc
, u32 reg
,
99 void __iomem
*mem
, spinlock_t
*lock
)
101 struct bcm6358_led
*led
;
106 led
= devm_kzalloc(dev
, sizeof(*led
), GFP_KERNEL
);
114 if (of_property_read_bool(nc
, "active-low"))
115 led
->active_low
= true;
117 led
->cdev
.name
= of_get_property(nc
, "label", NULL
) ? : nc
->name
;
118 led
->cdev
.default_trigger
= of_get_property(nc
,
119 "linux,default-trigger",
122 spin_lock_irqsave(lock
, flags
);
123 if (!of_property_read_string(nc
, "default-state", &state
)) {
124 if (!strcmp(state
, "on")) {
125 led
->cdev
.brightness
= LED_FULL
;
126 } else if (!strcmp(state
, "keep")) {
129 bcm6358_led_busy(led
->mem
);
131 val
= bcm6358_led_read(led
->mem
+ BCM6358_REG_MODE
);
132 val
&= BIT(led
->pin
);
133 if ((led
->active_low
&& !val
) ||
134 (!led
->active_low
&& val
))
135 led
->cdev
.brightness
= LED_FULL
;
137 led
->cdev
.brightness
= LED_OFF
;
139 led
->cdev
.brightness
= LED_OFF
;
142 led
->cdev
.brightness
= LED_OFF
;
144 bcm6358_led_mode(led
, led
->cdev
.brightness
);
145 spin_unlock_irqrestore(lock
, flags
);
147 led
->cdev
.brightness_set
= bcm6358_led_set
;
149 rc
= led_classdev_register(dev
, &led
->cdev
);
153 dev_dbg(dev
, "registered LED %s\n", led
->cdev
.name
);
158 static int bcm6358_leds_probe(struct platform_device
*pdev
)
160 struct device
*dev
= &pdev
->dev
;
161 struct device_node
*np
= pdev
->dev
.of_node
;
162 struct device_node
*child
;
163 struct resource
*mem_r
;
165 spinlock_t
*lock
; /* memory lock */
169 mem_r
= platform_get_resource(pdev
, IORESOURCE_MEM
, 0);
173 mem
= devm_ioremap_resource(dev
, mem_r
);
177 lock
= devm_kzalloc(dev
, sizeof(*lock
), GFP_KERNEL
);
181 spin_lock_init(lock
);
183 val
= bcm6358_led_busy(mem
);
184 val
&= ~(BCM6358_SLED_POLARITY
| BCM6358_SLED_CLKDIV_MASK
);
185 if (of_property_read_bool(np
, "brcm,clk-dat-low"))
186 val
|= BCM6358_SLED_POLARITY
;
187 of_property_read_u32(np
, "brcm,clk-div", &clk_div
);
190 val
|= BCM6358_SLED_CLKDIV_8
;
193 val
|= BCM6358_SLED_CLKDIV_4
;
196 val
|= BCM6358_SLED_CLKDIV_2
;
199 val
|= BCM6358_SLED_CLKDIV_1
;
202 bcm6358_led_write(mem
+ BCM6358_REG_CTRL
, val
);
204 for_each_available_child_of_node(np
, child
) {
208 if (of_property_read_u32(child
, "reg", ®
))
211 if (reg
>= BCM6358_SLED_MAX_COUNT
) {
212 dev_err(dev
, "invalid LED (%u >= %d)\n", reg
,
213 BCM6358_SLED_MAX_COUNT
);
217 rc
= bcm6358_led(dev
, child
, reg
, mem
, lock
);
227 static const struct of_device_id bcm6358_leds_of_match
[] = {
228 { .compatible
= "brcm,bcm6358-leds", },
231 MODULE_DEVICE_TABLE(of
, bcm6358_leds_of_match
);
233 static struct platform_driver bcm6358_leds_driver
= {
234 .probe
= bcm6358_leds_probe
,
236 .name
= "leds-bcm6358",
237 .of_match_table
= bcm6358_leds_of_match
,
241 module_platform_driver(bcm6358_leds_driver
);
243 MODULE_AUTHOR("Álvaro Fernández Rojas <noltari@gmail.com>");
244 MODULE_DESCRIPTION("LED driver for BCM6358 controllers");
245 MODULE_LICENSE("GPL v2");
246 MODULE_ALIAS("platform:leds-bcm6358");