ti_k3_arm64_rproc.c 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Texas Instruments' K3 ARM64 Remoteproc driver
  4. *
  5. * Copyright (C) 2018 Texas Instruments Incorporated - http://www.ti.com/
  6. * Lokesh Vutla <lokeshvutla@ti.com>
  7. *
  8. */
  9. #include <common.h>
  10. #include <dm.h>
  11. #include <remoteproc.h>
  12. #include <errno.h>
  13. #include <clk.h>
  14. #include <reset.h>
  15. #include <asm/io.h>
  16. #include <power-domain.h>
  17. #include <dm/device_compat.h>
  18. #include <linux/err.h>
  19. #include <linux/soc/ti/ti_sci_protocol.h>
  20. #include "ti_sci_proc.h"
  21. #define INVALID_ID 0xffff
  22. #define GTC_CNTCR_REG 0x0
  23. #define GTC_CNTFID0_REG 0x20
  24. #define GTC_CNTR_EN 0x3
  25. /**
  26. * struct k3_arm64_privdata - Structure representing Remote processor data.
  27. * @rproc_pwrdmn: rproc power domain data
  28. * @rproc_rst: rproc reset control data
  29. * @sci: Pointer to TISCI handle
  30. * @tsp: TISCI processor control helper structure
  31. * @gtc_clk: GTC clock description
  32. * @gtc_base: Timer base address.
  33. */
  34. struct k3_arm64_privdata {
  35. bool has_cluster_node;
  36. struct power_domain cluster_pwrdmn;
  37. struct power_domain rproc_pwrdmn;
  38. struct power_domain gtc_pwrdmn;
  39. struct reset_ctl rproc_rst;
  40. struct ti_sci_proc tsp;
  41. struct clk gtc_clk;
  42. void *gtc_base;
  43. };
  44. /**
  45. * k3_arm64_load() - Load up the Remote processor image
  46. * @dev: rproc device pointer
  47. * @addr: Address at which image is available
  48. * @size: size of the image
  49. *
  50. * Return: 0 if all goes good, else appropriate error message.
  51. */
  52. static int k3_arm64_load(struct udevice *dev, ulong addr, ulong size)
  53. {
  54. struct k3_arm64_privdata *rproc = dev_get_priv(dev);
  55. ulong gtc_rate;
  56. int ret;
  57. dev_dbg(dev, "%s addr = 0x%lx, size = 0x%lx\n", __func__, addr, size);
  58. /* request for the processor */
  59. ret = ti_sci_proc_request(&rproc->tsp);
  60. if (ret)
  61. return ret;
  62. ret = power_domain_on(&rproc->gtc_pwrdmn);
  63. if (ret) {
  64. dev_err(dev, "power_domain_on(&rproc->gtc_pwrdmn) failed: %d\n",
  65. ret);
  66. return ret;
  67. }
  68. gtc_rate = clk_get_rate(&rproc->gtc_clk);
  69. dev_dbg(dev, "GTC RATE= %d\n", (u32) gtc_rate);
  70. /* Store the clock frequency down for GTC users to pick up */
  71. writel((u32)gtc_rate, rproc->gtc_base + GTC_CNTFID0_REG);
  72. /* Enable the timer before starting remote core */
  73. writel(GTC_CNTR_EN, rproc->gtc_base + GTC_CNTCR_REG);
  74. /*
  75. * Setting the right clock frequency would have taken care by
  76. * assigned-clock-rates during the device probe. So no need to
  77. * set the frequency again here.
  78. */
  79. if (rproc->has_cluster_node) {
  80. ret = power_domain_on(&rproc->cluster_pwrdmn);
  81. if (ret) {
  82. dev_err(dev,
  83. "power_domain_on(&rproc->cluster_pwrdmn) failed: %d\n",
  84. ret);
  85. return ret;
  86. }
  87. }
  88. return ti_sci_proc_set_config(&rproc->tsp, addr, 0, 0);
  89. }
  90. /**
  91. * k3_arm64_start() - Start the remote processor
  92. * @dev: rproc device pointer
  93. *
  94. * Return: 0 if all went ok, else return appropriate error
  95. */
  96. static int k3_arm64_start(struct udevice *dev)
  97. {
  98. struct k3_arm64_privdata *rproc = dev_get_priv(dev);
  99. int ret;
  100. dev_dbg(dev, "%s\n", __func__);
  101. ret = power_domain_on(&rproc->rproc_pwrdmn);
  102. if (ret) {
  103. dev_err(dev,
  104. "power_domain_on(&rproc->rproc_pwrdmn) failed: %d\n",
  105. ret);
  106. return ret;
  107. }
  108. return ti_sci_proc_release(&rproc->tsp);
  109. }
  110. /**
  111. * k3_arm64_init() - Initialize the remote processor
  112. * @dev: rproc device pointer
  113. *
  114. * Return: 0 if all went ok, else return appropriate error
  115. */
  116. static int k3_arm64_init(struct udevice *dev)
  117. {
  118. dev_dbg(dev, "%s\n", __func__);
  119. /* Enable the module */
  120. dev_dbg(dev, "%s: rproc successfully initialized\n", __func__);
  121. return 0;
  122. }
  123. static const struct dm_rproc_ops k3_arm64_ops = {
  124. .init = k3_arm64_init,
  125. .load = k3_arm64_load,
  126. .start = k3_arm64_start,
  127. };
  128. static int ti_sci_proc_of_to_priv(struct udevice *dev, struct ti_sci_proc *tsp)
  129. {
  130. dev_dbg(dev, "%s\n", __func__);
  131. tsp->sci = ti_sci_get_by_phandle(dev, "ti,sci");
  132. if (IS_ERR(tsp->sci)) {
  133. dev_err(dev, "ti_sci get failed: %ld\n", PTR_ERR(tsp->sci));
  134. return PTR_ERR(tsp->sci);
  135. }
  136. tsp->proc_id = dev_read_u32_default(dev, "ti,sci-proc-id", INVALID_ID);
  137. if (tsp->proc_id == INVALID_ID) {
  138. dev_err(dev, "proc id not populated\n");
  139. return -ENOENT;
  140. }
  141. tsp->host_id = dev_read_u32_default(dev, "ti,sci-host-id", INVALID_ID);
  142. tsp->ops = &tsp->sci->ops.proc_ops;
  143. return 0;
  144. }
  145. /**
  146. * k3_of_to_priv() - generate private data from device tree
  147. * @dev: corresponding k3 remote processor device
  148. * @priv: pointer to driver specific private data
  149. *
  150. * Return: 0 if all goes good, else appropriate error message.
  151. */
  152. static int k3_arm64_of_to_priv(struct udevice *dev,
  153. struct k3_arm64_privdata *rproc)
  154. {
  155. int ret;
  156. dev_dbg(dev, "%s\n", __func__);
  157. /* Cluster needs to be powered on if firewalls are being configured */
  158. rproc->has_cluster_node = true;
  159. ret = power_domain_get_by_index(dev, &rproc->cluster_pwrdmn, 2);
  160. if (ret) {
  161. dev_dbg(dev, "warning: power_domain_get_cluster() failed: %d\n", ret);
  162. rproc->has_cluster_node = false;
  163. }
  164. ret = power_domain_get_by_index(dev, &rproc->rproc_pwrdmn, 1);
  165. if (ret) {
  166. dev_err(dev, "power_domain_get_rproc() failed: %d\n", ret);
  167. return ret;
  168. }
  169. ret = power_domain_get_by_index(dev, &rproc->gtc_pwrdmn, 0);
  170. if (ret) {
  171. dev_err(dev, "power_domain_get() failed: %d\n", ret);
  172. return ret;
  173. }
  174. ret = clk_get_by_index(dev, 0, &rproc->gtc_clk);
  175. if (ret) {
  176. dev_err(dev, "clk_get failed: %d\n", ret);
  177. return ret;
  178. }
  179. ret = reset_get_by_index(dev, 0, &rproc->rproc_rst);
  180. if (ret) {
  181. dev_err(dev, "reset_get() failed: %d\n", ret);
  182. return ret;
  183. }
  184. ret = ti_sci_proc_of_to_priv(dev, &rproc->tsp);
  185. if (ret)
  186. return ret;
  187. rproc->gtc_base = dev_read_addr_ptr(dev);
  188. if (!rproc->gtc_base) {
  189. dev_err(dev, "Get address failed\n");
  190. return -ENODEV;
  191. }
  192. return 0;
  193. }
  194. /**
  195. * k3_arm64_probe() - Basic probe
  196. * @dev: corresponding k3 remote processor device
  197. *
  198. * Return: 0 if all goes good, else appropriate error message.
  199. */
  200. static int k3_arm64_probe(struct udevice *dev)
  201. {
  202. struct k3_arm64_privdata *priv;
  203. int ret;
  204. dev_dbg(dev, "%s\n", __func__);
  205. priv = dev_get_priv(dev);
  206. ret = k3_arm64_of_to_priv(dev, priv);
  207. if (ret) {
  208. dev_dbg(dev, "%s: Probe failed with error %d\n", __func__, ret);
  209. return ret;
  210. }
  211. dev_dbg(dev, "Remoteproc successfully probed\n");
  212. return 0;
  213. }
  214. static const struct udevice_id k3_arm64_ids[] = {
  215. { .compatible = "ti,am654-arm64"},
  216. { .compatible = "ti,am654-rproc"},
  217. {}
  218. };
  219. U_BOOT_DRIVER(k3_arm64) = {
  220. .name = "k3_arm64",
  221. .of_match = k3_arm64_ids,
  222. .id = UCLASS_REMOTEPROC,
  223. .ops = &k3_arm64_ops,
  224. .probe = k3_arm64_probe,
  225. .priv_auto = sizeof(struct k3_arm64_privdata),
  226. .flags = DM_FLAG_DEFAULT_PD_CTRL_OFF,
  227. };