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
;
99 enum led_default_state state
;
103 led
= devm_kzalloc(dev
, sizeof(*led
), GFP_KERNEL
);
111 if (of_property_read_bool(nc
, "active-low"))
112 led
->active_low
= true;
114 init_data
.fwnode
= of_fwnode_handle(nc
);
116 state
= led_init_default_state_get(init_data
.fwnode
);
118 case LEDS_DEFSTATE_ON
:
119 led
->cdev
.brightness
= LED_FULL
;
121 case LEDS_DEFSTATE_KEEP
:
122 val
= bcm6358_led_read(led
->mem
+ BCM6358_REG_MODE
);
123 val
&= BIT(led
->pin
);
124 if ((led
->active_low
&& !val
) || (!led
->active_low
&& val
))
125 led
->cdev
.brightness
= LED_FULL
;
127 led
->cdev
.brightness
= LED_OFF
;
130 led
->cdev
.brightness
= LED_OFF
;
133 bcm6358_led_set(&led
->cdev
, led
->cdev
.brightness
);
135 led
->cdev
.brightness_set
= bcm6358_led_set
;
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
);
151 spinlock_t
*lock
; /* memory lock */
155 mem
= devm_platform_ioremap_resource(pdev
, 0);
159 lock
= devm_kzalloc(dev
, sizeof(*lock
), GFP_KERNEL
);
163 spin_lock_init(lock
);
165 val
= bcm6358_led_busy(mem
);
166 val
&= ~(BCM6358_SLED_POLARITY
| BCM6358_SLED_CLKDIV_MASK
);
167 if (of_property_read_bool(np
, "brcm,clk-dat-low"))
168 val
|= BCM6358_SLED_POLARITY
;
169 of_property_read_u32(np
, "brcm,clk-div", &clk_div
);
172 val
|= BCM6358_SLED_CLKDIV_8
;
175 val
|= BCM6358_SLED_CLKDIV_4
;
178 val
|= BCM6358_SLED_CLKDIV_2
;
181 val
|= BCM6358_SLED_CLKDIV_1
;
184 bcm6358_led_write(mem
+ BCM6358_REG_CTRL
, val
);
186 for_each_available_child_of_node_scoped(np
, child
) {
190 if (of_property_read_u32(child
, "reg", ®
))
193 if (reg
>= BCM6358_SLED_MAX_COUNT
) {
194 dev_err(dev
, "invalid LED (%u >= %d)\n", reg
,
195 BCM6358_SLED_MAX_COUNT
);
199 rc
= bcm6358_led(dev
, child
, reg
, mem
, lock
);
207 static const struct of_device_id bcm6358_leds_of_match
[] = {
208 { .compatible
= "brcm,bcm6358-leds", },
211 MODULE_DEVICE_TABLE(of
, bcm6358_leds_of_match
);
213 static struct platform_driver bcm6358_leds_driver
= {
214 .probe
= bcm6358_leds_probe
,
216 .name
= "leds-bcm6358",
217 .of_match_table
= bcm6358_leds_of_match
,
221 module_platform_driver(bcm6358_leds_driver
);
223 MODULE_AUTHOR("Álvaro Fernández Rojas <noltari@gmail.com>");
224 MODULE_DESCRIPTION("LED driver for BCM6358 controllers");
225 MODULE_LICENSE("GPL v2");
226 MODULE_ALIAS("platform:leds-bcm6358");