reset-gpio.c 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  1. // SPDX-License-Identifier: GPL-2.0
  2. #include <linux/gpio/consumer.h>
  3. #include <linux/mod_devicetable.h>
  4. #include <linux/module.h>
  5. #include <linux/of.h>
  6. #include <linux/platform_device.h>
  7. #include <linux/reset-controller.h>
  8. struct reset_gpio_priv {
  9. struct reset_controller_dev rc;
  10. struct gpio_desc *reset;
  11. };
  12. static inline struct reset_gpio_priv
  13. *rc_to_reset_gpio(struct reset_controller_dev *rc)
  14. {
  15. return container_of(rc, struct reset_gpio_priv, rc);
  16. }
  17. static int reset_gpio_assert(struct reset_controller_dev *rc, unsigned long id)
  18. {
  19. struct reset_gpio_priv *priv = rc_to_reset_gpio(rc);
  20. gpiod_set_value_cansleep(priv->reset, 1);
  21. return 0;
  22. }
  23. static int reset_gpio_deassert(struct reset_controller_dev *rc,
  24. unsigned long id)
  25. {
  26. struct reset_gpio_priv *priv = rc_to_reset_gpio(rc);
  27. gpiod_set_value_cansleep(priv->reset, 0);
  28. return 0;
  29. }
  30. static int reset_gpio_status(struct reset_controller_dev *rc, unsigned long id)
  31. {
  32. struct reset_gpio_priv *priv = rc_to_reset_gpio(rc);
  33. return gpiod_get_value_cansleep(priv->reset);
  34. }
  35. static const struct reset_control_ops reset_gpio_ops = {
  36. .assert = reset_gpio_assert,
  37. .deassert = reset_gpio_deassert,
  38. .status = reset_gpio_status,
  39. };
  40. static int reset_gpio_of_xlate(struct reset_controller_dev *rcdev,
  41. const struct of_phandle_args *reset_spec)
  42. {
  43. return reset_spec->args[0];
  44. }
  45. static void reset_gpio_of_node_put(void *data)
  46. {
  47. of_node_put(data);
  48. }
  49. static int reset_gpio_probe(struct platform_device *pdev)
  50. {
  51. struct device *dev = &pdev->dev;
  52. struct of_phandle_args *platdata = dev_get_platdata(dev);
  53. struct reset_gpio_priv *priv;
  54. int ret;
  55. if (!platdata)
  56. return -EINVAL;
  57. priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
  58. if (!priv)
  59. return -ENOMEM;
  60. platform_set_drvdata(pdev, &priv->rc);
  61. priv->reset = devm_gpiod_get(dev, "reset", GPIOD_OUT_HIGH);
  62. if (IS_ERR(priv->reset))
  63. return dev_err_probe(dev, PTR_ERR(priv->reset),
  64. "Could not get reset gpios\n");
  65. priv->rc.ops = &reset_gpio_ops;
  66. priv->rc.owner = THIS_MODULE;
  67. priv->rc.dev = dev;
  68. priv->rc.of_args = platdata;
  69. ret = devm_add_action_or_reset(dev, reset_gpio_of_node_put,
  70. priv->rc.of_node);
  71. if (ret)
  72. return ret;
  73. /* Cells to match GPIO specifier, but it's not really used */
  74. priv->rc.of_reset_n_cells = 2;
  75. priv->rc.of_xlate = reset_gpio_of_xlate;
  76. priv->rc.nr_resets = 1;
  77. return devm_reset_controller_register(dev, &priv->rc);
  78. }
  79. static const struct platform_device_id reset_gpio_ids[] = {
  80. { .name = "reset-gpio", },
  81. {}
  82. };
  83. MODULE_DEVICE_TABLE(platform, reset_gpio_ids);
  84. static struct platform_driver reset_gpio_driver = {
  85. .probe = reset_gpio_probe,
  86. .id_table = reset_gpio_ids,
  87. .driver = {
  88. .name = "reset-gpio",
  89. },
  90. };
  91. module_platform_driver(reset_gpio_driver);
  92. MODULE_AUTHOR("Krzysztof Kozlowski <krzysztof.kozlowski@linaro.org>");
  93. MODULE_DESCRIPTION("Generic GPIO reset driver");
  94. MODULE_LICENSE("GPL");