reset-meson.c 1.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Amlogic Meson Reset Controller driver
  4. *
  5. * Copyright (c) 2018 BayLibre, SAS.
  6. * Author: Neil Armstrong <narmstrong@baylibre.com>
  7. */
  8. #include <common.h>
  9. #include <dm.h>
  10. #include <log.h>
  11. #include <malloc.h>
  12. #include <reset-uclass.h>
  13. #include <regmap.h>
  14. #include <linux/bitops.h>
  15. #define REG_COUNT 8
  16. #define BITS_PER_REG 32
  17. #define LEVEL_OFFSET 0x7c
  18. struct meson_reset_priv {
  19. struct regmap *regmap;
  20. };
  21. static int meson_reset_request(struct reset_ctl *reset_ctl)
  22. {
  23. if (reset_ctl->id > (REG_COUNT * BITS_PER_REG))
  24. return -EINVAL;
  25. return 0;
  26. }
  27. static int meson_reset_level(struct reset_ctl *reset_ctl, bool assert)
  28. {
  29. struct meson_reset_priv *priv = dev_get_priv(reset_ctl->dev);
  30. uint bank = reset_ctl->id / BITS_PER_REG;
  31. uint offset = reset_ctl->id % BITS_PER_REG;
  32. uint reg_offset = LEVEL_OFFSET + (bank << 2);
  33. uint val;
  34. regmap_read(priv->regmap, reg_offset, &val);
  35. if (assert)
  36. val &= ~BIT(offset);
  37. else
  38. val |= BIT(offset);
  39. regmap_write(priv->regmap, reg_offset, val);
  40. return 0;
  41. }
  42. static int meson_reset_assert(struct reset_ctl *reset_ctl)
  43. {
  44. return meson_reset_level(reset_ctl, true);
  45. }
  46. static int meson_reset_deassert(struct reset_ctl *reset_ctl)
  47. {
  48. return meson_reset_level(reset_ctl, false);
  49. }
  50. struct reset_ops meson_reset_ops = {
  51. .request = meson_reset_request,
  52. .rst_assert = meson_reset_assert,
  53. .rst_deassert = meson_reset_deassert,
  54. };
  55. static const struct udevice_id meson_reset_ids[] = {
  56. { .compatible = "amlogic,meson-gxbb-reset" },
  57. { .compatible = "amlogic,meson-axg-reset" },
  58. { }
  59. };
  60. static int meson_reset_probe(struct udevice *dev)
  61. {
  62. struct meson_reset_priv *priv = dev_get_priv(dev);
  63. return regmap_init_mem(dev_ofnode(dev), &priv->regmap);
  64. }
  65. U_BOOT_DRIVER(meson_reset) = {
  66. .name = "meson_reset",
  67. .id = UCLASS_RESET,
  68. .of_match = meson_reset_ids,
  69. .probe = meson_reset_probe,
  70. .ops = &meson_reset_ops,
  71. .priv_auto = sizeof(struct meson_reset_priv),
  72. };