1 // SPDX-License-Identifier: GPL-2.0-or-later
3 * Driver for BCM6358 memory-mapped LEDs, based on leds-syscon.c
5 * Copyright 2015 Álvaro Fernández Rojas <noltari@gmail.com>
7 #include <linux/delay.h>
9 #include <linux/leds.h>
10 #include <linux/module.h>
12 #include <linux/platform_device.h>
13 #include <linux/spinlock.h>
15 #define BCM6358_REG_MODE 0x0
16 #define BCM6358_REG_CTRL 0x4
18 #define BCM6358_SLED_CLKDIV_MASK 3
19 #define BCM6358_SLED_CLKDIV_1 0
20 #define BCM6358_SLED_CLKDIV_2 1
21 #define BCM6358_SLED_CLKDIV_4 2
22 #define BCM6358_SLED_CLKDIV_8 3
24 #define BCM6358_SLED_POLARITY BIT(2)
25 #define BCM6358_SLED_BUSY BIT(3)
27 #define BCM6358_SLED_MAX_COUNT 32
28 #define BCM6358_SLED_WAIT 100
31 * struct bcm6358_led - state container for bcm6358 based LEDs
32 * @cdev: LED class device for this LED
33 * @mem: memory resource
35 * @pin: LED pin number
36 * @active_low: LED is active low
39 struct led_classdev cdev
;
46 static void bcm6358_led_write(void __iomem
*reg
, unsigned long data
)
48 #ifdef CONFIG_CPU_BIG_ENDIAN
49 iowrite32be(data
, reg
);
55 static unsigned long bcm6358_led_read(void __iomem
*reg
)
57 #ifdef CONFIG_CPU_BIG_ENDIAN
58 return ioread32be(reg
);
64 static unsigned long bcm6358_led_busy(void __iomem
*mem
)
68 while ((val
= bcm6358_led_read(mem
+ BCM6358_REG_CTRL
)) &
70 udelay(BCM6358_SLED_WAIT
);
75 static void bcm6358_led_set(struct led_classdev
*led_cdev
,
76 enum led_brightness value
)
78 struct bcm6358_led
*led
=
79 container_of(led_cdev
, struct bcm6358_led
, cdev
);
80 unsigned long flags
, val
;
82 spin_lock_irqsave(led
->lock
, flags
);
83 bcm6358_led_busy(led
->mem
);
84 val
= bcm6358_led_read(led
->mem
+ BCM6358_REG_MODE
);
85 if ((led
->active_low
&& value
== LED_OFF
) ||
86 (!led
->active_low
&& value
!= LED_OFF
))
89 val
&= ~(BIT(led
->pin
));
90 bcm6358_led_write(led
->mem
+ BCM6358_REG_MODE
, val
);
91 spin_unlock_irqrestore(led
->lock
, flags
);
94 static int bcm6358_led(struct device
*dev
, struct device_node
*nc
, u32 reg
,
95 void __iomem
*mem
, spinlock_t
*lock
)
97 struct bcm6358_led
*led
;
101 led
= devm_kzalloc(dev
, sizeof(*led
), GFP_KERNEL
);
109 if (of_property_read_bool(nc
, "active-low"))
110 led
->active_low
= true;
112 led
->cdev
.name
= of_get_property(nc
, "label", NULL
) ? : nc
->name
;
113 led
->cdev
.default_trigger
= of_get_property(nc
,
114 "linux,default-trigger",
117 if (!of_property_read_string(nc
, "default-state", &state
)) {
118 if (!strcmp(state
, "on")) {
119 led
->cdev
.brightness
= LED_FULL
;
120 } else if (!strcmp(state
, "keep")) {
122 val
= bcm6358_led_read(led
->mem
+ BCM6358_REG_MODE
);
123 val
&= BIT(led
->pin
);
124 if ((led
->active_low
&& !val
) ||
125 (!led
->active_low
&& val
))
126 led
->cdev
.brightness
= LED_FULL
;
128 led
->cdev
.brightness
= LED_OFF
;
130 led
->cdev
.brightness
= LED_OFF
;
133 led
->cdev
.brightness
= LED_OFF
;
136 bcm6358_led_set(&led
->cdev
, led
->cdev
.brightness
);
138 led
->cdev
.brightness_set
= bcm6358_led_set
;
140 rc
= led_classdev_register(dev
, &led
->cdev
);
144 dev_dbg(dev
, "registered LED %s\n", led
->cdev
.name
);
149 static int bcm6358_leds_probe(struct platform_device
*pdev
)
151 struct device
*dev
= &pdev
->dev
;
152 struct device_node
*np
= pdev
->dev
.of_node
;
153 struct device_node
*child
;
154 struct resource
*mem_r
;
156 spinlock_t
*lock
; /* memory lock */
160 mem_r
= platform_get_resource(pdev
, IORESOURCE_MEM
, 0);
164 mem
= devm_ioremap_resource(dev
, mem_r
);
168 lock
= devm_kzalloc(dev
, sizeof(*lock
), GFP_KERNEL
);
172 spin_lock_init(lock
);
174 val
= bcm6358_led_busy(mem
);
175 val
&= ~(BCM6358_SLED_POLARITY
| BCM6358_SLED_CLKDIV_MASK
);
176 if (of_property_read_bool(np
, "brcm,clk-dat-low"))
177 val
|= BCM6358_SLED_POLARITY
;
178 of_property_read_u32(np
, "brcm,clk-div", &clk_div
);
181 val
|= BCM6358_SLED_CLKDIV_8
;
184 val
|= BCM6358_SLED_CLKDIV_4
;
187 val
|= BCM6358_SLED_CLKDIV_2
;
190 val
|= BCM6358_SLED_CLKDIV_1
;
193 bcm6358_led_write(mem
+ BCM6358_REG_CTRL
, val
);
195 for_each_available_child_of_node(np
, child
) {
199 if (of_property_read_u32(child
, "reg", ®
))
202 if (reg
>= BCM6358_SLED_MAX_COUNT
) {
203 dev_err(dev
, "invalid LED (%u >= %d)\n", reg
,
204 BCM6358_SLED_MAX_COUNT
);
208 rc
= bcm6358_led(dev
, child
, reg
, mem
, lock
);
218 static const struct of_device_id bcm6358_leds_of_match
[] = {
219 { .compatible
= "brcm,bcm6358-leds", },
222 MODULE_DEVICE_TABLE(of
, bcm6358_leds_of_match
);
224 static struct platform_driver bcm6358_leds_driver
= {
225 .probe
= bcm6358_leds_probe
,
227 .name
= "leds-bcm6358",
228 .of_match_table
= bcm6358_leds_of_match
,
232 module_platform_driver(bcm6358_leds_driver
);
234 MODULE_AUTHOR("Álvaro Fernández Rojas <noltari@gmail.com>");
235 MODULE_DESCRIPTION("LED driver for BCM6358 controllers");
236 MODULE_LICENSE("GPL v2");
237 MODULE_ALIAS("platform:leds-bcm6358");