reset-meson-common.c 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. // SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause
  2. /*
  3. * Amlogic Meson Reset core functions
  4. *
  5. * Copyright (c) 2016-2024 BayLibre, SAS.
  6. * Authors: Neil Armstrong <narmstrong@baylibre.com>
  7. * Jerome Brunet <jbrunet@baylibre.com>
  8. */
  9. #include <linux/device.h>
  10. #include <linux/module.h>
  11. #include <linux/regmap.h>
  12. #include <linux/reset-controller.h>
  13. #include "reset-meson.h"
  14. struct meson_reset {
  15. const struct meson_reset_param *param;
  16. struct reset_controller_dev rcdev;
  17. struct regmap *map;
  18. };
  19. static void meson_reset_offset_and_bit(struct meson_reset *data,
  20. unsigned long id,
  21. unsigned int *offset,
  22. unsigned int *bit)
  23. {
  24. unsigned int stride = regmap_get_reg_stride(data->map);
  25. *offset = (id / (stride * BITS_PER_BYTE)) * stride;
  26. *bit = id % (stride * BITS_PER_BYTE);
  27. }
  28. static int meson_reset_reset(struct reset_controller_dev *rcdev,
  29. unsigned long id)
  30. {
  31. struct meson_reset *data =
  32. container_of(rcdev, struct meson_reset, rcdev);
  33. unsigned int offset, bit;
  34. meson_reset_offset_and_bit(data, id, &offset, &bit);
  35. offset += data->param->reset_offset;
  36. return regmap_write(data->map, offset, BIT(bit));
  37. }
  38. static int meson_reset_level(struct reset_controller_dev *rcdev,
  39. unsigned long id, bool assert)
  40. {
  41. struct meson_reset *data =
  42. container_of(rcdev, struct meson_reset, rcdev);
  43. unsigned int offset, bit;
  44. meson_reset_offset_and_bit(data, id, &offset, &bit);
  45. offset += data->param->level_offset;
  46. assert ^= data->param->level_low_reset;
  47. return regmap_update_bits(data->map, offset,
  48. BIT(bit), assert ? BIT(bit) : 0);
  49. }
  50. static int meson_reset_status(struct reset_controller_dev *rcdev,
  51. unsigned long id)
  52. {
  53. struct meson_reset *data =
  54. container_of(rcdev, struct meson_reset, rcdev);
  55. unsigned int val, offset, bit;
  56. meson_reset_offset_and_bit(data, id, &offset, &bit);
  57. offset += data->param->level_offset;
  58. regmap_read(data->map, offset, &val);
  59. val = !!(BIT(bit) & val);
  60. return val ^ data->param->level_low_reset;
  61. }
  62. static int meson_reset_assert(struct reset_controller_dev *rcdev,
  63. unsigned long id)
  64. {
  65. return meson_reset_level(rcdev, id, true);
  66. }
  67. static int meson_reset_deassert(struct reset_controller_dev *rcdev,
  68. unsigned long id)
  69. {
  70. return meson_reset_level(rcdev, id, false);
  71. }
  72. static int meson_reset_level_toggle(struct reset_controller_dev *rcdev,
  73. unsigned long id)
  74. {
  75. int ret;
  76. ret = meson_reset_assert(rcdev, id);
  77. if (ret)
  78. return ret;
  79. return meson_reset_deassert(rcdev, id);
  80. }
  81. const struct reset_control_ops meson_reset_ops = {
  82. .reset = meson_reset_reset,
  83. .assert = meson_reset_assert,
  84. .deassert = meson_reset_deassert,
  85. .status = meson_reset_status,
  86. };
  87. EXPORT_SYMBOL_NS_GPL(meson_reset_ops, "MESON_RESET");
  88. const struct reset_control_ops meson_reset_toggle_ops = {
  89. .reset = meson_reset_level_toggle,
  90. .assert = meson_reset_assert,
  91. .deassert = meson_reset_deassert,
  92. .status = meson_reset_status,
  93. };
  94. EXPORT_SYMBOL_NS_GPL(meson_reset_toggle_ops, "MESON_RESET");
  95. int meson_reset_controller_register(struct device *dev, struct regmap *map,
  96. const struct meson_reset_param *param)
  97. {
  98. struct meson_reset *data;
  99. data = devm_kzalloc(dev, sizeof(*data), GFP_KERNEL);
  100. if (!data)
  101. return -ENOMEM;
  102. data->param = param;
  103. data->map = map;
  104. data->rcdev.owner = dev->driver->owner;
  105. data->rcdev.nr_resets = param->reset_num;
  106. data->rcdev.ops = data->param->reset_ops;
  107. data->rcdev.of_node = dev->of_node;
  108. return devm_reset_controller_register(dev, &data->rcdev);
  109. }
  110. EXPORT_SYMBOL_NS_GPL(meson_reset_controller_register, "MESON_RESET");
  111. MODULE_DESCRIPTION("Amlogic Meson Reset Core function");
  112. MODULE_AUTHOR("Neil Armstrong <narmstrong@baylibre.com>");
  113. MODULE_AUTHOR("Jerome Brunet <jbrunet@baylibre.com>");
  114. MODULE_LICENSE("Dual BSD/GPL");
  115. MODULE_IMPORT_NS("MESON_RESET");