Message ID | 1584665829-4881-6-git-send-email-alex.nemirovsky@cortina-access.com |
---|---|
State | New |
Headers | show |
Series | Cortina Access Drivers Package 2 | expand |
Hi Alex, On Thu, 19 Mar 2020 at 18:57, Alex Nemirovsky <alex.nemirovsky at cortina-access.com> wrote: > > From: Jway Lin <jway.lin at cortina-access.com> > > Add Cortina Access LED controller support for CAxxxx SOCs > > Signed-off-by: Jway Lin <jway.lin at cortina-access.com> > Signed-off-by: Alex Nemirovsky <alex.nemirovsky at cortina-access.com> > CC: Simon Glass <sjg at chromium.org> > --- > > Changes in v3: None > Changes in v2: None > > MAINTAINERS | 2 + > drivers/led/Kconfig | 8 ++ > drivers/led/Makefile | 1 + > drivers/led/led_cortina.c | 308 ++++++++++++++++++++++++++++++++++++++++++++++ > 4 files changed, 319 insertions(+) > create mode 100644 drivers/led/led_cortina.c > > diff --git a/MAINTAINERS b/MAINTAINERS > index b147faa..24a2655 100644 > --- a/MAINTAINERS > +++ b/MAINTAINERS > @@ -183,6 +183,7 @@ F: drivers/serial/serial_cortina.c > F: drivers/mmc/ca_dw_mmc.c > F: drivers/i2c/i2c-cortina.c > F: drivers/i2c/i2c-cortina.h > +F: drivers/led/led_cortina.c > > ARM/CZ.NIC TURRIS MOX SUPPORT > M: Marek Behun <marek.behun at nic.cz> > @@ -676,6 +677,7 @@ F: drivers/serial/serial_cortina.c > F: drivers/mmc/ca_dw_mmc.c > F: drivers/i2c/i2c-cortina.c > F: drivers/i2c/i2c-cortina.h > +F: drivers/led/led_cortina.c > > MIPS MSCC > M: Gregory CLEMENT <gregory.clement at bootlin.com> > diff --git a/drivers/led/Kconfig b/drivers/led/Kconfig > index 6675934..cc87fbf 100644 > --- a/drivers/led/Kconfig > +++ b/drivers/led/Kconfig > @@ -35,6 +35,14 @@ config LED_BCM6858 > This option enables support for LEDs connected to the BCM6858 > HW has blinking capabilities and up to 32 LEDs can be controlled. > > +config LED_CORTINA > + bool "LED Support for Cortina Access CAxxxx SoCs" > + depends on LED && (CORTINA_PLATFORM) > + help > + This option enables support for LEDs connected to the Cortina > + Access CAxxxx SOCs. > + > + > config LED_BLINK > bool "Support LED blinking" > depends on LED > diff --git a/drivers/led/Makefile b/drivers/led/Makefile > index 3654dd3..8e3ae7f 100644 > --- a/drivers/led/Makefile > +++ b/drivers/led/Makefile > @@ -8,3 +8,4 @@ obj-$(CONFIG_LED_BCM6328) += led_bcm6328.o > obj-$(CONFIG_LED_BCM6358) += led_bcm6358.o > obj-$(CONFIG_LED_BCM6858) += led_bcm6858.o > obj-$(CONFIG_$(SPL_)LED_GPIO) += led_gpio.o > +obj-$(CONFIG_LED_CORTINA) += led_cortina.o > diff --git a/drivers/led/led_cortina.c b/drivers/led/led_cortina.c > new file mode 100644 > index 0000000..53435e8 > --- /dev/null > +++ b/drivers/led/led_cortina.c > @@ -0,0 +1,308 @@ > +// SPDX-License-Identifier: GPL-2.0+ > + > +/* > + * Copyright (C) 2020 Cortina-Access > + * Author: Jway Lin <jway.lin at cortina-access.com> > + * > + */ > + > +#include <common.h> > +#include <dm.h> > +#include <errno.h> > +#include <led.h> > +#include <asm/io.h> > +#include <dm/lists.h> > +#include <linux/compat.h> > + > +#define CORTINA_LED_NUM 16 > + > +#define BIT(nr) (1UL << (nr)) That is already defined somewhere > + > +#define cortina_LED_CONTROL 0x00 Drop the cortina_ prefix if you can.. It should be upper case. Since this is local to the file it just adds code. > +#define cortina_LED_CONFIG_0 0x04 > +#define cortina_LED_CONFIG_1 0x08 > +#define cortina_LED_CONFIG_2 0x0c > +#define cortina_LED_CONFIG_3 0x10 > +#define cortina_LED_CONFIG_4 0x14 > +#define cortina_LED_CONFIG_5 0x18 > +#define cortina_LED_CONFIG_6 0x1c > +#define cortina_LED_CONFIG_7 0x20 > +#define cortina_LED_CONFIG_8 0x24 > +#define cortina_LED_CONFIG_9 0x28 > +#define cortina_LED_CONFIG_10 0x2c > +#define cortina_LED_CONFIG_11 0x30 > +#define cortina_LED_CONFIG_12 0x34 > +#define cortina_LED_CONFIG_13 0x38 > +#define cortina_LED_CONFIG_14 0x3c > +#define cortina_LED_CONFIG_15 0x40 What are all those used for? > + > +#define cortina_LED_MAX_HW_BLINK 127 > +#define cortina_LED_MAX_COUNT CORTINA_LED_NUM > +#define cortina_LED_MAX_PORT 8 > + > +/* LED_CONTROL fields */ > +#define cortina_LED_BLINK_RATE1_OFFSET 0 > +#define cortina_LED_BLINK_RATE1_MASK 0xFF Lower-case hex > +#define cortina_LED_BLINK_RATE2_OFFSET 8 > +#define cortina_LED_BLINK_RATE2_MASK 0xFF > +#define cortina_LED_CLK_TEST BIT(16) > +#define cortina_LED_CLK_POLARITY BIT(17) > +#define cortina_LED_CLK_TEST_MODE BIT(16) > +#define cortina_LED_CLK_TEST_RX_TEST BIT(30) > +#define cortina_LED_CLK_TEST_TX_TEST BIT(31) > + > +/* LED_CONFIG fields */ > +#define cortina_LED_EVENT_ON_OFFSET 0 > +#define cortina_LED_EVENT_ON_MASK 0x7 > +#define cortina_LED_EVENT_BLINK_OFFSET 3 > +#define cortina_LED_EVENT_BLINK_MASK 0x7 > +#define cortina_LED_EVENT_OFF_OFFSET 6 > +#define cortina_LED_EVENT_OFF_MASK 0x7 > +#define cortina_LED_OFF_ON_OFFSET 9 > +#define cortina_LED_OFF_ON_MASK 0x3 > +#define cortina_LED_PORT_OFFSET 11 > +#define cortina_LED_PORT_MASK 0x7 > +#define cortina_LED_OFF_VAL BIT(14) > +#define cortina_LED_SW_EVENT BIT(15) > +#define cortina_LED_BLINK_SEL BIT(16) > + Need a struct comment > +struct cortina_led_cfg { > + void __iomem *regs; > + spinlock_t *lock; /* protect LED resource access */ > + int idx; > + bool active_low; > + > + int off_event; > + int blink_event; > + int on_event; > + int port; > + int blink; > + int enable; > +}; > + > +struct cortina_led_top_cfg { here too > + void __iomem *regs; > + u16 blink_rate1; > + u16 blink_rate2; > +}; > + > +static struct cortina_led_top_cfg glb_led_ctrl; You are not allowed to use BSS in driver model. It's not clear why you have two structures as there are no comments. Can you not combine them? > + > +static void cortina_led_write(void __iomem *reg, unsigned long data) > +{ > + writel(data, reg); > +} > + > +static unsigned long cortina_led_read(void __iomem *reg) > +{ > + return readl(reg); > +} > + > +static enum led_state_t cortina_led_get_state(struct udevice *dev) > +{ > + struct cortina_led_cfg *priv = dev_get_priv(dev); > + enum led_state_t state = LEDST_OFF; > + u32 val; > + > + val = readl(priv->regs); > + > + if (val & cortina_LED_SW_EVENT) > + state = LEDST_ON; > + > + return state; > +} > + > +static int cortina_led_set_state(struct udevice *dev, enum led_state_t state) > +{ > + u32 val; > + struct cortina_led_cfg *priv = dev_get_priv(dev); > + > + val = readl(priv->regs); > + > + switch (state) { > + case LEDST_OFF: > + val &= ~cortina_LED_SW_EVENT; > + val &= ~(cortina_LED_OFF_ON_MASK << cortina_LED_OFF_ON_OFFSET); > + val |= 0x3 << cortina_LED_OFF_ON_OFFSET; > + cortina_led_write(priv->regs, val); > + break; > + case LEDST_ON: > + val |= cortina_LED_SW_EVENT; > + val &= ~(cortina_LED_OFF_ON_MASK << cortina_LED_OFF_ON_OFFSET); > + val |= 0x1 << cortina_LED_OFF_ON_OFFSET; > + cortina_led_write(priv->regs, val); > + break; > + case LEDST_TOGGLE: > + if (cortina_led_get_state(dev) == LEDST_OFF) > + return cortina_led_set_state(dev, LEDST_ON); > + else > + return cortina_led_set_state(dev, LEDST_OFF); > + break; > + > + default: > + return -EINVAL; > + } > + > + return 0; > +} > + > +static const struct led_ops cortina_led_ops = { > + .get_state = cortina_led_get_state, > + .set_state = cortina_led_set_state, > +}; > + > +static int cortina_led_probe(struct udevice *dev) > +{ > + struct led_uc_plat *uc_plat = dev_get_uclass_platdata(dev); > + void __iomem *regs; > + u32 reg_value, val; > + u16 rate1, rate2; > + struct cortina_led_cfg *priv = dev_get_priv(dev); > + unsigned int pin; > + u32 blink, port, off_event, blink_event, on_event; > + > + /* Top-level LED node */ > + if (!uc_plat->label) { > + regs = dev_remap_addr(dev); > + > + if (!regs) > + return -EINVAL; > + > + glb_led_ctrl.regs = regs; > + > + reg_value = 0; > + reg_value |= cortina_LED_CLK_POLARITY; > + > + rate1 = dev_read_u32_default(dev, "cortina, blink_rate1", 256); > + rate2 = dev_read_u32_default(dev, "cortina, blink_rate2", 512); Strictly speaking you should read your platdata into a plat struct in your ofdata_to_platdata method. Also do you have a binding file for this? I think we sholld use hyphen for properties instead of underscore. > + > + val = rate1 / 16 - 1; > + glb_led_ctrl.blink_rate1 = val > cortina_LED_MAX_HW_BLINK ? > + cortina_LED_MAX_HW_BLINK : val; > + reg_value |= (glb_led_ctrl.blink_rate1 & cortina_LED_BLINK_RATE1_MASK) > + << cortina_LED_BLINK_RATE1_OFFSET; > + > + val = rate2 / 16 - 1; > + glb_led_ctrl.blink_rate2 = val > cortina_LED_MAX_HW_BLINK ? > + cortina_LED_MAX_HW_BLINK : val; > + reg_value |= (glb_led_ctrl.blink_rate2 & cortina_LED_BLINK_RATE2_MASK) > + << cortina_LED_BLINK_RATE2_OFFSET; > + > + cortina_led_write(glb_led_ctrl.regs, reg_value); > + } else { > + regs = dev_remap_addr(dev_get_parent(dev)); > + > + if (!regs) > + return -EINVAL; > + > + pin = dev_read_u32_default(dev, "pin", cortina_LED_MAX_COUNT); > + > + if (pin >= cortina_LED_MAX_COUNT) > + return -EINVAL; > + > + priv->regs = regs + 4 + (pin * 4); > + > + val = cortina_led_read(priv->regs); > + > + if (dev_read_bool(dev, "active-low")) { > + priv->active_low = true; > + val |= cortina_LED_OFF_VAL; > + } else { > + priv->active_low = false; > + val &= ~cortina_LED_OFF_VAL; > + } > + > + blink = dev_read_u32_default(dev, "blink-sel", 0); > + > + if (blink == 0) { > + priv->blink = 0; > + val &= ~cortina_LED_BLINK_SEL; > + } else if (blink == 1) { > + priv->blink = 1; > + val |= cortina_LED_BLINK_SEL; > + } > + > + //Todo : cortina_led_config(); > + off_event = dev_read_u32_default(dev, "off-event", 3); > + val &= ~(cortina_LED_EVENT_OFF_MASK << cortina_LED_EVENT_OFF_OFFSET); > + if (off_event != 3) { > + priv->off_event = off_event; > + val |= BIT(off_event) << cortina_LED_EVENT_OFF_OFFSET; > + } > + > + blink_event = dev_read_u32_default(dev, "blink-event", 3); > + val &= ~(cortina_LED_EVENT_BLINK_MASK << > + cortina_LED_EVENT_BLINK_OFFSET); > + > + if (blink_event != 3) { > + priv->blink_event = blink_event; > + val |= BIT(blink_event) << cortina_LED_EVENT_BLINK_OFFSET; > + } > + > + on_event = dev_read_u32_default(dev, "on-event", 3); > + val &= ~(cortina_LED_EVENT_ON_MASK << cortina_LED_EVENT_ON_OFFSET); > + if (on_event != 3) { > + priv->on_event = on_event; > + val |= BIT(on_event) << cortina_LED_EVENT_ON_OFFSET; > + } > + > + port = dev_read_u32_default(dev, "port", 0); > + priv->port = port; > + val &= ~(cortina_LED_PORT_MASK << cortina_LED_PORT_OFFSET); > + val |= port << cortina_LED_PORT_OFFSET; > + > + priv->enable = 0; > + > + /* force off */ > + val &= ~(cortina_LED_OFF_ON_MASK << cortina_LED_OFF_ON_OFFSET); > + val |= 0x3 << cortina_LED_OFF_ON_OFFSET; > + > + cortina_led_write(priv->regs, val); > + } funny indent? > + > + return 0; > +} > + > +static int cortina_led_bind(struct udevice *parent) > +{ > + ofnode node; > + > + dev_for_each_subnode(node, parent) { > + struct led_uc_plat *uc_plat; > + struct udevice *dev; > + const char *label; > + int ret; > + > + label = ofnode_read_string(node, "label"); > + if (!label) { > + debug("%s: node %s has no label\n", __func__, > + ofnode_get_name(node)); > + return -EINVAL; > + } > + > + ret = device_bind_driver_to_node(parent, "ca-leds", > + ofnode_get_name(node), > + node, &dev); Does device_bind_ofnode() help here? > + if (ret) > + return ret; > + uc_plat = dev_get_uclass_platdata(dev); > + uc_plat->label = label; > + } > + > + return 0; > +} > + > +static const struct udevice_id ca_led_ids[] = { > + { .compatible = "cortina,ca-leds" }, > + { /* sentinel */ } > +}; > + > +U_BOOT_DRIVER(cortina_led) = { > + .name = "ca-leds", > + .id = UCLASS_LED, > + .of_match = ca_led_ids, > + .bind = cortina_led_bind, > + .probe = cortina_led_probe, > + .priv_auto_alloc_size = sizeof(struct cortina_led_cfg), > + .ops = &cortina_led_ops, > +}; > -- > 2.7.4 > Regards, Simon
diff --git a/MAINTAINERS b/MAINTAINERS index b147faa..24a2655 100644 --- a/MAINTAINERS +++ b/MAINTAINERS @@ -183,6 +183,7 @@ F: drivers/serial/serial_cortina.c F: drivers/mmc/ca_dw_mmc.c F: drivers/i2c/i2c-cortina.c F: drivers/i2c/i2c-cortina.h +F: drivers/led/led_cortina.c ARM/CZ.NIC TURRIS MOX SUPPORT M: Marek Behun <marek.behun at nic.cz> @@ -676,6 +677,7 @@ F: drivers/serial/serial_cortina.c F: drivers/mmc/ca_dw_mmc.c F: drivers/i2c/i2c-cortina.c F: drivers/i2c/i2c-cortina.h +F: drivers/led/led_cortina.c MIPS MSCC M: Gregory CLEMENT <gregory.clement at bootlin.com> diff --git a/drivers/led/Kconfig b/drivers/led/Kconfig index 6675934..cc87fbf 100644 --- a/drivers/led/Kconfig +++ b/drivers/led/Kconfig @@ -35,6 +35,14 @@ config LED_BCM6858 This option enables support for LEDs connected to the BCM6858 HW has blinking capabilities and up to 32 LEDs can be controlled. +config LED_CORTINA + bool "LED Support for Cortina Access CAxxxx SoCs" + depends on LED && (CORTINA_PLATFORM) + help + This option enables support for LEDs connected to the Cortina + Access CAxxxx SOCs. + + config LED_BLINK bool "Support LED blinking" depends on LED diff --git a/drivers/led/Makefile b/drivers/led/Makefile index 3654dd3..8e3ae7f 100644 --- a/drivers/led/Makefile +++ b/drivers/led/Makefile @@ -8,3 +8,4 @@ obj-$(CONFIG_LED_BCM6328) += led_bcm6328.o obj-$(CONFIG_LED_BCM6358) += led_bcm6358.o obj-$(CONFIG_LED_BCM6858) += led_bcm6858.o obj-$(CONFIG_$(SPL_)LED_GPIO) += led_gpio.o +obj-$(CONFIG_LED_CORTINA) += led_cortina.o diff --git a/drivers/led/led_cortina.c b/drivers/led/led_cortina.c new file mode 100644 index 0000000..53435e8 --- /dev/null +++ b/drivers/led/led_cortina.c @@ -0,0 +1,308 @@ +// SPDX-License-Identifier: GPL-2.0+ + +/* + * Copyright (C) 2020 Cortina-Access + * Author: Jway Lin <jway.lin at cortina-access.com> + * + */ + +#include <common.h> +#include <dm.h> +#include <errno.h> +#include <led.h> +#include <asm/io.h> +#include <dm/lists.h> +#include <linux/compat.h> + +#define CORTINA_LED_NUM 16 + +#define BIT(nr) (1UL << (nr)) + +#define cortina_LED_CONTROL 0x00 +#define cortina_LED_CONFIG_0 0x04 +#define cortina_LED_CONFIG_1 0x08 +#define cortina_LED_CONFIG_2 0x0c +#define cortina_LED_CONFIG_3 0x10 +#define cortina_LED_CONFIG_4 0x14 +#define cortina_LED_CONFIG_5 0x18 +#define cortina_LED_CONFIG_6 0x1c +#define cortina_LED_CONFIG_7 0x20 +#define cortina_LED_CONFIG_8 0x24 +#define cortina_LED_CONFIG_9 0x28 +#define cortina_LED_CONFIG_10 0x2c +#define cortina_LED_CONFIG_11 0x30 +#define cortina_LED_CONFIG_12 0x34 +#define cortina_LED_CONFIG_13 0x38 +#define cortina_LED_CONFIG_14 0x3c +#define cortina_LED_CONFIG_15 0x40 + +#define cortina_LED_MAX_HW_BLINK 127 +#define cortina_LED_MAX_COUNT CORTINA_LED_NUM +#define cortina_LED_MAX_PORT 8 + +/* LED_CONTROL fields */ +#define cortina_LED_BLINK_RATE1_OFFSET 0 +#define cortina_LED_BLINK_RATE1_MASK 0xFF +#define cortina_LED_BLINK_RATE2_OFFSET 8 +#define cortina_LED_BLINK_RATE2_MASK 0xFF +#define cortina_LED_CLK_TEST BIT(16) +#define cortina_LED_CLK_POLARITY BIT(17) +#define cortina_LED_CLK_TEST_MODE BIT(16) +#define cortina_LED_CLK_TEST_RX_TEST BIT(30) +#define cortina_LED_CLK_TEST_TX_TEST BIT(31) + +/* LED_CONFIG fields */ +#define cortina_LED_EVENT_ON_OFFSET 0 +#define cortina_LED_EVENT_ON_MASK 0x7 +#define cortina_LED_EVENT_BLINK_OFFSET 3 +#define cortina_LED_EVENT_BLINK_MASK 0x7 +#define cortina_LED_EVENT_OFF_OFFSET 6 +#define cortina_LED_EVENT_OFF_MASK 0x7 +#define cortina_LED_OFF_ON_OFFSET 9 +#define cortina_LED_OFF_ON_MASK 0x3 +#define cortina_LED_PORT_OFFSET 11 +#define cortina_LED_PORT_MASK 0x7 +#define cortina_LED_OFF_VAL BIT(14) +#define cortina_LED_SW_EVENT BIT(15) +#define cortina_LED_BLINK_SEL BIT(16) + +struct cortina_led_cfg { + void __iomem *regs; + spinlock_t *lock; /* protect LED resource access */ + int idx; + bool active_low; + + int off_event; + int blink_event; + int on_event; + int port; + int blink; + int enable; +}; + +struct cortina_led_top_cfg { + void __iomem *regs; + u16 blink_rate1; + u16 blink_rate2; +}; + +static struct cortina_led_top_cfg glb_led_ctrl; + +static void cortina_led_write(void __iomem *reg, unsigned long data) +{ + writel(data, reg); +} + +static unsigned long cortina_led_read(void __iomem *reg) +{ + return readl(reg); +} + +static enum led_state_t cortina_led_get_state(struct udevice *dev) +{ + struct cortina_led_cfg *priv = dev_get_priv(dev); + enum led_state_t state = LEDST_OFF; + u32 val; + + val = readl(priv->regs); + + if (val & cortina_LED_SW_EVENT) + state = LEDST_ON; + + return state; +} + +static int cortina_led_set_state(struct udevice *dev, enum led_state_t state) +{ + u32 val; + struct cortina_led_cfg *priv = dev_get_priv(dev); + + val = readl(priv->regs); + + switch (state) { + case LEDST_OFF: + val &= ~cortina_LED_SW_EVENT; + val &= ~(cortina_LED_OFF_ON_MASK << cortina_LED_OFF_ON_OFFSET); + val |= 0x3 << cortina_LED_OFF_ON_OFFSET; + cortina_led_write(priv->regs, val); + break; + case LEDST_ON: + val |= cortina_LED_SW_EVENT; + val &= ~(cortina_LED_OFF_ON_MASK << cortina_LED_OFF_ON_OFFSET); + val |= 0x1 << cortina_LED_OFF_ON_OFFSET; + cortina_led_write(priv->regs, val); + break; + case LEDST_TOGGLE: + if (cortina_led_get_state(dev) == LEDST_OFF) + return cortina_led_set_state(dev, LEDST_ON); + else + return cortina_led_set_state(dev, LEDST_OFF); + break; + + default: + return -EINVAL; + } + + return 0; +} + +static const struct led_ops cortina_led_ops = { + .get_state = cortina_led_get_state, + .set_state = cortina_led_set_state, +}; + +static int cortina_led_probe(struct udevice *dev) +{ + struct led_uc_plat *uc_plat = dev_get_uclass_platdata(dev); + void __iomem *regs; + u32 reg_value, val; + u16 rate1, rate2; + struct cortina_led_cfg *priv = dev_get_priv(dev); + unsigned int pin; + u32 blink, port, off_event, blink_event, on_event; + + /* Top-level LED node */ + if (!uc_plat->label) { + regs = dev_remap_addr(dev); + + if (!regs) + return -EINVAL; + + glb_led_ctrl.regs = regs; + + reg_value = 0; + reg_value |= cortina_LED_CLK_POLARITY; + + rate1 = dev_read_u32_default(dev, "cortina, blink_rate1", 256); + rate2 = dev_read_u32_default(dev, "cortina, blink_rate2", 512); + + val = rate1 / 16 - 1; + glb_led_ctrl.blink_rate1 = val > cortina_LED_MAX_HW_BLINK ? + cortina_LED_MAX_HW_BLINK : val; + reg_value |= (glb_led_ctrl.blink_rate1 & cortina_LED_BLINK_RATE1_MASK) + << cortina_LED_BLINK_RATE1_OFFSET; + + val = rate2 / 16 - 1; + glb_led_ctrl.blink_rate2 = val > cortina_LED_MAX_HW_BLINK ? + cortina_LED_MAX_HW_BLINK : val; + reg_value |= (glb_led_ctrl.blink_rate2 & cortina_LED_BLINK_RATE2_MASK) + << cortina_LED_BLINK_RATE2_OFFSET; + + cortina_led_write(glb_led_ctrl.regs, reg_value); + } else { + regs = dev_remap_addr(dev_get_parent(dev)); + + if (!regs) + return -EINVAL; + + pin = dev_read_u32_default(dev, "pin", cortina_LED_MAX_COUNT); + + if (pin >= cortina_LED_MAX_COUNT) + return -EINVAL; + + priv->regs = regs + 4 + (pin * 4); + + val = cortina_led_read(priv->regs); + + if (dev_read_bool(dev, "active-low")) { + priv->active_low = true; + val |= cortina_LED_OFF_VAL; + } else { + priv->active_low = false; + val &= ~cortina_LED_OFF_VAL; + } + + blink = dev_read_u32_default(dev, "blink-sel", 0); + + if (blink == 0) { + priv->blink = 0; + val &= ~cortina_LED_BLINK_SEL; + } else if (blink == 1) { + priv->blink = 1; + val |= cortina_LED_BLINK_SEL; + } + + //Todo : cortina_led_config(); + off_event = dev_read_u32_default(dev, "off-event", 3); + val &= ~(cortina_LED_EVENT_OFF_MASK << cortina_LED_EVENT_OFF_OFFSET); + if (off_event != 3) { + priv->off_event = off_event; + val |= BIT(off_event) << cortina_LED_EVENT_OFF_OFFSET; + } + + blink_event = dev_read_u32_default(dev, "blink-event", 3); + val &= ~(cortina_LED_EVENT_BLINK_MASK << + cortina_LED_EVENT_BLINK_OFFSET); + + if (blink_event != 3) { + priv->blink_event = blink_event; + val |= BIT(blink_event) << cortina_LED_EVENT_BLINK_OFFSET; + } + + on_event = dev_read_u32_default(dev, "on-event", 3); + val &= ~(cortina_LED_EVENT_ON_MASK << cortina_LED_EVENT_ON_OFFSET); + if (on_event != 3) { + priv->on_event = on_event; + val |= BIT(on_event) << cortina_LED_EVENT_ON_OFFSET; + } + + port = dev_read_u32_default(dev, "port", 0); + priv->port = port; + val &= ~(cortina_LED_PORT_MASK << cortina_LED_PORT_OFFSET); + val |= port << cortina_LED_PORT_OFFSET; + + priv->enable = 0; + + /* force off */ + val &= ~(cortina_LED_OFF_ON_MASK << cortina_LED_OFF_ON_OFFSET); + val |= 0x3 << cortina_LED_OFF_ON_OFFSET; + + cortina_led_write(priv->regs, val); + } + + return 0; +} + +static int cortina_led_bind(struct udevice *parent) +{ + ofnode node; + + dev_for_each_subnode(node, parent) { + struct led_uc_plat *uc_plat; + struct udevice *dev; + const char *label; + int ret; + + label = ofnode_read_string(node, "label"); + if (!label) { + debug("%s: node %s has no label\n", __func__, + ofnode_get_name(node)); + return -EINVAL; + } + + ret = device_bind_driver_to_node(parent, "ca-leds", + ofnode_get_name(node), + node, &dev); + if (ret) + return ret; + uc_plat = dev_get_uclass_platdata(dev); + uc_plat->label = label; + } + + return 0; +} + +static const struct udevice_id ca_led_ids[] = { + { .compatible = "cortina,ca-leds" }, + { /* sentinel */ } +}; + +U_BOOT_DRIVER(cortina_led) = { + .name = "ca-leds", + .id = UCLASS_LED, + .of_match = ca_led_ids, + .bind = cortina_led_bind, + .probe = cortina_led_probe, + .priv_auto_alloc_size = sizeof(struct cortina_led_cfg), + .ops = &cortina_led_ops, +};