led_gpio.c 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133
  1. /*
  2. * Copyright (c) 2015 Google, Inc
  3. * Written by Simon Glass <sjg@chromium.org>
  4. *
  5. * SPDX-License-Identifier: GPL-2.0+
  6. */
  7. #include <common.h>
  8. #include <dm.h>
  9. #include <errno.h>
  10. #include <led.h>
  11. #include <asm/gpio.h>
  12. #include <dm/lists.h>
  13. DECLARE_GLOBAL_DATA_PTR;
  14. struct led_gpio_priv {
  15. struct gpio_desc gpio;
  16. };
  17. static int gpio_led_set_state(struct udevice *dev, enum led_state_t state)
  18. {
  19. struct led_gpio_priv *priv = dev_get_priv(dev);
  20. int ret;
  21. if (!dm_gpio_is_valid(&priv->gpio))
  22. return -EREMOTEIO;
  23. switch (state) {
  24. case LEDST_OFF:
  25. case LEDST_ON:
  26. break;
  27. case LEDST_TOGGLE:
  28. ret = dm_gpio_get_value(&priv->gpio);
  29. if (ret < 0)
  30. return ret;
  31. state = !ret;
  32. break;
  33. default:
  34. return -ENOSYS;
  35. }
  36. return dm_gpio_set_value(&priv->gpio, state);
  37. }
  38. static enum led_state_t gpio_led_get_state(struct udevice *dev)
  39. {
  40. struct led_gpio_priv *priv = dev_get_priv(dev);
  41. int ret;
  42. if (!dm_gpio_is_valid(&priv->gpio))
  43. return -EREMOTEIO;
  44. ret = dm_gpio_get_value(&priv->gpio);
  45. if (ret < 0)
  46. return ret;
  47. return ret ? LEDST_ON : LEDST_OFF;
  48. }
  49. static int led_gpio_probe(struct udevice *dev)
  50. {
  51. struct led_uc_plat *uc_plat = dev_get_uclass_platdata(dev);
  52. struct led_gpio_priv *priv = dev_get_priv(dev);
  53. /* Ignore the top-level LED node */
  54. if (!uc_plat->label)
  55. return 0;
  56. return gpio_request_by_name(dev, "gpios", 0, &priv->gpio, GPIOD_IS_OUT);
  57. }
  58. static int led_gpio_remove(struct udevice *dev)
  59. {
  60. /*
  61. * The GPIO driver may have already been removed. We will need to
  62. * address this more generally.
  63. */
  64. #ifndef CONFIG_SANDBOX
  65. struct led_gpio_priv *priv = dev_get_priv(dev);
  66. if (dm_gpio_is_valid(&priv->gpio))
  67. dm_gpio_free(dev, &priv->gpio);
  68. #endif
  69. return 0;
  70. }
  71. static int led_gpio_bind(struct udevice *parent)
  72. {
  73. struct udevice *dev;
  74. ofnode node;
  75. int ret;
  76. dev_for_each_subnode(node, parent) {
  77. struct led_uc_plat *uc_plat;
  78. const char *label;
  79. label = ofnode_read_string(node, "label");
  80. if (!label) {
  81. debug("%s: node %s has no label\n", __func__,
  82. ofnode_get_name(node));
  83. return -EINVAL;
  84. }
  85. ret = device_bind_driver_to_node(parent, "gpio_led",
  86. ofnode_get_name(node),
  87. node, &dev);
  88. if (ret)
  89. return ret;
  90. uc_plat = dev_get_uclass_platdata(dev);
  91. uc_plat->label = label;
  92. }
  93. return 0;
  94. }
  95. static const struct led_ops gpio_led_ops = {
  96. .set_state = gpio_led_set_state,
  97. .get_state = gpio_led_get_state,
  98. };
  99. static const struct udevice_id led_gpio_ids[] = {
  100. { .compatible = "gpio-leds" },
  101. { }
  102. };
  103. U_BOOT_DRIVER(led_gpio) = {
  104. .name = "gpio_led",
  105. .id = UCLASS_LED,
  106. .of_match = led_gpio_ids,
  107. .ops = &gpio_led_ops,
  108. .priv_auto_alloc_size = sizeof(struct led_gpio_priv),
  109. .bind = led_gpio_bind,
  110. .probe = led_gpio_probe,
  111. .remove = led_gpio_remove,
  112. };