* [PATCH 1/3] dt-bindings: leds: nxp,pca963x: add multicolor LED support
2026-06-29 19:52 [PATCH 0/3] leds: pca963x: Add multicolor support and enable Monza RGB LEDs Loic Poulain
@ 2026-06-29 19:52 ` Loic Poulain
2026-06-29 19:58 ` sashiko-bot
2026-06-30 7:24 ` Krzysztof Kozlowski
2026-06-29 19:52 ` [PATCH 2/3] leds: pca963x: add multicolor LED class support Loic Poulain
2026-06-29 19:52 ` [PATCH 3/3] arm64: dts: monaco-arduino-monza: microcontroller LEDs Loic Poulain
2 siblings, 2 replies; 9+ messages in thread
From: Loic Poulain @ 2026-06-29 19:52 UTC (permalink / raw)
To: Lee Jones, Pavel Machek, Rob Herring, Krzysztof Kozlowski,
Conor Dooley, Laurent Pinchart, Bjorn Andersson, Konrad Dybcio
Cc: linux-leds, devicetree, linux-kernel, linux-arm-msm, Loic Poulain
Add support for grouping individual PCA963x channels into a multicolor
LED by introducing a multi-led@N node pattern. This follows the
convention established by other multicolor LED drivers such as
kinetic,ktd202x.
---
.../devicetree/bindings/leds/nxp,pca963x.yaml | 83 +++++++++++++++++++++-
1 file changed, 81 insertions(+), 2 deletions(-)
diff --git a/Documentation/devicetree/bindings/leds/nxp,pca963x.yaml b/Documentation/devicetree/bindings/leds/nxp,pca963x.yaml
index 938d0e48fe51bce82779c4457c8e99cb6d80fe70..09ec140c5092950c54e1d4a55f52e11e8dd7b0f0 100644
--- a/Documentation/devicetree/bindings/leds/nxp,pca963x.yaml
+++ b/Documentation/devicetree/bindings/leds/nxp,pca963x.yaml
@@ -74,6 +74,39 @@ patternProperties:
required:
- reg
+ "^multi-led@[0-9a-f]+$":
+ type: object
+ $ref: leds-class-multicolor.yaml#
+ unevaluatedProperties: false
+
+ properties:
+ reg:
+ minimum: 0
+
+ "#address-cells":
+ const: 1
+
+ "#size-cells":
+ const: 0
+
+ patternProperties:
+ "^led@[0-9a-f]+$":
+ type: object
+ $ref: common.yaml#
+ unevaluatedProperties: false
+
+ properties:
+ reg:
+ minimum: 0
+
+ required:
+ - reg
+
+ required:
+ - reg
+ - "#address-cells"
+ - "#size-cells"
+
allOf:
- if:
properties:
@@ -84,13 +117,13 @@ allOf:
- nxp,pca9633
then:
patternProperties:
- "^led@[0-9a-f]+$":
+ "^.*led@[0-9a-f]+$":
properties:
reg:
maximum: 3
else:
patternProperties:
- "^led@[0-9a-f]+$":
+ "^.*led@[0-9a-f]+$":
properties:
reg:
maximum: 7
@@ -137,4 +170,50 @@ examples:
};
};
+ - |
+ #include <dt-bindings/leds/common.h>
+
+ i2c {
+ #address-cells = <1>;
+ #size-cells = <0>;
+
+ led-controller@62 {
+ compatible = "nxp,pca9633";
+ reg = <0x62>;
+ #address-cells = <1>;
+ #size-cells = <0>;
+
+ /* Three channels controlling one RGB LED */
+ multi-led@0 {
+ reg = <0>;
+ color = <LED_COLOR_ID_RGB>;
+ function = LED_FUNCTION_STATUS;
+ #address-cells = <1>;
+ #size-cells = <0>;
+
+ led@0 {
+ reg = <0>;
+ color = <LED_COLOR_ID_RED>;
+ };
+
+ led@1 {
+ reg = <1>;
+ color = <LED_COLOR_ID_GREEN>;
+ };
+
+ led@2 {
+ reg = <2>;
+ color = <LED_COLOR_ID_BLUE>;
+ };
+ };
+
+ /* Remaining channel used as a plain white LED */
+ led@3 {
+ reg = <3>;
+ color = <LED_COLOR_ID_WHITE>;
+ function = LED_FUNCTION_STATUS;
+ };
+ };
+ };
+
...
--
2.34.1
^ permalink raw reply related [flat|nested] 9+ messages in thread* [PATCH 2/3] leds: pca963x: add multicolor LED class support
2026-06-29 19:52 [PATCH 0/3] leds: pca963x: Add multicolor support and enable Monza RGB LEDs Loic Poulain
2026-06-29 19:52 ` [PATCH 1/3] dt-bindings: leds: nxp,pca963x: add multicolor LED support Loic Poulain
@ 2026-06-29 19:52 ` Loic Poulain
2026-06-29 19:59 ` sashiko-bot
2026-06-29 19:52 ` [PATCH 3/3] arm64: dts: monaco-arduino-monza: microcontroller LEDs Loic Poulain
2 siblings, 1 reply; 9+ messages in thread
From: Loic Poulain @ 2026-06-29 19:52 UTC (permalink / raw)
To: Lee Jones, Pavel Machek, Rob Herring, Krzysztof Kozlowski,
Conor Dooley, Laurent Pinchart, Bjorn Andersson, Konrad Dybcio
Cc: linux-leds, devicetree, linux-kernel, linux-arm-msm, Loic Poulain
Allow grouping of individual PCA963x PWM channels into a single
multicolor LED device by adding support for the LED multicolor class.
A child node with sub-children is treated as a multicolor group,
others are treated as single leds, keeping full backwards compatibility.
Signed-off-by: Loic Poulain <loic.poulain@oss.qualcomm.com>
---
drivers/leds/Kconfig | 1 +
drivers/leds/leds-pca963x.c | 126 ++++++++++++++++++++++++++++++++++++++------
2 files changed, 110 insertions(+), 17 deletions(-)
diff --git a/drivers/leds/Kconfig b/drivers/leds/Kconfig
index f4a0a3c8c8705e0f10ba26584277dbb2d5eac5b5..14df88f92b12bbe43908b67f9480cf23056e27e2 100644
--- a/drivers/leds/Kconfig
+++ b/drivers/leds/Kconfig
@@ -596,6 +596,7 @@ config LEDS_PCA963X
tristate "LED support for PCA963x I2C chip"
depends on LEDS_CLASS
depends on I2C
+ select LEDS_CLASS_MULTICOLOR
help
This option enables support for LEDs connected to the PCA963x
LED driver chip accessed via the I2C bus. Supported
diff --git a/drivers/leds/leds-pca963x.c b/drivers/leds/leds-pca963x.c
index e3a81c60ee27c96e5050a829523dfd43e1f0663f..f6f6bafcc2bd5bad51a3184c4cb08fc50693a0a5 100644
--- a/drivers/leds/leds-pca963x.c
+++ b/drivers/leds/leds-pca963x.c
@@ -27,6 +27,7 @@
#include <linux/string.h>
#include <linux/ctype.h>
#include <linux/leds.h>
+#include <linux/led-class-multicolor.h>
#include <linux/err.h>
#include <linux/i2c.h>
#include <linux/property.h>
@@ -101,8 +102,11 @@ struct pca963x;
struct pca963x_led {
struct pca963x *chip;
struct led_classdev led_cdev;
+ struct led_classdev_mc mc_cdev;
+ struct mc_subled subleds[4];
int led_num; /* 0 .. 15 potentially */
bool blinking;
+ bool is_mc;
u8 gdc;
u8 gfrq;
};
@@ -205,7 +209,7 @@ static int pca963x_power_state(struct pca963x_led *led)
unsigned long *leds_on = &led->chip->leds_on;
unsigned long cached_leds = *leds_on;
- if (led->led_cdev.brightness)
+ if (led->is_mc ? led->mc_cdev.led_cdev.brightness : led->led_cdev.brightness)
set_bit(led->led_num, leds_on);
else
clear_bit(led->led_num, leds_on);
@@ -237,6 +241,28 @@ static int pca963x_led_set(struct led_classdev *led_cdev,
return ret;
}
+static int pca963x_led_mc_set(struct led_classdev *led_cdev,
+ enum led_brightness value)
+{
+ struct led_classdev_mc *mc_cdev = lcdev_to_mccdev(led_cdev);
+ struct pca963x_led *led = container_of(mc_cdev, struct pca963x_led, mc_cdev);
+ unsigned int i;
+ int ret;
+
+ led_mc_calc_color_components(mc_cdev, value);
+
+ guard(mutex)(&led->chip->mutex);
+
+ for (i = 0; i < mc_cdev->num_colors; i++) {
+ led->led_num = mc_cdev->subled_info[i].channel;
+ ret = pca963x_brightness(led, mc_cdev->subled_info[i].brightness);
+ if (ret < 0)
+ return ret;
+ }
+
+ return pca963x_power_state(led);
+}
+
static unsigned int pca963x_period_scale(struct pca963x_led *led,
unsigned int val)
{
@@ -300,6 +326,84 @@ static int pca963x_blink_set(struct led_classdev *led_cdev,
return 0;
}
+static int pca963x_register_single_led(struct device *dev,
+ struct pca963x_led *led, u32 reg,
+ struct fwnode_handle *fwnode,
+ bool hw_blink)
+{
+ struct led_init_data init_data = {};
+ char default_label[32];
+ struct i2c_client *client = led->chip->client;
+
+ led->led_num = reg;
+ led->is_mc = false;
+ led->led_cdev.brightness_set_blocking = pca963x_led_set;
+ if (hw_blink)
+ led->led_cdev.blink_set = pca963x_blink_set;
+
+ init_data.fwnode = fwnode;
+ init_data.devicename = "pca963x";
+ snprintf(default_label, sizeof(default_label), "%d:%.2x:%u",
+ client->adapter->nr, client->addr, reg);
+ init_data.default_label = default_label;
+
+ return devm_led_classdev_register_ext(dev, &led->led_cdev, &init_data);
+}
+
+static int pca963x_register_mc_led(struct device *dev,
+ struct pca963x_led *led, u32 reg,
+ struct fwnode_handle *fwnode,
+ const struct pca963x_chipdef *chipdef)
+{
+ struct mc_subled *subleds = led->subleds;
+ unsigned int num_colors = 0;
+ struct led_init_data init_data = {};
+ char default_label[32];
+ struct i2c_client *client = led->chip->client;
+ int ret;
+
+ fwnode_for_each_child_node_scoped(fwnode, sub) {
+ u32 color, subreg;
+
+ if (num_colors >= ARRAY_SIZE(led->subleds)) {
+ dev_err(dev, "Too many sub-LEDs for node %pfw\n", fwnode);
+ return -EINVAL;
+ }
+
+ ret = fwnode_property_read_u32(sub, "reg", &subreg);
+ if (ret || subreg >= chipdef->n_leds) {
+ dev_err(dev, "Invalid 'reg' for sub-LED %pfw\n", sub);
+ return -EINVAL;
+ }
+
+ ret = fwnode_property_read_u32(sub, "color", &color);
+ if (ret) {
+ dev_err(dev, "Missing 'color' for sub-LED %pfw\n", sub);
+ return ret;
+ }
+
+ subleds[num_colors].channel = subreg;
+ subleds[num_colors].color_index = color;
+ subleds[num_colors].intensity = LED_FULL;
+ num_colors++;
+ }
+
+ led->led_num = reg;
+ led->is_mc = true;
+ led->mc_cdev.subled_info = subleds;
+ led->mc_cdev.num_colors = num_colors;
+ led->mc_cdev.led_cdev.max_brightness = LED_FULL;
+ led->mc_cdev.led_cdev.brightness_set_blocking = pca963x_led_mc_set;
+
+ init_data.fwnode = fwnode;
+ init_data.devicename = "pca963x";
+ snprintf(default_label, sizeof(default_label), "%d:%.2x:%u",
+ client->adapter->nr, client->addr, reg);
+ init_data.default_label = default_label;
+
+ return devm_led_classdev_multicolor_register_ext(dev, &led->mc_cdev, &init_data);
+}
+
static int pca963x_register_leds(struct i2c_client *client,
struct pca963x *chip)
{
@@ -338,9 +442,6 @@ static int pca963x_register_leds(struct i2c_client *client,
return ret;
device_for_each_child_node_scoped(dev, child) {
- struct led_init_data init_data = {};
- char default_label[32];
-
ret = fwnode_property_read_u32(child, "reg", ®);
if (ret || reg >= chipdef->n_leds) {
dev_err(dev, "Invalid 'reg' property for node %pfw\n",
@@ -348,22 +449,13 @@ static int pca963x_register_leds(struct i2c_client *client,
return -EINVAL;
}
- led->led_num = reg;
led->chip = chip;
- led->led_cdev.brightness_set_blocking = pca963x_led_set;
- if (hw_blink)
- led->led_cdev.blink_set = pca963x_blink_set;
led->blinking = false;
- init_data.fwnode = child;
- /* for backwards compatibility */
- init_data.devicename = "pca963x";
- snprintf(default_label, sizeof(default_label), "%d:%.2x:%u",
- client->adapter->nr, client->addr, reg);
- init_data.default_label = default_label;
-
- ret = devm_led_classdev_register_ext(dev, &led->led_cdev,
- &init_data);
+ if (fwnode_get_child_node_count(child) > 0)
+ ret = pca963x_register_mc_led(dev, led, reg, child, chipdef);
+ else
+ ret = pca963x_register_single_led(dev, led, reg, child, hw_blink);
if (ret) {
dev_err(dev, "Failed to register LED for node %pfw\n",
child);
--
2.34.1
^ permalink raw reply related [flat|nested] 9+ messages in thread* [PATCH 3/3] arm64: dts: monaco-arduino-monza: microcontroller LEDs
2026-06-29 19:52 [PATCH 0/3] leds: pca963x: Add multicolor support and enable Monza RGB LEDs Loic Poulain
2026-06-29 19:52 ` [PATCH 1/3] dt-bindings: leds: nxp,pca963x: add multicolor LED support Loic Poulain
2026-06-29 19:52 ` [PATCH 2/3] leds: pca963x: add multicolor LED class support Loic Poulain
@ 2026-06-29 19:52 ` Loic Poulain
2026-06-29 19:58 ` sashiko-bot
2026-06-30 9:09 ` Konrad Dybcio
2 siblings, 2 replies; 9+ messages in thread
From: Loic Poulain @ 2026-06-29 19:52 UTC (permalink / raw)
To: Lee Jones, Pavel Machek, Rob Herring, Krzysztof Kozlowski,
Conor Dooley, Laurent Pinchart, Bjorn Andersson, Konrad Dybcio
Cc: linux-leds, devicetree, linux-kernel, linux-arm-msm, Loic Poulain
Onboard MCU/STM32 implements a led controller compatible with PCA9635.
There are four RGB LEDs controlled via channels 0-2, 3-5, 6-8 and 9-11.
---
arch/arm64/boot/dts/qcom/monaco-arduino-monza.dts | 110 ++++++++++++++++++++++
1 file changed, 110 insertions(+)
diff --git a/arch/arm64/boot/dts/qcom/monaco-arduino-monza.dts b/arch/arm64/boot/dts/qcom/monaco-arduino-monza.dts
index 379b796f261f4c8f6b7712c5d340b20be1b9217c..71685e54a3781c9b03fc41cacecfea77650a7182 100644
--- a/arch/arm64/boot/dts/qcom/monaco-arduino-monza.dts
+++ b/arch/arm64/boot/dts/qcom/monaco-arduino-monza.dts
@@ -7,6 +7,7 @@
#include <dt-bindings/gpio/gpio.h>
#include <dt-bindings/input/input.h>
+#include <dt-bindings/leds/common.h>
#include <dt-bindings/sound/qcom,q6dsp-lpass-ports.h>
#include "monaco.dtsi"
@@ -156,6 +157,115 @@ vreg_nvme: regulator-3p3-m2 {
};
};
+&cci1 {
+ status = "okay";
+};
+
+&cci1_i2c1 {
+ leds_controller: leds-controller@22 {
+ compatible = "nxp,pca9635";
+ #address-cells = <1>;
+ #size-cells = <0>;
+ reg = <0x22>;
+
+ multi-led@0 {
+ reg = <0>;
+ label = "stm-led-1";
+ color = <LED_COLOR_ID_RGB>;
+ function = LED_FUNCTION_STATUS;
+ #address-cells = <1>;
+ #size-cells = <0>;
+
+ led@0 {
+ reg = <0>;
+ color = <LED_COLOR_ID_RED>;
+ };
+
+ led@1 {
+ reg = <1>;
+ color = <LED_COLOR_ID_GREEN>;
+ };
+
+ led@2 {
+ reg = <2>;
+ color = <LED_COLOR_ID_BLUE>;
+ };
+ };
+
+ multi-led@1 {
+ reg = <1>;
+ label = "stm-led-2";
+ color = <LED_COLOR_ID_RGB>;
+ function = LED_FUNCTION_STATUS;
+ #address-cells = <1>;
+ #size-cells = <0>;
+
+ led@3 {
+ reg = <3>;
+ color = <LED_COLOR_ID_RED>;
+ };
+
+ led@4 {
+ reg = <4>;
+ color = <LED_COLOR_ID_GREEN>;
+ };
+
+ led@5 {
+ reg = <5>;
+ color = <LED_COLOR_ID_BLUE>;
+ };
+ };
+
+ multi-led@2 {
+ reg = <2>;
+ label = "stm-led-3";
+ color = <LED_COLOR_ID_RGB>;
+ function = LED_FUNCTION_STATUS;
+ #address-cells = <1>;
+ #size-cells = <0>;
+
+ led@6 {
+ reg = <6>;
+ color = <LED_COLOR_ID_RED>;
+ };
+
+ led@7 {
+ reg = <7>;
+ color = <LED_COLOR_ID_GREEN>;
+ };
+
+ led@8 {
+ reg = <8>;
+ color = <LED_COLOR_ID_BLUE>;
+ };
+ };
+
+ multi-led@3 {
+ reg = <3>;
+ label = "stm-led-4";
+ color = <LED_COLOR_ID_RGB>;
+ function = LED_FUNCTION_STATUS;
+ #address-cells = <1>;
+ #size-cells = <0>;
+
+ led@9 {
+ reg = <9>;
+ color = <LED_COLOR_ID_RED>;
+ };
+
+ led@10 {
+ reg = <10>;
+ color = <LED_COLOR_ID_GREEN>;
+ };
+
+ led@11 {
+ reg = <11>;
+ color = <LED_COLOR_ID_BLUE>;
+ };
+ };
+ };
+};
+
ðernet0 {
phy-mode = "2500base-x";
phy-handle = <&hsgmii_phy0>;
--
2.34.1
^ permalink raw reply related [flat|nested] 9+ messages in thread