apr_tal_rpmsg.c 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2017-2020 The Linux Foundation. All rights reserved.
  4. */
  5. #include <linux/module.h>
  6. #include <linux/spinlock.h>
  7. #include <linux/mutex.h>
  8. #include <linux/wait.h>
  9. #include <linux/delay.h>
  10. #include <linux/rpmsg.h>
  11. #include <ipc/apr_tal.h>
  12. #include <linux/of_device.h>
  13. enum apr_channel_state {
  14. APR_CH_DISCONNECTED,
  15. APR_CH_CONNECTED,
  16. };
  17. #define APR_MAXIMUM_NUM_OF_RETRIES 2
  18. static struct apr_svc_ch_dev
  19. apr_svc_ch[APR_DL_MAX][APR_DEST_MAX][APR_CLIENT_MAX];
  20. /**
  21. * apr_tal_write() - Write a message across to the remote processor
  22. * @apr_ch: apr channel handle
  23. * @data: buffer that needs to be transferred over the channel
  24. * @pkt_priv: private data of the packet
  25. * @len: length of the buffer
  26. *
  27. * Returns len of buffer successfully transferred on success
  28. * and an appropriate error value on failure.
  29. */
  30. int apr_tal_write(struct apr_svc_ch_dev *apr_ch, void *data,
  31. struct apr_pkt_priv *pkt_priv, int len)
  32. {
  33. int rc = 0, retries = 0;
  34. unsigned long flags = 0;
  35. struct rpmsg_device *rpdev = NULL;
  36. if (!apr_ch || len > APR_MAX_BUF ||
  37. apr_ch->channel_state != APR_CH_CONNECTED)
  38. return -EINVAL;
  39. spin_lock_irqsave(&apr_ch->w_lock, flags);
  40. rpdev = apr_ch->handle;
  41. if (!rpdev) {
  42. spin_unlock_irqrestore(&apr_ch->w_lock, flags);
  43. return -EINVAL;
  44. }
  45. do {
  46. if (rc == -EAGAIN)
  47. udelay(50);
  48. rc = rpmsg_trysend(rpdev->ept, data, len);
  49. } while (rc == -EAGAIN && retries++ < APR_MAXIMUM_NUM_OF_RETRIES);
  50. spin_unlock_irqrestore(&apr_ch->w_lock, flags);
  51. if (rc)
  52. pr_err_ratelimited("%s: Unable to send the packet, rc:%d\n",
  53. __func__, rc);
  54. else
  55. rc = len;
  56. return rc;
  57. }
  58. EXPORT_SYMBOL(apr_tal_write);
  59. /**
  60. * apr_tal_rx_intents_config() - Configure glink intents for remote processor
  61. * @apr_ch: apr channel handle
  62. * @num_of_intents: number of intents
  63. * @size: size of the intents
  64. *
  65. * This api is not supported with RPMSG. Returns 0 to indicate success
  66. */
  67. int apr_tal_rx_intents_config(struct apr_svc_ch_dev *apr_ch,
  68. int num_of_intents, uint32_t size)
  69. {
  70. pr_debug("%s: NO-OP\n", __func__);
  71. return 0;
  72. }
  73. EXPORT_SYMBOL(apr_tal_rx_intents_config);
  74. /**
  75. * apr_tal_start_rx_rt() - Set RT thread priority for APR RX transfer
  76. * @apr_ch: apr channel handle
  77. *
  78. * This api is not supported with RPMSG as message transfer occurs
  79. * in client's context. Returns 0 to indicate success.
  80. */
  81. int apr_tal_start_rx_rt(struct apr_svc_ch_dev *apr_ch)
  82. {
  83. pr_debug("%s: NO-OP\n", __func__);
  84. return 0;
  85. }
  86. EXPORT_SYMBOL(apr_tal_start_rx_rt);
  87. /**
  88. * apr_tal_end_rx_rt() - Remove RT thread priority for APR RX transfer
  89. * @apr_ch: apr channel handle
  90. *
  91. * This api is not supported with RPMSG. Returns 0 to indicate success
  92. */
  93. int apr_tal_end_rx_rt(struct apr_svc_ch_dev *apr_ch)
  94. {
  95. pr_debug("%s: NO-OP\n", __func__);
  96. return 0;
  97. }
  98. EXPORT_SYMBOL(apr_tal_end_rx_rt);
  99. /**
  100. * apr_tal_open() - Open a transport channel for data transfer
  101. * on remote processor.
  102. * @clnt: apr client, audio or voice
  103. * @dest: destination remote processor for which apr channel is requested for.
  104. * @dl: type of data link
  105. * @func: callback function to handle data transfer from remote processor
  106. * @priv: private data of the client
  107. *
  108. * Returns apr_svc_ch_dev handle on success and NULL on failure.
  109. */
  110. struct apr_svc_ch_dev *apr_tal_open(uint32_t clnt, uint32_t dest, uint32_t dl,
  111. apr_svc_cb_fn func, void *priv)
  112. {
  113. int rc = 0;
  114. struct apr_svc_ch_dev *apr_ch = NULL;
  115. if ((clnt != APR_CLIENT_AUDIO) || (dest >= APR_DEST_MAX) ||
  116. (dl != APR_DL_SMD)) {
  117. pr_err("%s: Invalid params, clnt:%d, dest:%d, dl:%d\n",
  118. __func__, clnt, dest, dl);
  119. return NULL;
  120. }
  121. apr_ch = &apr_svc_ch[dl][dest][clnt];
  122. mutex_lock(&apr_ch->m_lock);
  123. if (!apr_ch->handle) {
  124. rc = wait_event_timeout(apr_ch->wait,
  125. (apr_ch->channel_state == APR_CH_CONNECTED), 5 * HZ);
  126. if (rc == 0) {
  127. pr_err("%s: TIMEOUT for APR_CH_CONNECTED event\n",
  128. __func__);
  129. rc = -ETIMEDOUT;
  130. goto unlock;
  131. }
  132. }
  133. pr_debug("%s: Channel connected, returning handle :%pK\n",
  134. __func__, apr_ch->handle);
  135. apr_ch->func = func;
  136. apr_ch->priv = priv;
  137. unlock:
  138. mutex_unlock(&apr_ch->m_lock);
  139. return rc ? NULL : apr_ch;
  140. }
  141. EXPORT_SYMBOL(apr_tal_open);
  142. /**
  143. * apr_tal_close() - Close transport channel on remote processor.
  144. * @apr_ch: apr channel handle
  145. *
  146. * Returns 0 on success and an appropriate error value on failure.
  147. */
  148. int apr_tal_close(struct apr_svc_ch_dev *apr_ch)
  149. {
  150. int rc = 0;
  151. if (!apr_ch || !apr_ch->handle) {
  152. rc = -EINVAL;
  153. goto exit;
  154. }
  155. mutex_lock(&apr_ch->m_lock);
  156. apr_ch->func = NULL;
  157. apr_ch->priv = NULL;
  158. mutex_unlock(&apr_ch->m_lock);
  159. exit:
  160. return rc;
  161. }
  162. EXPORT_SYMBOL(apr_tal_close);
  163. static int apr_tal_rpmsg_callback(struct rpmsg_device *rpdev,
  164. void *data, int len, void *priv, u32 addr)
  165. {
  166. struct apr_svc_ch_dev *apr_ch = dev_get_drvdata(&rpdev->dev);
  167. unsigned long flags = 0;
  168. if (!apr_ch || !data) {
  169. pr_err("%s: Invalid apr_ch or ptr\n", __func__);
  170. return -EINVAL;
  171. }
  172. dev_dbg(&rpdev->dev, "%s: Rx packet received, len:%d\n",
  173. __func__, len);
  174. spin_lock_irqsave(&apr_ch->r_lock, flags);
  175. if (apr_ch->func)
  176. apr_ch->func((void *)data, len, apr_ch->priv);
  177. spin_unlock_irqrestore(&apr_ch->r_lock, flags);
  178. return 0;
  179. }
  180. static int apr_tal_rpmsg_probe(struct rpmsg_device *rpdev)
  181. {
  182. struct apr_svc_ch_dev *apr_ch = NULL;
  183. int ret = 0;
  184. const char* dest;
  185. ret = of_property_read_string(rpdev->dev.parent->of_node,
  186. "mbox-names", &dest);
  187. if(ret < 0){
  188. pr_err("%s no parent source pid found\n", __func__);
  189. return -EINVAL;
  190. }
  191. if (!strcmp(rpdev->id.name, "apr_audio_svc")) {
  192. dev_info(&rpdev->dev, "%s: Channel[%s] state[Up]\n",
  193. __func__, rpdev->id.name);
  194. } else {
  195. dev_err(&rpdev->dev, "%s, Invalid Channel [%s]\n",
  196. __func__, rpdev->id.name);
  197. return -EINVAL;
  198. }
  199. if(strstr(dest, "adsp")) {
  200. apr_ch =
  201. &apr_svc_ch[APR_DL_SMD][APR_DEST_QDSP6][APR_CLIENT_AUDIO];
  202. apr_ch->handle = rpdev;
  203. apr_ch->channel_state = APR_CH_CONNECTED;
  204. dev_set_drvdata(&rpdev->dev, apr_ch);
  205. wake_up(&apr_ch->wait);
  206. } else if(strstr(dest, "mpss")) {
  207. apr_ch =
  208. &apr_svc_ch[APR_DL_SMD][APR_DEST_MODEM][APR_CLIENT_AUDIO];
  209. apr_ch->handle = rpdev;
  210. apr_ch->channel_state = APR_CH_CONNECTED;
  211. dev_set_drvdata(&rpdev->dev, apr_ch);
  212. wake_up(&apr_ch->wait);
  213. } else {
  214. dev_err(&rpdev->dev, "%s, unsupported dest %s\n",
  215. __func__, dest);
  216. return -EINVAL;
  217. }
  218. return ret;
  219. }
  220. static void apr_tal_rpmsg_remove(struct rpmsg_device *rpdev)
  221. {
  222. struct apr_svc_ch_dev *apr_ch = dev_get_drvdata(&rpdev->dev);
  223. if (!apr_ch) {
  224. dev_err(&rpdev->dev, "%s: Invalid apr_ch\n", __func__);
  225. return;
  226. }
  227. dev_info(&rpdev->dev, "%s: Channel[%s] state[Down]\n",
  228. __func__, rpdev->id.name);
  229. apr_ch->handle = NULL;
  230. apr_ch->channel_state = APR_CH_DISCONNECTED;
  231. dev_set_drvdata(&rpdev->dev, NULL);
  232. }
  233. static const struct rpmsg_device_id apr_tal_rpmsg_match[] = {
  234. { "apr_audio_svc" },
  235. {}
  236. };
  237. static struct rpmsg_driver apr_tal_rpmsg_driver = {
  238. .probe = apr_tal_rpmsg_probe,
  239. .remove = apr_tal_rpmsg_remove,
  240. .callback = apr_tal_rpmsg_callback,
  241. .id_table = apr_tal_rpmsg_match,
  242. .drv = {
  243. .name = "apr_tal_rpmsg",
  244. },
  245. };
  246. /**
  247. * apr_tal_int() - Registers rpmsg driver with rpmsg framework.
  248. *
  249. * Returns 0 on success and an appropriate error value on failure.
  250. */
  251. int apr_tal_init(void)
  252. {
  253. int i, j, k;
  254. int ret;
  255. memset(apr_svc_ch, 0, sizeof(struct apr_svc_ch_dev));
  256. for (i = 0; i < APR_DL_MAX; i++) {
  257. for (j = 0; j < APR_DEST_MAX; j++) {
  258. for (k = 0; k < APR_CLIENT_MAX; k++) {
  259. init_waitqueue_head(&apr_svc_ch[i][j][k].wait);
  260. spin_lock_init(&apr_svc_ch[i][j][k].w_lock);
  261. spin_lock_init(&apr_svc_ch[i][j][k].r_lock);
  262. mutex_init(&apr_svc_ch[i][j][k].m_lock);
  263. }
  264. }
  265. }
  266. ret = register_rpmsg_driver(&apr_tal_rpmsg_driver);
  267. return ret;
  268. }
  269. EXPORT_SYMBOL(apr_tal_init);
  270. /**
  271. * apr_tal_exit() - De-register rpmsg driver with rpmsg framework.
  272. */
  273. void apr_tal_exit(void)
  274. {
  275. unregister_rpmsg_driver(&apr_tal_rpmsg_driver);
  276. }
  277. EXPORT_SYMBOL(apr_tal_exit);