led_gpio.c 3.1 KB

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