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 led_init_data init_data
= {};
98 struct bcm6358_led
*led
;
102 led
= devm_kzalloc(dev
, sizeof(*led
), GFP_KERNEL
);
110 if (of_property_read_bool(nc
, "active-low"))
111 led
->active_low
= true;
113 if (!of_property_read_string(nc
, "default-state", &state
)) {
114 if (!strcmp(state
, "on")) {
115 led
->cdev
.brightness
= LED_FULL
;
116 } else if (!strcmp(state
, "keep")) {
118 val
= bcm6358_led_read(led
->mem
+ BCM6358_REG_MODE
);
119 val
&= BIT(led
->pin
);
120 if ((led
->active_low
&& !val
) ||
121 (!led
->active_low
&& val
))
122 led
->cdev
.brightness
= LED_FULL
;
124 led
->cdev
.brightness
= LED_OFF
;
126 led
->cdev
.brightness
= LED_OFF
;
129 led
->cdev
.brightness
= LED_OFF
;
132 bcm6358_led_set(&led
->cdev
, led
->cdev
.brightness
);
134 led
->cdev
.brightness_set
= bcm6358_led_set
;
135 init_data
.fwnode
= of_fwnode_handle(nc
);
137 rc
= devm_led_classdev_register_ext(dev
, &led
->cdev
, &init_data
);
141 dev_dbg(dev
, "registered LED %s\n", led
->cdev
.name
);
146 static int bcm6358_leds_probe(struct platform_device
*pdev
)
148 struct device
*dev
= &pdev
->dev
;
149 struct device_node
*np
= dev_of_node(&pdev
->dev
);
150 struct device_node
*child
;
152 spinlock_t
*lock
; /* memory lock */
156 mem
= devm_platform_ioremap_resource(pdev
, 0);
160 lock
= devm_kzalloc(dev
, sizeof(*lock
), GFP_KERNEL
);
164 spin_lock_init(lock
);
166 val
= bcm6358_led_busy(mem
);
167 val
&= ~(BCM6358_SLED_POLARITY
| BCM6358_SLED_CLKDIV_MASK
);
168 if (of_property_read_bool(np
, "brcm,clk-dat-low"))
169 val
|= BCM6358_SLED_POLARITY
;
170 of_property_read_u32(np
, "brcm,clk-div", &clk_div
);
173 val
|= BCM6358_SLED_CLKDIV_8
;
176 val
|= BCM6358_SLED_CLKDIV_4
;
179 val
|= BCM6358_SLED_CLKDIV_2
;
182 val
|= BCM6358_SLED_CLKDIV_1
;
185 bcm6358_led_write(mem
+ BCM6358_REG_CTRL
, val
);
187 for_each_available_child_of_node(np
, child
) {
191 if (of_property_read_u32(child
, "reg", ®
))
194 if (reg
>= BCM6358_SLED_MAX_COUNT
) {
195 dev_err(dev
, "invalid LED (%u >= %d)\n", reg
,
196 BCM6358_SLED_MAX_COUNT
);
200 rc
= bcm6358_led(dev
, child
, reg
, mem
, lock
);
210 static const struct of_device_id bcm6358_leds_of_match
[] = {
211 { .compatible
= "brcm,bcm6358-leds", },
214 MODULE_DEVICE_TABLE(of
, bcm6358_leds_of_match
);
216 static struct platform_driver bcm6358_leds_driver
= {
217 .probe
= bcm6358_leds_probe
,
219 .name
= "leds-bcm6358",
220 .of_match_table
= bcm6358_leds_of_match
,
224 module_platform_driver(bcm6358_leds_driver
);
226 MODULE_AUTHOR("Álvaro Fernández Rojas <noltari@gmail.com>");
227 MODULE_DESCRIPTION("LED driver for BCM6358 controllers");
228 MODULE_LICENSE("GPL v2");
229 MODULE_ALIAS("platform:leds-bcm6358");