/* * Copyright (c) 2015 Google, Inc * Written by Simon Glass <sjg@chromium.org> * * SPDX-License-Identifier: GPL-2.0+ */ #include <common.h> #include <dm.h> #include <errno.h> #include <led.h> #include <asm/gpio.h> #include <dm/lists.h> DECLARE_GLOBAL_DATA_PTR; struct led_gpio_priv { struct gpio_desc gpio; }; static int gpio_led_set_on(struct udevice *dev, int on) { struct led_gpio_priv *priv = dev_get_priv(dev); if (!dm_gpio_is_valid(&priv->gpio)) return -EREMOTEIO; return dm_gpio_set_value(&priv->gpio, on); } static int led_gpio_probe(struct udevice *dev) { struct led_uclass_plat *uc_plat = dev_get_uclass_platdata(dev); struct led_gpio_priv *priv = dev_get_priv(dev); /* Ignore the top-level LED node */ if (!uc_plat->label) return 0; return gpio_request_by_name(dev, "gpios", 0, &priv->gpio, GPIOD_IS_OUT); } static int led_gpio_remove(struct udevice *dev) { /* * The GPIO driver may have already been removed. We will need to * address this more generally. */ #ifndef CONFIG_SANDBOX struct led_gpio_priv *priv = dev_get_priv(dev); if (dm_gpio_is_valid(&priv->gpio)) dm_gpio_free(dev, &priv->gpio); #endif return 0; } static int led_gpio_bind(struct udevice *parent) { const void *blob = gd->fdt_blob; struct udevice *dev; int node; int ret; for (node = fdt_first_subnode(blob, parent->of_offset); node > 0; node = fdt_next_subnode(blob, node)) { struct led_uclass_plat *uc_plat; const char *label; label = fdt_getprop(blob, node, "label", NULL); if (!label) { debug("%s: node %s has no label\n", __func__, fdt_get_name(blob, node, NULL)); return -EINVAL; } ret = device_bind_driver_to_node(parent, "gpio_led", fdt_get_name(blob, node, NULL), node, &dev); if (ret) return ret; uc_plat = dev_get_uclass_platdata(dev); uc_plat->label = label; } return 0; } static const struct led_ops gpio_led_ops = { .set_on = gpio_led_set_on, }; static const struct udevice_id led_gpio_ids[] = { { .compatible = "gpio-leds" }, { } }; U_BOOT_DRIVER(led_gpio) = { .name = "gpio_led", .id = UCLASS_LED, .of_match = led_gpio_ids, .ops = &gpio_led_ops, .priv_auto_alloc_size = sizeof(struct led_gpio_priv), .bind = led_gpio_bind, .probe = led_gpio_probe, .remove = led_gpio_remove, };