apr_tal_rpmsg.c 7.8 KB

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