reset-uclass.c 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Copyright (c) 2016, NVIDIA CORPORATION.
  4. */
  5. #define LOG_CATEGORY UCLASS_RESET
  6. #include <common.h>
  7. #include <dm.h>
  8. #include <fdtdec.h>
  9. #include <log.h>
  10. #include <malloc.h>
  11. #include <reset.h>
  12. #include <reset-uclass.h>
  13. #include <dm/devres.h>
  14. #include <dm/lists.h>
  15. static inline struct reset_ops *reset_dev_ops(struct udevice *dev)
  16. {
  17. return (struct reset_ops *)dev->driver->ops;
  18. }
  19. static int reset_of_xlate_default(struct reset_ctl *reset_ctl,
  20. struct ofnode_phandle_args *args)
  21. {
  22. debug("%s(reset_ctl=%p)\n", __func__, reset_ctl);
  23. if (args->args_count != 1) {
  24. debug("Invalid args_count: %d\n", args->args_count);
  25. return -EINVAL;
  26. }
  27. reset_ctl->id = args->args[0];
  28. return 0;
  29. }
  30. static int reset_get_by_index_tail(int ret, ofnode node,
  31. struct ofnode_phandle_args *args,
  32. const char *list_name, int index,
  33. struct reset_ctl *reset_ctl)
  34. {
  35. struct udevice *dev_reset;
  36. struct reset_ops *ops;
  37. assert(reset_ctl);
  38. reset_ctl->dev = NULL;
  39. if (ret)
  40. return ret;
  41. ret = uclass_get_device_by_ofnode(UCLASS_RESET, args->node,
  42. &dev_reset);
  43. if (ret) {
  44. debug("%s: uclass_get_device_by_ofnode() failed: %d\n",
  45. __func__, ret);
  46. debug("%s %d\n", ofnode_get_name(args->node), args->args[0]);
  47. return ret;
  48. }
  49. ops = reset_dev_ops(dev_reset);
  50. reset_ctl->dev = dev_reset;
  51. if (ops->of_xlate)
  52. ret = ops->of_xlate(reset_ctl, args);
  53. else
  54. ret = reset_of_xlate_default(reset_ctl, args);
  55. if (ret) {
  56. debug("of_xlate() failed: %d\n", ret);
  57. return ret;
  58. }
  59. ret = ops->request ? ops->request(reset_ctl) : 0;
  60. if (ret) {
  61. debug("ops->request() failed: %d\n", ret);
  62. return ret;
  63. }
  64. return 0;
  65. }
  66. int reset_get_by_index(struct udevice *dev, int index,
  67. struct reset_ctl *reset_ctl)
  68. {
  69. struct ofnode_phandle_args args;
  70. int ret;
  71. ret = dev_read_phandle_with_args(dev, "resets", "#reset-cells", 0,
  72. index, &args);
  73. return reset_get_by_index_tail(ret, dev_ofnode(dev), &args, "resets",
  74. index > 0, reset_ctl);
  75. }
  76. int reset_get_by_index_nodev(ofnode node, int index,
  77. struct reset_ctl *reset_ctl)
  78. {
  79. struct ofnode_phandle_args args;
  80. int ret;
  81. ret = ofnode_parse_phandle_with_args(node, "resets", "#reset-cells", 0,
  82. index, &args);
  83. return reset_get_by_index_tail(ret, node, &args, "resets",
  84. index > 0, reset_ctl);
  85. }
  86. static int __reset_get_bulk(struct udevice *dev, ofnode node,
  87. struct reset_ctl_bulk *bulk)
  88. {
  89. int i, ret, err, count;
  90. bulk->count = 0;
  91. count = ofnode_count_phandle_with_args(node, "resets", "#reset-cells",
  92. 0);
  93. if (count < 1)
  94. return count;
  95. bulk->resets = devm_kcalloc(dev, count, sizeof(struct reset_ctl),
  96. GFP_KERNEL);
  97. if (!bulk->resets)
  98. return -ENOMEM;
  99. for (i = 0; i < count; i++) {
  100. ret = reset_get_by_index_nodev(node, i, &bulk->resets[i]);
  101. if (ret < 0)
  102. goto bulk_get_err;
  103. ++bulk->count;
  104. }
  105. return 0;
  106. bulk_get_err:
  107. err = reset_release_all(bulk->resets, bulk->count);
  108. if (err)
  109. debug("%s: could release all resets for %p\n",
  110. __func__, dev);
  111. return ret;
  112. }
  113. int reset_get_bulk(struct udevice *dev, struct reset_ctl_bulk *bulk)
  114. {
  115. return __reset_get_bulk(dev, dev_ofnode(dev), bulk);
  116. }
  117. int reset_get_by_name(struct udevice *dev, const char *name,
  118. struct reset_ctl *reset_ctl)
  119. {
  120. int index = 0;
  121. debug("%s(dev=%p, name=%s, reset_ctl=%p)\n", __func__, dev, name,
  122. reset_ctl);
  123. reset_ctl->dev = NULL;
  124. if (name) {
  125. index = dev_read_stringlist_search(dev, "reset-names", name);
  126. if (index < 0) {
  127. debug("fdt_stringlist_search() failed: %d\n", index);
  128. return index;
  129. }
  130. }
  131. return reset_get_by_index(dev, index, reset_ctl);
  132. }
  133. int reset_request(struct reset_ctl *reset_ctl)
  134. {
  135. struct reset_ops *ops = reset_dev_ops(reset_ctl->dev);
  136. debug("%s(reset_ctl=%p)\n", __func__, reset_ctl);
  137. return ops->request ? ops->request(reset_ctl) : 0;
  138. }
  139. int reset_free(struct reset_ctl *reset_ctl)
  140. {
  141. struct reset_ops *ops = reset_dev_ops(reset_ctl->dev);
  142. debug("%s(reset_ctl=%p)\n", __func__, reset_ctl);
  143. return ops->rfree ? ops->rfree(reset_ctl) : 0;
  144. }
  145. int reset_assert(struct reset_ctl *reset_ctl)
  146. {
  147. struct reset_ops *ops = reset_dev_ops(reset_ctl->dev);
  148. debug("%s(reset_ctl=%p)\n", __func__, reset_ctl);
  149. return ops->rst_assert ? ops->rst_assert(reset_ctl) : 0;
  150. }
  151. int reset_assert_bulk(struct reset_ctl_bulk *bulk)
  152. {
  153. int i, ret;
  154. for (i = 0; i < bulk->count; i++) {
  155. ret = reset_assert(&bulk->resets[i]);
  156. if (ret < 0)
  157. return ret;
  158. }
  159. return 0;
  160. }
  161. int reset_deassert(struct reset_ctl *reset_ctl)
  162. {
  163. struct reset_ops *ops = reset_dev_ops(reset_ctl->dev);
  164. debug("%s(reset_ctl=%p)\n", __func__, reset_ctl);
  165. return ops->rst_deassert ? ops->rst_deassert(reset_ctl) : 0;
  166. }
  167. int reset_deassert_bulk(struct reset_ctl_bulk *bulk)
  168. {
  169. int i, ret;
  170. for (i = 0; i < bulk->count; i++) {
  171. ret = reset_deassert(&bulk->resets[i]);
  172. if (ret < 0)
  173. return ret;
  174. }
  175. return 0;
  176. }
  177. int reset_status(struct reset_ctl *reset_ctl)
  178. {
  179. struct reset_ops *ops = reset_dev_ops(reset_ctl->dev);
  180. debug("%s(reset_ctl=%p)\n", __func__, reset_ctl);
  181. return ops->rst_status ? ops->rst_status(reset_ctl) : 0;
  182. }
  183. int reset_release_all(struct reset_ctl *reset_ctl, int count)
  184. {
  185. int i, ret;
  186. for (i = 0; i < count; i++) {
  187. debug("%s(reset_ctl[%d]=%p)\n", __func__, i, &reset_ctl[i]);
  188. /* check if reset has been previously requested */
  189. if (!reset_ctl[i].dev)
  190. continue;
  191. ret = reset_assert(&reset_ctl[i]);
  192. if (ret)
  193. return ret;
  194. ret = reset_free(&reset_ctl[i]);
  195. if (ret)
  196. return ret;
  197. }
  198. return 0;
  199. }
  200. static void devm_reset_release(struct udevice *dev, void *res)
  201. {
  202. reset_free(res);
  203. }
  204. struct reset_ctl *devm_reset_control_get_by_index(struct udevice *dev,
  205. int index)
  206. {
  207. int rc;
  208. struct reset_ctl *reset_ctl;
  209. reset_ctl = devres_alloc(devm_reset_release, sizeof(struct reset_ctl),
  210. __GFP_ZERO);
  211. if (unlikely(!reset_ctl))
  212. return ERR_PTR(-ENOMEM);
  213. rc = reset_get_by_index(dev, index, reset_ctl);
  214. if (rc)
  215. return ERR_PTR(rc);
  216. devres_add(dev, reset_ctl);
  217. return reset_ctl;
  218. }
  219. struct reset_ctl *devm_reset_control_get(struct udevice *dev, const char *id)
  220. {
  221. int rc;
  222. struct reset_ctl *reset_ctl;
  223. reset_ctl = devres_alloc(devm_reset_release, sizeof(struct reset_ctl),
  224. __GFP_ZERO);
  225. if (unlikely(!reset_ctl))
  226. return ERR_PTR(-ENOMEM);
  227. rc = reset_get_by_name(dev, id, reset_ctl);
  228. if (rc)
  229. return ERR_PTR(rc);
  230. devres_add(dev, reset_ctl);
  231. return reset_ctl;
  232. }
  233. struct reset_ctl *devm_reset_control_get_optional(struct udevice *dev,
  234. const char *id)
  235. {
  236. struct reset_ctl *r = devm_reset_control_get(dev, id);
  237. if (IS_ERR(r))
  238. return NULL;
  239. return r;
  240. }
  241. static void devm_reset_bulk_release(struct udevice *dev, void *res)
  242. {
  243. struct reset_ctl_bulk *bulk = res;
  244. reset_release_all(bulk->resets, bulk->count);
  245. }
  246. struct reset_ctl_bulk *devm_reset_bulk_get_by_node(struct udevice *dev,
  247. ofnode node)
  248. {
  249. int rc;
  250. struct reset_ctl_bulk *bulk;
  251. bulk = devres_alloc(devm_reset_bulk_release,
  252. sizeof(struct reset_ctl_bulk),
  253. __GFP_ZERO);
  254. /* this looks like a leak, but devres takes care of it */
  255. if (unlikely(!bulk))
  256. return ERR_PTR(-ENOMEM);
  257. rc = __reset_get_bulk(dev, node, bulk);
  258. if (rc)
  259. return ERR_PTR(rc);
  260. devres_add(dev, bulk);
  261. return bulk;
  262. }
  263. struct reset_ctl_bulk *devm_reset_bulk_get_optional_by_node(struct udevice *dev,
  264. ofnode node)
  265. {
  266. struct reset_ctl_bulk *bulk;
  267. bulk = devm_reset_bulk_get_by_node(dev, node);
  268. if (IS_ERR(bulk))
  269. return NULL;
  270. return bulk;
  271. }
  272. struct reset_ctl_bulk *devm_reset_bulk_get(struct udevice *dev)
  273. {
  274. return devm_reset_bulk_get_by_node(dev, dev_ofnode(dev));
  275. }
  276. struct reset_ctl_bulk *devm_reset_bulk_get_optional(struct udevice *dev)
  277. {
  278. return devm_reset_bulk_get_optional_by_node(dev, dev_ofnode(dev));
  279. }
  280. UCLASS_DRIVER(reset) = {
  281. .id = UCLASS_RESET,
  282. .name = "reset",
  283. };