wcd-dsp-glink.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2016-2018, The Linux Foundation. All rights reserved.
  4. */
  5. #include <linux/init.h>
  6. #include <linux/kernel.h>
  7. #include <linux/module.h>
  8. #include <linux/spinlock.h>
  9. #include <linux/errno.h>
  10. #include <linux/fs.h>
  11. #include <linux/uaccess.h>
  12. #include <linux/slab.h>
  13. #include <linux/cdev.h>
  14. #include <linux/platform_device.h>
  15. #include <linux/of_device.h>
  16. #include <linux/vmalloc.h>
  17. #include <linux/rpmsg.h>
  18. #include "sound/wcd-dsp-glink.h"
  19. #define WDSP_GLINK_DRIVER_NAME "wcd-dsp-glink"
  20. #define WDSP_MAX_WRITE_SIZE (256 * 1024)
  21. #define WDSP_MAX_READ_SIZE (4 * 1024)
  22. #define WDSP_WRITE_PKT_SIZE (sizeof(struct wdsp_write_pkt))
  23. #define WDSP_CMD_PKT_SIZE (sizeof(struct wdsp_cmd_pkt))
  24. #define MINOR_NUMBER_COUNT 1
  25. #define RESP_QUEUE_SIZE 3
  26. #define TIMEOUT_MS 2000
  27. enum wdsp_ch_state {
  28. WDSP_CH_DISCONNECTED,
  29. WDSP_CH_CONNECTED,
  30. };
  31. struct wdsp_glink_dev {
  32. struct class *cls;
  33. struct device *dev;
  34. struct cdev cdev;
  35. dev_t dev_num;
  36. };
  37. struct wdsp_rsp_que {
  38. /* Size of valid data in buffer */
  39. u32 buf_size;
  40. /* Response buffer */
  41. u8 buf[WDSP_MAX_READ_SIZE];
  42. };
  43. struct wdsp_ch {
  44. struct wdsp_glink_priv *wpriv;
  45. /* rpmsg handle */
  46. void *handle;
  47. /* Channel states like connect, disconnect */
  48. int ch_state;
  49. char ch_name[RPMSG_NAME_SIZE];
  50. spinlock_t ch_lock;
  51. };
  52. struct wdsp_tx_buf {
  53. struct work_struct tx_work;
  54. /* Glink channel information */
  55. struct wdsp_ch *ch;
  56. /* Tx buffer to send to glink */
  57. u8 buf[0];
  58. };
  59. struct wdsp_glink_priv {
  60. /* Respone buffer related */
  61. u8 rsp_cnt;
  62. struct wdsp_rsp_que rsp[RESP_QUEUE_SIZE];
  63. u8 write_idx;
  64. u8 read_idx;
  65. struct completion rsp_complete;
  66. spinlock_t rsp_lock;
  67. /* Glink channel related */
  68. int no_of_channels;
  69. struct wdsp_ch **ch;
  70. struct workqueue_struct *work_queue;
  71. /* Wait for all channels state before sending any command */
  72. wait_queue_head_t ch_state_wait;
  73. struct wdsp_glink_dev *wdev;
  74. struct device *dev;
  75. };
  76. static struct wdsp_glink_priv *wpriv;
  77. static struct wdsp_ch *wdsp_get_ch(char *ch_name)
  78. {
  79. int i;
  80. for (i = 0; i < wpriv->no_of_channels; i++) {
  81. if (!strcmp(ch_name, wpriv->ch[i]->ch_name))
  82. return wpriv->ch[i];
  83. }
  84. return NULL;
  85. }
  86. /*
  87. * wdsp_rpmsg_callback - Rpmsg callback for responses
  88. * rpdev: Rpmsg device structure
  89. * data: Pointer to the Rx data
  90. * len: Size of the Rx data
  91. * priv: Private pointer to the channel
  92. * addr: Address variable
  93. * Returns 0 on success and an appropriate error value on failure
  94. */
  95. static int wdsp_rpmsg_callback(struct rpmsg_device *rpdev, void *data,
  96. int len, void *priv, u32 addr__unused)
  97. {
  98. struct wdsp_ch *ch = dev_get_drvdata(&rpdev->dev);
  99. struct wdsp_glink_priv *wpriv;
  100. unsigned long flags;
  101. u8 *rx_buf;
  102. u8 rsp_cnt = 0;
  103. if (!ch || !data) {
  104. pr_err("%s: Invalid ch or data\n", __func__);
  105. return -EINVAL;
  106. }
  107. wpriv = ch->wpriv;
  108. rx_buf = (u8 *)data;
  109. if (len > WDSP_MAX_READ_SIZE) {
  110. dev_info_ratelimited(wpriv->dev, "%s: Size %d is greater than allowed %d\n",
  111. __func__, len, WDSP_MAX_READ_SIZE);
  112. len = WDSP_MAX_READ_SIZE;
  113. }
  114. dev_dbg_ratelimited(wpriv->dev, "%s: copy into buffer %d\n", __func__,
  115. wpriv->rsp_cnt);
  116. if (wpriv->rsp_cnt >= RESP_QUEUE_SIZE) {
  117. dev_info_ratelimited(wpriv->dev, "%s: Resp Queue is Full. Ignore new one.\n",
  118. __func__);
  119. return -EINVAL;
  120. }
  121. spin_lock_irqsave(&wpriv->rsp_lock, flags);
  122. rsp_cnt = wpriv->rsp_cnt;
  123. memcpy(wpriv->rsp[wpriv->write_idx].buf, rx_buf, len);
  124. wpriv->rsp[wpriv->write_idx].buf_size = len;
  125. wpriv->write_idx = (wpriv->write_idx + 1) % RESP_QUEUE_SIZE;
  126. wpriv->rsp_cnt = ++rsp_cnt;
  127. spin_unlock_irqrestore(&wpriv->rsp_lock, flags);
  128. complete(&wpriv->rsp_complete);
  129. return 0;
  130. }
  131. /*
  132. * wdsp_rpmsg_probe - Rpmsg channel probe function
  133. * rpdev: Rpmsg device structure
  134. * Returns 0 on success and an appropriate error value on failure
  135. */
  136. static int wdsp_rpmsg_probe(struct rpmsg_device *rpdev)
  137. {
  138. struct wdsp_ch *ch;
  139. ch = wdsp_get_ch(rpdev->id.name);
  140. if (!ch) {
  141. dev_err(&rpdev->dev, "%s, Invalid Channel [%s]\n",
  142. __func__, rpdev->id.name);
  143. return -EINVAL;
  144. }
  145. dev_dbg(&rpdev->dev, "%s: Channel[%s] state[Up]\n",
  146. __func__, rpdev->id.name);
  147. spin_lock(&ch->ch_lock);
  148. ch->handle = rpdev;
  149. ch->ch_state = WDSP_CH_CONNECTED;
  150. spin_unlock(&ch->ch_lock);
  151. dev_set_drvdata(&rpdev->dev, ch);
  152. wake_up(&wpriv->ch_state_wait);
  153. return 0;
  154. }
  155. /*
  156. * wdsp_rpmsg_remove - Rpmsg channel remove function
  157. * rpdev: Rpmsg device structure
  158. */
  159. static void wdsp_rpmsg_remove(struct rpmsg_device *rpdev)
  160. {
  161. struct wdsp_ch *ch = dev_get_drvdata(&rpdev->dev);
  162. if (!ch) {
  163. dev_err(&rpdev->dev, "%s: Invalid ch\n", __func__);
  164. return;
  165. }
  166. dev_dbg(&rpdev->dev, "%s: Channel[%s] state[Down]\n",
  167. __func__, rpdev->id.name);
  168. spin_lock(&ch->ch_lock);
  169. ch->handle = NULL;
  170. ch->ch_state = WDSP_CH_DISCONNECTED;
  171. spin_unlock(&ch->ch_lock);
  172. dev_set_drvdata(&rpdev->dev, NULL);
  173. }
  174. static bool wdsp_is_ch_connected(struct wdsp_glink_priv *wpriv)
  175. {
  176. int i;
  177. for (i = 0; i < wpriv->no_of_channels; i++) {
  178. spin_lock(&wpriv->ch[i]->ch_lock);
  179. if (wpriv->ch[i]->ch_state != WDSP_CH_CONNECTED) {
  180. spin_unlock(&wpriv->ch[i]->ch_lock);
  181. return false;
  182. }
  183. spin_unlock(&wpriv->ch[i]->ch_lock);
  184. }
  185. return true;
  186. }
  187. static int wdsp_wait_for_all_ch_connect(struct wdsp_glink_priv *wpriv)
  188. {
  189. int ret;
  190. ret = wait_event_timeout(wpriv->ch_state_wait,
  191. wdsp_is_ch_connected(wpriv),
  192. msecs_to_jiffies(TIMEOUT_MS));
  193. if (!ret) {
  194. dev_err_ratelimited(wpriv->dev, "%s: All channels are not connected\n",
  195. __func__);
  196. ret = -ETIMEDOUT;
  197. goto err;
  198. }
  199. ret = 0;
  200. err:
  201. return ret;
  202. }
  203. /*
  204. * wdsp_tx_buf_work - Work queue function to send tx buffer to glink
  205. * work: Work structure
  206. */
  207. static void wdsp_tx_buf_work(struct work_struct *work)
  208. {
  209. struct wdsp_glink_priv *wpriv;
  210. struct wdsp_ch *ch;
  211. struct wdsp_tx_buf *tx_buf;
  212. struct wdsp_write_pkt *wpkt;
  213. struct wdsp_cmd_pkt *cpkt;
  214. int ret = 0;
  215. struct rpmsg_device *rpdev = NULL;
  216. tx_buf = container_of(work, struct wdsp_tx_buf,
  217. tx_work);
  218. ch = tx_buf->ch;
  219. wpriv = ch->wpriv;
  220. wpkt = (struct wdsp_write_pkt *)tx_buf->buf;
  221. cpkt = (struct wdsp_cmd_pkt *)wpkt->payload;
  222. dev_dbg(wpriv->dev, "%s: ch name = %s, payload size = %d\n",
  223. __func__, cpkt->ch_name, cpkt->payload_size);
  224. spin_lock(&ch->ch_lock);
  225. rpdev = ch->handle;
  226. if (rpdev && ch->ch_state == WDSP_CH_CONNECTED) {
  227. spin_unlock(&ch->ch_lock);
  228. ret = rpmsg_send(rpdev->ept, cpkt->payload,
  229. cpkt->payload_size);
  230. if (ret < 0)
  231. dev_err(wpriv->dev, "%s: rpmsg send failed, ret = %d\n",
  232. __func__, ret);
  233. } else {
  234. spin_unlock(&ch->ch_lock);
  235. if (rpdev)
  236. dev_err(wpriv->dev, "%s: channel %s is not in connected state\n",
  237. __func__, ch->ch_name);
  238. else
  239. dev_err(wpriv->dev, "%s: rpdev is NULL\n", __func__);
  240. }
  241. vfree(tx_buf);
  242. }
  243. /*
  244. * wdsp_glink_read - Read API to send the data to userspace
  245. * file: Pointer to the file structure
  246. * buf: Pointer to the userspace buffer
  247. * count: Number bytes to read from the file
  248. * ppos: Pointer to the position into the file
  249. * Returns 0 on success and an appropriate error value on failure
  250. */
  251. static ssize_t wdsp_glink_read(struct file *file, char __user *buf,
  252. size_t count, loff_t *ppos)
  253. {
  254. int ret = 0, ret1 = 0;
  255. struct wdsp_rsp_que *read_rsp = NULL;
  256. struct wdsp_glink_priv *wpriv;
  257. unsigned long flags;
  258. wpriv = (struct wdsp_glink_priv *)file->private_data;
  259. if (!wpriv) {
  260. pr_err("%s: Invalid private data\n", __func__);
  261. return -EINVAL;
  262. }
  263. if (count > WDSP_MAX_READ_SIZE) {
  264. dev_info_ratelimited(wpriv->dev, "%s: count = %zd is more than WDSP_MAX_READ_SIZE\n",
  265. __func__, count);
  266. count = WDSP_MAX_READ_SIZE;
  267. }
  268. /*
  269. * Complete signal has given from gwdsp_rpmsg_callback()
  270. * or from flush API. Also use interruptible wait_for_completion API
  271. * to allow the system to go in suspend.
  272. */
  273. ret = wait_for_completion_interruptible(&wpriv->rsp_complete);
  274. if (ret < 0)
  275. return ret;
  276. read_rsp = kzalloc(sizeof(struct wdsp_rsp_que), GFP_KERNEL);
  277. if (!read_rsp)
  278. return -ENOMEM;
  279. spin_lock_irqsave(&wpriv->rsp_lock, flags);
  280. if (wpriv->rsp_cnt) {
  281. wpriv->rsp_cnt--;
  282. dev_dbg(wpriv->dev, "%s: rsp_cnt=%d read from buffer %d\n",
  283. __func__, wpriv->rsp_cnt, wpriv->read_idx);
  284. memcpy(read_rsp, &wpriv->rsp[wpriv->read_idx],
  285. sizeof(struct wdsp_rsp_que));
  286. wpriv->read_idx = (wpriv->read_idx + 1) % RESP_QUEUE_SIZE;
  287. spin_unlock_irqrestore(&wpriv->rsp_lock, flags);
  288. if (count < read_rsp->buf_size) {
  289. ret1 = copy_to_user(buf, read_rsp->buf, count);
  290. /* Return the number of bytes copied */
  291. ret = count;
  292. } else {
  293. ret1 = copy_to_user(buf, read_rsp->buf,
  294. read_rsp->buf_size);
  295. /* Return the number of bytes copied */
  296. ret = read_rsp->buf_size;
  297. }
  298. if (ret1) {
  299. dev_err_ratelimited(wpriv->dev, "%s: copy_to_user failed %d\n",
  300. __func__, ret);
  301. ret = -EFAULT;
  302. goto done;
  303. }
  304. } else {
  305. /*
  306. * This will execute only if flush API is called or
  307. * something wrong with ref_cnt
  308. */
  309. dev_dbg(wpriv->dev, "%s: resp count = %d\n", __func__,
  310. wpriv->rsp_cnt);
  311. spin_unlock_irqrestore(&wpriv->rsp_lock, flags);
  312. ret = -EINVAL;
  313. }
  314. done:
  315. kfree(read_rsp);
  316. return ret;
  317. }
  318. /*
  319. * wdsp_glink_write - Write API to receive the data from userspace
  320. * file: Pointer to the file structure
  321. * buf: Pointer to the userspace buffer
  322. * count: Number bytes to read from the file
  323. * ppos: Pointer to the position into the file
  324. * Returns 0 on success and an appropriate error value on failure
  325. */
  326. static ssize_t wdsp_glink_write(struct file *file, const char __user *buf,
  327. size_t count, loff_t *ppos)
  328. {
  329. int ret = 0, i, tx_buf_size;
  330. struct wdsp_write_pkt *wpkt;
  331. struct wdsp_cmd_pkt *cpkt;
  332. struct wdsp_tx_buf *tx_buf;
  333. struct wdsp_glink_priv *wpriv;
  334. size_t pkt_max_size;
  335. wpriv = (struct wdsp_glink_priv *)file->private_data;
  336. if (!wpriv) {
  337. pr_err("%s: Invalid private data\n", __func__);
  338. ret = -EINVAL;
  339. goto done;
  340. }
  341. if ((count < WDSP_WRITE_PKT_SIZE) ||
  342. (count > WDSP_MAX_WRITE_SIZE)) {
  343. dev_err_ratelimited(wpriv->dev, "%s: Invalid count = %zd\n",
  344. __func__, count);
  345. ret = -EINVAL;
  346. goto done;
  347. }
  348. dev_dbg(wpriv->dev, "%s: count = %zd\n", __func__, count);
  349. tx_buf_size = count + sizeof(struct wdsp_tx_buf);
  350. tx_buf = vzalloc(tx_buf_size);
  351. if (!tx_buf) {
  352. ret = -ENOMEM;
  353. goto done;
  354. }
  355. ret = copy_from_user(tx_buf->buf, buf, count);
  356. if (ret) {
  357. dev_err_ratelimited(wpriv->dev, "%s: copy_from_user failed %d\n",
  358. __func__, ret);
  359. ret = -EFAULT;
  360. goto free_buf;
  361. }
  362. wpkt = (struct wdsp_write_pkt *)tx_buf->buf;
  363. switch (wpkt->pkt_type) {
  364. case WDSP_REG_PKT:
  365. /* Keep this case to support backward compatibility */
  366. vfree(tx_buf);
  367. break;
  368. case WDSP_READY_PKT:
  369. ret = wdsp_wait_for_all_ch_connect(wpriv);
  370. if (ret < 0)
  371. dev_err_ratelimited(wpriv->dev, "%s: Channels not in connected state\n",
  372. __func__);
  373. vfree(tx_buf);
  374. break;
  375. case WDSP_CMD_PKT:
  376. if (count <= (WDSP_WRITE_PKT_SIZE + WDSP_CMD_PKT_SIZE)) {
  377. dev_err_ratelimited(wpriv->dev, "%s: Invalid cmd pkt size = %zd\n",
  378. __func__, count);
  379. ret = -EINVAL;
  380. goto free_buf;
  381. }
  382. cpkt = (struct wdsp_cmd_pkt *)wpkt->payload;
  383. pkt_max_size = sizeof(struct wdsp_write_pkt) +
  384. sizeof(struct wdsp_cmd_pkt) +
  385. cpkt->payload_size;
  386. if (count < pkt_max_size) {
  387. dev_err_ratelimited(wpriv->dev, "%s: Invalid cmd pkt count = %zd, pkt_size = %zd\n",
  388. __func__, count, pkt_max_size);
  389. ret = -EINVAL;
  390. goto free_buf;
  391. }
  392. for (i = 0; i < wpriv->no_of_channels; i++) {
  393. if (!strcmp(cpkt->ch_name, wpriv->ch[i]->ch_name)) {
  394. tx_buf->ch = wpriv->ch[i];
  395. break;
  396. }
  397. }
  398. if (!tx_buf->ch) {
  399. dev_err_ratelimited(wpriv->dev, "%s: Failed to get channel\n",
  400. __func__);
  401. ret = -EINVAL;
  402. goto free_buf;
  403. }
  404. dev_dbg(wpriv->dev, "%s: requested ch_name: %s, pkt_size: %zd\n",
  405. __func__, cpkt->ch_name, pkt_max_size);
  406. spin_lock(&tx_buf->ch->ch_lock);
  407. if (tx_buf->ch->ch_state != WDSP_CH_CONNECTED) {
  408. spin_unlock(&tx_buf->ch->ch_lock);
  409. ret = -ENETRESET;
  410. dev_err_ratelimited(wpriv->dev, "%s: Channels are not in connected state\n",
  411. __func__);
  412. goto free_buf;
  413. }
  414. spin_unlock(&tx_buf->ch->ch_lock);
  415. INIT_WORK(&tx_buf->tx_work, wdsp_tx_buf_work);
  416. queue_work(wpriv->work_queue, &tx_buf->tx_work);
  417. break;
  418. default:
  419. dev_err_ratelimited(wpriv->dev, "%s: Invalid packet type\n",
  420. __func__);
  421. ret = -EINVAL;
  422. vfree(tx_buf);
  423. break;
  424. }
  425. goto done;
  426. free_buf:
  427. vfree(tx_buf);
  428. done:
  429. return ret;
  430. }
  431. /*
  432. * wdsp_glink_open - Open API to initialize private data
  433. * inode: Pointer to the inode structure
  434. * file: Pointer to the file structure
  435. * Returns 0 on success and an appropriate error value on failure
  436. */
  437. static int wdsp_glink_open(struct inode *inode, struct file *file)
  438. {
  439. pr_debug("%s: wpriv = %pK\n", __func__, wpriv);
  440. file->private_data = wpriv;
  441. return 0;
  442. }
  443. /*
  444. * wdsp_glink_flush - Flush API to unblock read.
  445. * file: Pointer to the file structure
  446. * id: Lock owner ID
  447. * Returns 0 on success and an appropriate error value on failure
  448. */
  449. static int wdsp_glink_flush(struct file *file, fl_owner_t id)
  450. {
  451. struct wdsp_glink_priv *wpriv;
  452. wpriv = (struct wdsp_glink_priv *)file->private_data;
  453. if (!wpriv) {
  454. pr_err("%s: Invalid private data\n", __func__);
  455. return -EINVAL;
  456. }
  457. complete(&wpriv->rsp_complete);
  458. return 0;
  459. }
  460. /*
  461. * wdsp_glink_release - Release API to clean up resources.
  462. * Whenever a file structure is shared across multiple threads,
  463. * release won't be invoked until all copies are closed
  464. * (file->f_count.counter should be 0). If we need to flush pending
  465. * data when any copy is closed, you should implement the flush method.
  466. *
  467. * inode: Pointer to the inode structure
  468. * file: Pointer to the file structure
  469. * Returns 0 on success and an appropriate error value on failure
  470. */
  471. static int wdsp_glink_release(struct inode *inode, struct file *file)
  472. {
  473. pr_debug("%s: file->private_data = %pK\n", __func__,
  474. file->private_data);
  475. file->private_data = NULL;
  476. return 0;
  477. }
  478. static struct rpmsg_driver wdsp_rpmsg_driver = {
  479. .probe = wdsp_rpmsg_probe,
  480. .remove = wdsp_rpmsg_remove,
  481. .callback = wdsp_rpmsg_callback,
  482. /* Update this dynamically before register_rpmsg() */
  483. .id_table = NULL,
  484. .drv = {
  485. .name = "wdsp_rpmsg",
  486. },
  487. };
  488. static int wdsp_register_rpmsg(struct platform_device *pdev,
  489. struct wdsp_glink_dev *wdev)
  490. {
  491. int ret = 0;
  492. int i, no_of_channels;
  493. struct rpmsg_device_id *wdsp_rpmsg_id_table, *id_table;
  494. const char *ch_name = NULL;
  495. wpriv = devm_kzalloc(&pdev->dev,
  496. sizeof(struct wdsp_glink_priv), GFP_KERNEL);
  497. if (!wpriv)
  498. return -ENOMEM;
  499. no_of_channels = of_property_count_strings(pdev->dev.of_node,
  500. "qcom,wdsp-channels");
  501. if (no_of_channels < 0) {
  502. dev_err(&pdev->dev, "%s: channel name parse error %d\n",
  503. __func__, no_of_channels);
  504. return -EINVAL;
  505. }
  506. wpriv->ch = devm_kzalloc(&pdev->dev,
  507. (sizeof(struct wdsp_glink_priv *) * no_of_channels),
  508. GFP_KERNEL);
  509. if (!wpriv->ch)
  510. return -ENOMEM;
  511. for (i = 0; i < no_of_channels; i++) {
  512. ret = of_property_read_string_index(pdev->dev.of_node,
  513. "qcom,wdsp-channels", i,
  514. &ch_name);
  515. if (ret) {
  516. dev_err(&pdev->dev, "%s: channel name parse error %d\n",
  517. __func__, ret);
  518. return -EINVAL;
  519. }
  520. wpriv->ch[i] = devm_kzalloc(&pdev->dev,
  521. sizeof(struct wdsp_glink_priv),
  522. GFP_KERNEL);
  523. if (!wpriv->ch[i])
  524. return -ENOMEM;
  525. strlcpy(wpriv->ch[i]->ch_name, ch_name, RPMSG_NAME_SIZE);
  526. wpriv->ch[i]->wpriv = wpriv;
  527. spin_lock_init(&wpriv->ch[i]->ch_lock);
  528. }
  529. init_waitqueue_head(&wpriv->ch_state_wait);
  530. init_completion(&wpriv->rsp_complete);
  531. spin_lock_init(&wpriv->rsp_lock);
  532. wpriv->wdev = wdev;
  533. wpriv->dev = wdev->dev;
  534. wpriv->work_queue = create_singlethread_workqueue("wdsp_glink_wq");
  535. if (!wpriv->work_queue) {
  536. dev_err(&pdev->dev, "%s: Error creating wdsp_glink_wq\n",
  537. __func__);
  538. return -EINVAL;
  539. }
  540. wdsp_rpmsg_id_table = devm_kzalloc(&pdev->dev,
  541. (sizeof(struct rpmsg_device_id) *
  542. (no_of_channels + 1)),
  543. GFP_KERNEL);
  544. if (!wdsp_rpmsg_id_table) {
  545. ret = -ENOMEM;
  546. goto err;
  547. }
  548. wpriv->no_of_channels = no_of_channels;
  549. id_table = wdsp_rpmsg_id_table;
  550. for (i = 0; i < no_of_channels; i++) {
  551. strlcpy(id_table->name, wpriv->ch[i]->ch_name,
  552. RPMSG_NAME_SIZE);
  553. id_table++;
  554. }
  555. wdsp_rpmsg_driver.id_table = wdsp_rpmsg_id_table;
  556. ret = register_rpmsg_driver(&wdsp_rpmsg_driver);
  557. if (ret < 0) {
  558. dev_err(&pdev->dev, "%s: Rpmsg driver register failed, err = %d\n",
  559. __func__, ret);
  560. goto err;
  561. }
  562. return 0;
  563. err:
  564. destroy_workqueue(wpriv->work_queue);
  565. return ret;
  566. }
  567. static const struct file_operations wdsp_glink_fops = {
  568. .owner = THIS_MODULE,
  569. .open = wdsp_glink_open,
  570. .read = wdsp_glink_read,
  571. .write = wdsp_glink_write,
  572. .flush = wdsp_glink_flush,
  573. .release = wdsp_glink_release,
  574. };
  575. static int wdsp_glink_probe(struct platform_device *pdev)
  576. {
  577. int ret;
  578. struct wdsp_glink_dev *wdev;
  579. wdev = devm_kzalloc(&pdev->dev, sizeof(*wdev), GFP_KERNEL);
  580. if (!wdev) {
  581. ret = -ENOMEM;
  582. goto done;
  583. }
  584. ret = alloc_chrdev_region(&wdev->dev_num, 0, MINOR_NUMBER_COUNT,
  585. WDSP_GLINK_DRIVER_NAME);
  586. if (ret < 0) {
  587. dev_err(&pdev->dev, "%s: Failed to alloc char dev, err = %d\n",
  588. __func__, ret);
  589. goto err_chrdev;
  590. }
  591. wdev->cls = class_create(THIS_MODULE, WDSP_GLINK_DRIVER_NAME);
  592. if (IS_ERR(wdev->cls)) {
  593. ret = PTR_ERR(wdev->cls);
  594. dev_err(&pdev->dev, "%s: Failed to create class, err = %d\n",
  595. __func__, ret);
  596. goto err_class;
  597. }
  598. wdev->dev = device_create(wdev->cls, NULL, wdev->dev_num,
  599. NULL, WDSP_GLINK_DRIVER_NAME);
  600. if (IS_ERR(wdev->dev)) {
  601. ret = PTR_ERR(wdev->dev);
  602. dev_err(&pdev->dev, "%s: Failed to create device, err = %d\n",
  603. __func__, ret);
  604. goto err_dev_create;
  605. }
  606. cdev_init(&wdev->cdev, &wdsp_glink_fops);
  607. ret = cdev_add(&wdev->cdev, wdev->dev_num, MINOR_NUMBER_COUNT);
  608. if (ret < 0) {
  609. dev_err(&pdev->dev, "%s: Failed to register char dev, err = %d\n",
  610. __func__, ret);
  611. goto err_cdev_add;
  612. }
  613. ret = wdsp_register_rpmsg(pdev, wdev);
  614. if (ret < 0) {
  615. dev_err(&pdev->dev, "%s: Failed to register with rpmsg, err = %d\n",
  616. __func__, ret);
  617. goto err_cdev_add;
  618. }
  619. platform_set_drvdata(pdev, wpriv);
  620. goto done;
  621. err_cdev_add:
  622. device_destroy(wdev->cls, wdev->dev_num);
  623. err_dev_create:
  624. class_destroy(wdev->cls);
  625. err_class:
  626. unregister_chrdev_region(0, MINOR_NUMBER_COUNT);
  627. err_chrdev:
  628. done:
  629. return ret;
  630. }
  631. static int wdsp_glink_remove(struct platform_device *pdev)
  632. {
  633. struct wdsp_glink_priv *wpriv = platform_get_drvdata(pdev);
  634. unregister_rpmsg_driver(&wdsp_rpmsg_driver);
  635. if (wpriv) {
  636. flush_workqueue(wpriv->work_queue);
  637. destroy_workqueue(wpriv->work_queue);
  638. if (wpriv->wdev) {
  639. cdev_del(&wpriv->wdev->cdev);
  640. device_destroy(wpriv->wdev->cls, wpriv->wdev->dev_num);
  641. class_destroy(wpriv->wdev->cls);
  642. unregister_chrdev_region(0, MINOR_NUMBER_COUNT);
  643. }
  644. }
  645. return 0;
  646. }
  647. static const struct of_device_id wdsp_glink_of_match[] = {
  648. {.compatible = "qcom,wcd-dsp-glink"},
  649. { }
  650. };
  651. MODULE_DEVICE_TABLE(of, wdsp_glink_of_match);
  652. static struct platform_driver wdsp_glink_driver = {
  653. .probe = wdsp_glink_probe,
  654. .remove = wdsp_glink_remove,
  655. .driver = {
  656. .name = WDSP_GLINK_DRIVER_NAME,
  657. .owner = THIS_MODULE,
  658. .of_match_table = wdsp_glink_of_match,
  659. },
  660. };
  661. static int __init wdsp_glink_init(void)
  662. {
  663. return platform_driver_register(&wdsp_glink_driver);
  664. }
  665. static void __exit wdsp_glink_exit(void)
  666. {
  667. platform_driver_unregister(&wdsp_glink_driver);
  668. }
  669. module_init(wdsp_glink_init);
  670. module_exit(wdsp_glink_exit);
  671. MODULE_DESCRIPTION("SoC WCD_DSP GLINK Driver");
  672. MODULE_LICENSE("GPL v2");