wcd-spi-ac.c 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999
  1. /* Copyright (c) 2018-2019, The Linux Foundation. All rights reserved.
  2. *
  3. * This program is free software; you can redistribute it and/or modify
  4. * it under the terms of the GNU General Public License version 2 and
  5. * only version 2 as published by the Free Software Foundation.
  6. *
  7. * This program is distributed in the hope that it will be useful,
  8. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. * GNU General Public License for more details.
  11. */
  12. #include <linux/init.h>
  13. #include <linux/module.h>
  14. #include <linux/of.h>
  15. #include <linux/platform_device.h>
  16. #include <linux/cdev.h>
  17. #include <linux/proc_fs.h>
  18. #include <linux/poll.h>
  19. #include <linux/slab.h>
  20. #include <linux/notifier.h>
  21. #include <linux/wcd-spi-ac-params.h>
  22. #include <soc/wcd-spi-ac.h>
  23. #include <soc/qcom/msm_qmi_interface.h>
  24. #include "wcd_spi_ctl_v01.h"
  25. #define WCD_SPI_AC_PFS_ENTRY_MAX_LEN 16
  26. #define WCD_SPI_AC_WRITE_CMD_MIN_SIZE \
  27. (sizeof(struct wcd_spi_ac_write_cmd))
  28. #define WCD_SPI_AC_WRITE_CMD_MAX_SIZE \
  29. (WCD_SPI_AC_WRITE_CMD_MIN_SIZE + \
  30. (WCD_SPI_AC_MAX_BUFFERS * \
  31. sizeof(struct wcd_spi_ac_buf_data)))
  32. #define WCD_SPI_AC_MUTEX_LOCK(dev, lock) \
  33. { \
  34. dev_dbg(dev, "%s: mutex_lock(%s)\n", \
  35. __func__, __stringify_1(lock)); \
  36. mutex_lock(&lock); \
  37. }
  38. #define WCD_SPI_AC_MUTEX_UNLOCK(dev, lock) \
  39. { \
  40. dev_dbg(dev, "%s: mutex_unlock(%s)\n", \
  41. __func__, __stringify_1(lock)); \
  42. mutex_unlock(&lock); \
  43. }
  44. /*
  45. * All bits of status should be cleared for SPI access
  46. * to be released.
  47. */
  48. #define WCD_SPI_AC_STATUS_RELEASE_ACCESS 0x00
  49. #define WCD_SPI_AC_LOCAL_ACCESS 0x00
  50. #define WCD_SPI_AC_REMOTE_ACCESS 0x01
  51. #define WCD_SPI_CTL_INS_ID 0
  52. #define WCD_SPI_AC_QMI_TIMEOUT_MS 100
  53. struct wcd_spi_ac_priv {
  54. /* Pointer to device for this driver */
  55. struct device *dev;
  56. /* Pointer to parent's device */
  57. struct device *parent;
  58. /* char dev related */
  59. struct class *cls;
  60. struct device *chardev;
  61. struct cdev cdev;
  62. dev_t cdev_num;
  63. /* proc entry related */
  64. struct proc_dir_entry *pfs_root;
  65. struct proc_dir_entry *pfs_status;
  66. /* service status related */
  67. u8 svc_offline;
  68. u8 svc_offline_change;
  69. wait_queue_head_t svc_poll_wait;
  70. struct mutex status_lock;
  71. /* state maintenence related */
  72. u32 state;
  73. struct mutex state_lock;
  74. u8 current_access;
  75. /* qmi related */
  76. struct qmi_handle *qmi_hdl;
  77. struct work_struct svc_arr_work;
  78. struct work_struct svc_exit_work;
  79. struct notifier_block nb;
  80. struct mutex svc_lock;
  81. struct workqueue_struct *qmi_wq;
  82. struct work_struct recv_msg_work;
  83. };
  84. static void wcd_spi_ac_status_change(struct wcd_spi_ac_priv *ac,
  85. u8 online)
  86. {
  87. WCD_SPI_AC_MUTEX_LOCK(ac->dev, ac->status_lock);
  88. ac->svc_offline = !online;
  89. /* Make sure the write is complete */
  90. wmb();
  91. xchg(&ac->svc_offline_change, 1);
  92. wake_up_interruptible(&ac->svc_poll_wait);
  93. dev_dbg(ac->dev,
  94. "%s request %u offline %u off_change %u\n",
  95. __func__, online, ac->svc_offline,
  96. ac->svc_offline_change);
  97. WCD_SPI_AC_MUTEX_UNLOCK(ac->dev, ac->status_lock);
  98. }
  99. static int wcd_spi_ac_status_open(struct inode *inode,
  100. struct file *file)
  101. {
  102. struct wcd_spi_ac_priv *ac = PDE_DATA(inode);
  103. file->private_data = ac;
  104. return 0;
  105. }
  106. static ssize_t wcd_spi_ac_status_read(struct file *file,
  107. char __user *buffer,
  108. size_t count, loff_t *offset)
  109. {
  110. struct wcd_spi_ac_priv *ac;
  111. char buf[WCD_SPI_AC_PFS_ENTRY_MAX_LEN];
  112. int len, ret;
  113. u8 offline;
  114. ac = (struct wcd_spi_ac_priv *) file->private_data;
  115. if (!ac) {
  116. pr_err("%s: Invalid private data for status\n",
  117. __func__);
  118. return -EINVAL;
  119. }
  120. WCD_SPI_AC_MUTEX_LOCK(ac->dev, ac->status_lock);
  121. offline = ac->svc_offline;
  122. /* Make sure the read is complete */
  123. rmb();
  124. dev_dbg(ac->dev, "%s: offline = %sline\n",
  125. __func__, offline ? "off" : "on");
  126. len = snprintf(buf, sizeof(buf), "%s\n",
  127. offline ? "OFFLINE" : "ONLINE");
  128. ret = simple_read_from_buffer(buffer, count, offset, buf, len);
  129. WCD_SPI_AC_MUTEX_UNLOCK(ac->dev, ac->status_lock);
  130. return ret;
  131. }
  132. static unsigned int wcd_spi_ac_status_poll(struct file *file,
  133. poll_table *wait)
  134. {
  135. struct wcd_spi_ac_priv *ac;
  136. unsigned int ret = 0;
  137. ac = (struct wcd_spi_ac_priv *) file->private_data;
  138. if (!ac) {
  139. pr_err("%s: Invalid private data for status\n",
  140. __func__);
  141. return -EINVAL;
  142. }
  143. dev_dbg(ac->dev, "%s: Poll wait, svc = %s\n",
  144. __func__, ac->svc_offline ? "offline" : "online");
  145. poll_wait(file, &ac->svc_poll_wait, wait);
  146. dev_dbg(ac->dev, "%s: Woken up Poll wait, svc = %s\n",
  147. __func__, ac->svc_offline ? "offline" : "online");
  148. WCD_SPI_AC_MUTEX_LOCK(ac->dev, ac->status_lock);
  149. if (xchg(&ac->svc_offline_change, 0))
  150. ret = POLLIN | POLLPRI | POLLRDNORM;
  151. dev_dbg(ac->dev, "%s: ret (%d) from poll_wait\n",
  152. __func__, ret);
  153. WCD_SPI_AC_MUTEX_UNLOCK(ac->dev, ac->status_lock);
  154. return ret;
  155. }
  156. static const struct file_operations wcd_spi_ac_status_ops = {
  157. .owner = THIS_MODULE,
  158. .open = wcd_spi_ac_status_open,
  159. .read = wcd_spi_ac_status_read,
  160. .poll = wcd_spi_ac_status_poll,
  161. };
  162. static int wcd_spi_ac_procfs_init(struct wcd_spi_ac_priv *ac)
  163. {
  164. int ret = 0;
  165. ac->pfs_root = proc_mkdir(WCD_SPI_AC_PROCFS_DIR_NAME, NULL);
  166. if (!ac->pfs_root) {
  167. dev_err(ac->dev, "%s: proc_mkdir failed\n", __func__);
  168. return -EINVAL;
  169. }
  170. ac->pfs_status = proc_create_data(WCD_SPI_AC_PROCFS_STATE_NAME,
  171. 0444, ac->pfs_root,
  172. &wcd_spi_ac_status_ops,
  173. ac);
  174. if (!ac->pfs_status) {
  175. dev_err(ac->dev, "%s: proc_create_data failed\n",
  176. __func__);
  177. ret = -EINVAL;
  178. goto rmdir_root;
  179. }
  180. proc_set_size(ac->pfs_status, WCD_SPI_AC_PFS_ENTRY_MAX_LEN);
  181. return 0;
  182. rmdir_root:
  183. proc_remove(ac->pfs_root);
  184. return ret;
  185. }
  186. static void wcd_spi_ac_procfs_deinit(struct wcd_spi_ac_priv *ac)
  187. {
  188. proc_remove(ac->pfs_status);
  189. proc_remove(ac->pfs_root);
  190. }
  191. static int wcd_spi_ac_request_access(struct wcd_spi_ac_priv *ac,
  192. bool is_svc_locked)
  193. {
  194. struct wcd_spi_req_access_msg_v01 req;
  195. struct wcd_spi_req_access_resp_v01 rsp;
  196. struct msg_desc req_desc, rsp_desc;
  197. int ret = 0;
  198. dev_dbg(ac->dev, "%s: is_svc_locked = %s\n",
  199. __func__, is_svc_locked ? "true" : "false");
  200. memset(&req, 0, sizeof(req));
  201. memset(&rsp, 0, sizeof(rsp));
  202. req.reason_valid = 1;
  203. req.reason = ac->state & 0x03;
  204. req_desc.max_msg_len = WCD_SPI_REQ_ACCESS_MSG_V01_MAX_MSG_LEN;
  205. req_desc.msg_id = WCD_SPI_REQ_ACCESS_MSG_V01;
  206. req_desc.ei_array = wcd_spi_req_access_msg_v01_ei;
  207. rsp_desc.max_msg_len = WCD_SPI_REQ_ACCESS_RESP_V01_MAX_MSG_LEN;
  208. rsp_desc.msg_id = WCD_SPI_REQ_ACCESS_RESP_V01;
  209. rsp_desc.ei_array = wcd_spi_req_access_resp_v01_ei;
  210. if (!is_svc_locked)
  211. WCD_SPI_AC_MUTEX_LOCK(ac->dev, ac->svc_lock);
  212. ret = qmi_send_req_wait(ac->qmi_hdl,
  213. &req_desc, &req, sizeof(req),
  214. &rsp_desc, &rsp, sizeof(rsp),
  215. WCD_SPI_AC_QMI_TIMEOUT_MS);
  216. if (ret) {
  217. dev_err(ac->dev, "%s: msg send failed %d\n",
  218. __func__, ret);
  219. goto done;
  220. }
  221. if (rsp.resp.result != QMI_RESULT_SUCCESS_V01) {
  222. ret = -EIO;
  223. dev_err(ac->dev, "%s: qmi resp error %d\n",
  224. __func__, rsp.resp.result);
  225. }
  226. done:
  227. if (!is_svc_locked)
  228. WCD_SPI_AC_MUTEX_UNLOCK(ac->dev, ac->svc_lock);
  229. return ret;
  230. }
  231. static int wcd_spi_ac_release_access(struct wcd_spi_ac_priv *ac,
  232. bool is_svc_locked)
  233. {
  234. struct wcd_spi_rel_access_msg_v01 req;
  235. struct wcd_spi_rel_access_resp_v01 rsp;
  236. struct msg_desc req_desc, rsp_desc;
  237. int ret = 0;
  238. dev_dbg(ac->dev, "%s: is_svc_locked = %s\n",
  239. __func__, is_svc_locked ? "true" : "false");
  240. memset(&req, 0, sizeof(req));
  241. memset(&rsp, 0, sizeof(rsp));
  242. req_desc.max_msg_len = WCD_SPI_REL_ACCESS_MSG_V01_MAX_MSG_LEN;
  243. req_desc.msg_id = WCD_SPI_REL_ACCESS_MSG_V01;
  244. req_desc.ei_array = wcd_spi_rel_access_msg_v01_ei;
  245. rsp_desc.max_msg_len = WCD_SPI_REL_ACCESS_RESP_V01_MAX_MSG_LEN;
  246. rsp_desc.msg_id = WCD_SPI_REL_ACCESS_RESP_V01;
  247. rsp_desc.ei_array = wcd_spi_rel_access_resp_v01_ei;
  248. if (!is_svc_locked)
  249. WCD_SPI_AC_MUTEX_LOCK(ac->dev, ac->svc_lock);
  250. ret = qmi_send_req_wait(ac->qmi_hdl,
  251. &req_desc, &req, sizeof(req),
  252. &rsp_desc, &rsp, sizeof(rsp),
  253. WCD_SPI_AC_QMI_TIMEOUT_MS);
  254. if (ret) {
  255. dev_err(ac->dev, "%s: msg send failed %d\n",
  256. __func__, ret);
  257. goto done;
  258. }
  259. if (rsp.resp.result != QMI_RESULT_SUCCESS_V01) {
  260. ret = -EIO;
  261. dev_err(ac->dev, "%s: qmi resp error %d\n",
  262. __func__, rsp.resp.result);
  263. }
  264. done:
  265. if (!is_svc_locked)
  266. WCD_SPI_AC_MUTEX_UNLOCK(ac->dev, ac->svc_lock);
  267. return ret;
  268. }
  269. static int wcd_spi_ac_buf_msg(
  270. struct wcd_spi_ac_priv *ac,
  271. u8 *data, int data_sz)
  272. {
  273. struct wcd_spi_ac_buf_data *buf_data;
  274. struct wcd_spi_buff_msg_v01 req;
  275. struct wcd_spi_buff_resp_v01 rsp;
  276. struct msg_desc req_desc, rsp_desc;
  277. int ret = 0;
  278. memset(&req, 0, sizeof(req));
  279. memset(&rsp, 0, sizeof(rsp));
  280. buf_data = (struct wcd_spi_ac_buf_data *) data;
  281. memcpy(req.buff_addr_1, buf_data,
  282. sizeof(*buf_data));
  283. if (data_sz - sizeof(*buf_data) != 0) {
  284. req.buff_addr_2_valid = 1;
  285. buf_data++;
  286. memcpy(req.buff_addr_2, buf_data,
  287. sizeof(*buf_data));
  288. }
  289. req_desc.max_msg_len = WCD_SPI_BUFF_MSG_V01_MAX_MSG_LEN;
  290. req_desc.msg_id = WCD_SPI_BUFF_MSG_V01;
  291. req_desc.ei_array = wcd_spi_buff_msg_v01_ei;
  292. rsp_desc.max_msg_len = WCD_SPI_BUFF_RESP_V01_MAX_MSG_LEN;
  293. rsp_desc.msg_id = WCD_SPI_BUFF_RESP_V01;
  294. rsp_desc.ei_array = wcd_spi_buff_resp_v01_ei;
  295. WCD_SPI_AC_MUTEX_LOCK(ac->dev, ac->svc_lock);
  296. ret = qmi_send_req_wait(ac->qmi_hdl,
  297. &req_desc, &req, sizeof(req),
  298. &rsp_desc, &rsp, sizeof(rsp),
  299. WCD_SPI_AC_QMI_TIMEOUT_MS);
  300. if (ret) {
  301. dev_err(ac->dev, "%s: msg send failed %d\n",
  302. __func__, ret);
  303. goto done;
  304. }
  305. if (rsp.resp.result != QMI_RESULT_SUCCESS_V01) {
  306. ret = -EIO;
  307. dev_err(ac->dev, "%s: qmi resp error %d\n",
  308. __func__, rsp.resp.result);
  309. }
  310. done:
  311. WCD_SPI_AC_MUTEX_UNLOCK(ac->dev, ac->svc_lock);
  312. return ret;
  313. }
  314. /*
  315. * wcd_spi_ac_set_sync: Sets the current status of the SPI
  316. * bus and requests access if not
  317. * already accesible.
  318. * @ac: pointer to the drivers private data
  319. * @value: value to be set in the status mask
  320. * @is_svc_locked: flag to indicate if svc_lock is acquired by caller
  321. */
  322. static int wcd_spi_ac_set_sync(struct wcd_spi_ac_priv *ac,
  323. u32 value, bool is_svc_locked)
  324. {
  325. int ret = 0;
  326. WCD_SPI_AC_MUTEX_LOCK(ac->dev, ac->state_lock);
  327. ac->state |= value;
  328. /* any non-zero state indicates us to request SPI access */
  329. wmb();
  330. dev_dbg(ac->dev, "%s: current state = 0x%x, current access 0x%x\n",
  331. __func__, ac->state, ac->current_access);
  332. if (ac->current_access == WCD_SPI_AC_REMOTE_ACCESS) {
  333. dev_dbg(ac->dev,
  334. "%s: requesting access, state = 0x%x\n",
  335. __func__, ac->state);
  336. ret = wcd_spi_ac_request_access(ac, is_svc_locked);
  337. if (!ret)
  338. ac->current_access = WCD_SPI_AC_LOCAL_ACCESS;
  339. }
  340. WCD_SPI_AC_MUTEX_UNLOCK(ac->dev, ac->state_lock);
  341. return ret;
  342. }
  343. /*
  344. * wcd_spi_ac_clear_sync: Clears the current status of the SPI
  345. * bus and releases access if applicable
  346. * @ac: pointer to the drivers private data
  347. * @value: value to be cleared in the status mask
  348. * @is_svc_locked: flag to indicate if svc_lock is acquired by caller
  349. */
  350. static int wcd_spi_ac_clear_sync(struct wcd_spi_ac_priv *ac,
  351. u32 value, bool is_svc_locked)
  352. {
  353. int ret = 0;
  354. WCD_SPI_AC_MUTEX_LOCK(ac->dev, ac->state_lock);
  355. ac->state &= ~(value);
  356. /* make sure value is written before read */
  357. wmb();
  358. dev_dbg(ac->dev, "%s: current state = 0x%x, current access 0x%x\n",
  359. __func__, ac->state, ac->current_access);
  360. /* state should be zero to release SPI access */
  361. if (!ac->state &&
  362. ac->current_access == WCD_SPI_AC_LOCAL_ACCESS) {
  363. dev_dbg(ac->dev,
  364. "%s: releasing access, state = 0x%x\n",
  365. __func__, ac->state);
  366. ret = wcd_spi_ac_release_access(ac, is_svc_locked);
  367. if (!ret)
  368. ac->current_access = WCD_SPI_AC_REMOTE_ACCESS;
  369. }
  370. WCD_SPI_AC_MUTEX_UNLOCK(ac->dev, ac->state_lock);
  371. return ret;
  372. }
  373. /*
  374. * wcd_spi_access_ctl: API to request/release the access
  375. * to wcd-spi bus.
  376. * @dev: handle to the wcd-spi-ac device
  377. * @request: enum to indicate access request or access release
  378. * @reason: reason for request/release. Must be one of the
  379. * valid reasons.
  380. * Returns success if the access handover was sucessful,
  381. * negative error code otherwise.
  382. */
  383. int wcd_spi_access_ctl(struct device *dev,
  384. enum wcd_spi_acc_req request,
  385. u32 reason)
  386. {
  387. struct wcd_spi_ac_priv *ac;
  388. int ret = 0;
  389. if (!dev) {
  390. pr_err("%s: invalid device\n", __func__);
  391. return -EINVAL;
  392. }
  393. /* only data_transfer and remote_down are valid reasons */
  394. if (reason != WCD_SPI_AC_DATA_TRANSFER &&
  395. reason != WCD_SPI_AC_REMOTE_DOWN) {
  396. pr_err("%s: Invalid reason 0x%x\n",
  397. __func__, reason);
  398. return -EINVAL;
  399. }
  400. ac = (struct wcd_spi_ac_priv *) dev_get_drvdata(dev);
  401. if (!ac) {
  402. dev_err(dev, "%s: invalid driver data\n", __func__);
  403. return -EINVAL;
  404. }
  405. dev_dbg(dev, "%s: request = 0x%x, reason = 0x%x\n",
  406. __func__, request, reason);
  407. switch (request) {
  408. case WCD_SPI_ACCESS_REQUEST:
  409. ret = wcd_spi_ac_set_sync(ac, reason, false);
  410. if (ret)
  411. dev_err(dev, "%s: set_sync(0x%x) failed %d\n",
  412. __func__, reason, ret);
  413. break;
  414. case WCD_SPI_ACCESS_RELEASE:
  415. ret = wcd_spi_ac_clear_sync(ac, reason, false);
  416. if (ret)
  417. dev_err(dev, "%s: clear_sync(0x%x) failed %d\n",
  418. __func__, reason, ret);
  419. break;
  420. default:
  421. dev_err(dev, "%s: invalid request 0x%x\n",
  422. __func__, request);
  423. break;
  424. }
  425. return ret;
  426. }
  427. EXPORT_SYMBOL(wcd_spi_access_ctl);
  428. static int wcd_spi_ac_cdev_open(struct inode *inode,
  429. struct file *file)
  430. {
  431. struct wcd_spi_ac_priv *ac;
  432. int ret = 0;
  433. ac = container_of(inode->i_cdev, struct wcd_spi_ac_priv, cdev);
  434. if (!ac) {
  435. pr_err("%s: Invalid private data\n", __func__);
  436. return -EINVAL;
  437. }
  438. WCD_SPI_AC_MUTEX_LOCK(ac->dev, ac->status_lock);
  439. if (ac->svc_offline) {
  440. dev_err(ac->dev, "%s: SVC is not online, cannot open driver\n",
  441. __func__);
  442. ret = -ENODEV;
  443. goto done;
  444. }
  445. file->private_data = ac;
  446. done:
  447. WCD_SPI_AC_MUTEX_UNLOCK(ac->dev, ac->status_lock);
  448. return ret;
  449. }
  450. static ssize_t wcd_spi_ac_cdev_write(struct file *file,
  451. const char __user *buf,
  452. size_t count,
  453. loff_t *ppos)
  454. {
  455. struct wcd_spi_ac_priv *ac;
  456. struct wcd_spi_ac_write_cmd *cmd_buf;
  457. int ret = 0;
  458. int data_sz;
  459. ac = (struct wcd_spi_ac_priv *) file->private_data;
  460. if (!ac) {
  461. pr_err("%s: Invalid private data\n", __func__);
  462. return -EINVAL;
  463. }
  464. if (count < WCD_SPI_AC_WRITE_CMD_MIN_SIZE ||
  465. count > WCD_SPI_AC_WRITE_CMD_MAX_SIZE) {
  466. dev_err(ac->dev, "%s: Invalid write count %zd\n",
  467. __func__, count);
  468. return -EINVAL;
  469. }
  470. cmd_buf = kzalloc(count, GFP_KERNEL);
  471. if (!cmd_buf)
  472. return -ENOMEM;
  473. if (get_user(cmd_buf->cmd_type, buf)) {
  474. dev_err(ac->dev, "%s: get_user failed\n", __func__);
  475. ret = -EFAULT;
  476. goto free_cmd_buf;
  477. }
  478. dev_dbg(ac->dev, "%s: write cmd type 0x%x\n",
  479. __func__, cmd_buf->cmd_type);
  480. switch (cmd_buf->cmd_type) {
  481. case WCD_SPI_AC_CMD_CONC_BEGIN:
  482. ret = wcd_spi_ac_set_sync(ac, WCD_SPI_AC_CONCURRENCY, false);
  483. if (ret) {
  484. dev_err(ac->dev, "%s: set_sync(CONC) fail %d\n",
  485. __func__, ret);
  486. goto free_cmd_buf;
  487. }
  488. break;
  489. case WCD_SPI_AC_CMD_CONC_END:
  490. ret = wcd_spi_ac_clear_sync(ac, WCD_SPI_AC_CONCURRENCY, false);
  491. if (ret) {
  492. dev_err(ac->dev, "%s: clear_sync(CONC) fail %d\n",
  493. __func__, ret);
  494. goto free_cmd_buf;
  495. }
  496. break;
  497. case WCD_SPI_AC_CMD_BUF_DATA:
  498. /* Read the buffer details and send to service */
  499. data_sz = count - sizeof(cmd_buf->cmd_type);
  500. if (!data_sz ||
  501. (data_sz % sizeof(struct wcd_spi_ac_buf_data))) {
  502. dev_err(ac->dev, "%s: size %d not multiple of %ld\n",
  503. __func__, data_sz,
  504. sizeof(struct wcd_spi_ac_buf_data));
  505. goto free_cmd_buf;
  506. }
  507. if (data_sz / sizeof(struct wcd_spi_ac_buf_data) >
  508. WCD_SPI_AC_MAX_BUFFERS) {
  509. dev_err(ac->dev, "%s: invalid size %d\n",
  510. __func__, data_sz);
  511. goto free_cmd_buf;
  512. }
  513. if (copy_from_user(cmd_buf->payload,
  514. buf + sizeof(cmd_buf->cmd_type),
  515. data_sz)) {
  516. dev_err(ac->dev, "%s: copy_from_user failed\n",
  517. __func__);
  518. ret = -EFAULT;
  519. goto free_cmd_buf;
  520. }
  521. ret = wcd_spi_ac_buf_msg(ac, cmd_buf->payload, data_sz);
  522. if (ret) {
  523. dev_err(ac->dev, "%s: _buf_msg failed %d\n",
  524. __func__, ret);
  525. goto free_cmd_buf;
  526. }
  527. ret = wcd_spi_ac_clear_sync(ac, WCD_SPI_AC_UNINITIALIZED,
  528. false);
  529. if (ret) {
  530. dev_err(ac->dev, "%s: clear_sync 0x%lx failed %d\n",
  531. __func__, WCD_SPI_AC_UNINITIALIZED, ret);
  532. goto free_cmd_buf;
  533. }
  534. break;
  535. default:
  536. dev_err(ac->dev, "%s: Invalid cmd_type 0x%x\n",
  537. __func__, cmd_buf->cmd_type);
  538. ret = -EINVAL;
  539. goto free_cmd_buf;
  540. }
  541. free_cmd_buf:
  542. kfree(cmd_buf);
  543. if (!ret)
  544. ret = count;
  545. return ret;
  546. }
  547. static int wcd_spi_ac_cdev_release(struct inode *inode,
  548. struct file *file)
  549. {
  550. struct wcd_spi_ac_priv *ac;
  551. int ret = 0;
  552. ac = (struct wcd_spi_ac_priv *) file->private_data;
  553. if (!ac) {
  554. pr_err("%s: Invalid private data\n", __func__);
  555. return -EINVAL;
  556. }
  557. ret = wcd_spi_ac_set_sync(ac, WCD_SPI_AC_UNINITIALIZED, false);
  558. if (ret)
  559. dev_err(ac->dev, "%s: set_sync(UNINITIALIZED) failed %d\n",
  560. __func__, ret);
  561. return ret;
  562. }
  563. static const struct file_operations wcd_spi_ac_cdev_fops = {
  564. .owner = THIS_MODULE,
  565. .open = wcd_spi_ac_cdev_open,
  566. .write = wcd_spi_ac_cdev_write,
  567. .release = wcd_spi_ac_cdev_release,
  568. };
  569. static int wcd_spi_ac_reg_chardev(struct wcd_spi_ac_priv *ac)
  570. {
  571. int ret;
  572. ret = alloc_chrdev_region(&ac->cdev_num, 0, 1,
  573. WCD_SPI_AC_CLIENT_CDEV_NAME);
  574. if (ret) {
  575. dev_err(ac->dev, "%s: alloc_chrdev_region failed %d\n",
  576. __func__, ret);
  577. return ret;
  578. }
  579. ac->cls = class_create(THIS_MODULE, WCD_SPI_AC_CLIENT_CDEV_NAME);
  580. if (IS_ERR(ac->cls)) {
  581. ret = PTR_ERR(ac->cls);
  582. dev_err(ac->dev, "%s: class_create failed %d\n",
  583. __func__, ret);
  584. goto unregister_chrdev;
  585. }
  586. ac->chardev = device_create(ac->cls, NULL, ac->cdev_num,
  587. NULL, WCD_SPI_AC_CLIENT_CDEV_NAME);
  588. if (IS_ERR(ac->chardev)) {
  589. ret = PTR_ERR(ac->chardev);
  590. dev_err(ac->dev, "%s: device_create failed %d\n",
  591. __func__, ret);
  592. goto destroy_class;
  593. }
  594. cdev_init(&ac->cdev, &wcd_spi_ac_cdev_fops);
  595. ret = cdev_add(&ac->cdev, ac->cdev_num, 1);
  596. if (ret) {
  597. dev_err(ac->dev, "%s: cdev_add failed %d\n",
  598. __func__, ret);
  599. goto destroy_device;
  600. }
  601. return 0;
  602. destroy_device:
  603. device_destroy(ac->cls, ac->cdev_num);
  604. destroy_class:
  605. class_destroy(ac->cls);
  606. unregister_chrdev:
  607. unregister_chrdev_region(0, 1);
  608. return ret;
  609. }
  610. static int wcd_spi_ac_unreg_chardev(struct wcd_spi_ac_priv *ac)
  611. {
  612. cdev_del(&ac->cdev);
  613. device_destroy(ac->cls, ac->cdev_num);
  614. class_destroy(ac->cls);
  615. unregister_chrdev_region(0, 1);
  616. return 0;
  617. }
  618. static void wcd_spi_ac_recv_msg(struct work_struct *work)
  619. {
  620. struct wcd_spi_ac_priv *ac;
  621. int rc = 0;
  622. ac = container_of(work, struct wcd_spi_ac_priv,
  623. recv_msg_work);
  624. if (!ac) {
  625. pr_err("%s: Invalid private data\n", __func__);
  626. return;
  627. }
  628. do {
  629. dev_dbg(ac->dev, "%s: msg received, rc = %d\n",
  630. __func__, rc);
  631. } while ((rc = qmi_recv_msg(ac->qmi_hdl)) == 0);
  632. if (rc != -ENOMSG)
  633. dev_err(ac->dev, "%s: qmi_recv_msg failed %d\n",
  634. __func__, rc);
  635. }
  636. static void wcd_spi_ac_clnt_notify(struct qmi_handle *hdl,
  637. enum qmi_event_type event, void *priv_data)
  638. {
  639. struct wcd_spi_ac_priv *ac;
  640. if (!priv_data) {
  641. pr_err("%s: Invalid private data\n", __func__);
  642. return;
  643. }
  644. ac = (struct wcd_spi_ac_priv *) priv_data;
  645. switch (event) {
  646. case QMI_RECV_MSG:
  647. queue_work(ac->qmi_wq, &ac->recv_msg_work);
  648. break;
  649. default:
  650. break;
  651. }
  652. }
  653. static void wcd_spi_ac_svc_arrive(struct work_struct *work)
  654. {
  655. struct wcd_spi_ac_priv *ac;
  656. int ret;
  657. ac = container_of(work, struct wcd_spi_ac_priv,
  658. svc_arr_work);
  659. if (!ac) {
  660. pr_err("%s: Invalid private data\n",
  661. __func__);
  662. return;
  663. }
  664. WCD_SPI_AC_MUTEX_LOCK(ac->dev, ac->svc_lock);
  665. ac->qmi_hdl = qmi_handle_create(wcd_spi_ac_clnt_notify,
  666. ac);
  667. if (!ac->qmi_hdl) {
  668. dev_err(ac->dev, "%s: qmi_handle_create failed\n",
  669. __func__);
  670. goto done;
  671. }
  672. ret = qmi_connect_to_service(ac->qmi_hdl,
  673. WCD_SPI_CTL_SERVICE_ID_V01,
  674. WCD_SPI_CTL_SERVICE_VERS_V01,
  675. WCD_SPI_CTL_INS_ID);
  676. if (ret) {
  677. dev_err(ac->dev, "%s, cant connect to service, error %d\n",
  678. __func__, ret);
  679. qmi_handle_destroy(ac->qmi_hdl);
  680. ac->qmi_hdl = NULL;
  681. goto done;
  682. }
  683. /* Mark service as online */
  684. wcd_spi_ac_status_change(ac, 1);
  685. /*
  686. * update the state and clear the WCD_SPI_AC_SVC_OFFLINE
  687. * bit to indicate that the service is now online.
  688. */
  689. ret = wcd_spi_ac_clear_sync(ac, WCD_SPI_AC_SVC_OFFLINE, true);
  690. if (ret)
  691. dev_err(ac->dev, "%s: clear_sync(SVC_OFFLINE) failed %d\n",
  692. __func__, ret);
  693. done:
  694. WCD_SPI_AC_MUTEX_UNLOCK(ac->dev, ac->svc_lock);
  695. }
  696. static void wcd_spi_ac_svc_exit(struct work_struct *work)
  697. {
  698. struct wcd_spi_ac_priv *ac;
  699. int ret = 0;
  700. ac = container_of(work, struct wcd_spi_ac_priv,
  701. svc_exit_work);
  702. if (!ac) {
  703. pr_err("%s: Invalid private data\n",
  704. __func__);
  705. return;
  706. }
  707. WCD_SPI_AC_MUTEX_LOCK(ac->dev, ac->svc_lock);
  708. ret = wcd_spi_ac_set_sync(ac, WCD_SPI_AC_SVC_OFFLINE, true);
  709. if (ret)
  710. dev_err(ac->dev, "%s: set_sync(SVC_OFFLINE) failed %d\n",
  711. __func__, ret);
  712. qmi_handle_destroy(ac->qmi_hdl);
  713. ac->qmi_hdl = NULL;
  714. wcd_spi_ac_status_change(ac, 0);
  715. WCD_SPI_AC_MUTEX_UNLOCK(ac->dev, ac->svc_lock);
  716. }
  717. static int wcd_spi_ac_svc_event(struct notifier_block *this,
  718. unsigned long event,
  719. void *data)
  720. {
  721. struct wcd_spi_ac_priv *ac;
  722. ac = container_of(this, struct wcd_spi_ac_priv, nb);
  723. if (!ac) {
  724. pr_err("%s: Invalid private data\n", __func__);
  725. return -EINVAL;
  726. }
  727. dev_dbg(ac->dev, "%s: event = 0x%lx", __func__, event);
  728. switch (event) {
  729. case QMI_SERVER_ARRIVE:
  730. schedule_work(&ac->svc_arr_work);
  731. break;
  732. case QMI_SERVER_EXIT:
  733. schedule_work(&ac->svc_exit_work);
  734. break;
  735. default:
  736. dev_err(ac->dev, "%s unhandled event %ld\n",
  737. __func__, event);
  738. break;
  739. }
  740. return 0;
  741. }
  742. static int wcd_spi_ac_probe(struct platform_device *pdev)
  743. {
  744. struct wcd_spi_ac_priv *ac;
  745. struct device *parent = pdev->dev.parent;
  746. int ret = 0;
  747. ac = devm_kzalloc(&pdev->dev, sizeof(*ac),
  748. GFP_KERNEL);
  749. if (!ac)
  750. return -ENOMEM;
  751. ac->dev = &pdev->dev;
  752. ac->parent = parent;
  753. ret = wcd_spi_ac_reg_chardev(ac);
  754. if (ret)
  755. return ret;
  756. ret = wcd_spi_ac_procfs_init(ac);
  757. if (ret)
  758. goto unreg_chardev;
  759. mutex_init(&ac->status_lock);
  760. mutex_init(&ac->state_lock);
  761. mutex_init(&ac->svc_lock);
  762. init_waitqueue_head(&ac->svc_poll_wait);
  763. ac->svc_offline = 1;
  764. ac->state = (WCD_SPI_AC_SVC_OFFLINE |
  765. WCD_SPI_AC_UNINITIALIZED);
  766. ac->current_access = WCD_SPI_AC_LOCAL_ACCESS;
  767. ac->nb.notifier_call = wcd_spi_ac_svc_event;
  768. INIT_WORK(&ac->svc_arr_work, wcd_spi_ac_svc_arrive);
  769. INIT_WORK(&ac->svc_exit_work, wcd_spi_ac_svc_exit);
  770. INIT_WORK(&ac->recv_msg_work, wcd_spi_ac_recv_msg);
  771. ac->qmi_wq = create_singlethread_workqueue("qmi_wq");
  772. if (!ac->qmi_wq) {
  773. dev_err(&pdev->dev,
  774. "%s: create_singlethread_workqueue failed\n",
  775. __func__);
  776. goto deinit_procfs;
  777. }
  778. dev_set_drvdata(&pdev->dev, ac);
  779. ret = qmi_svc_event_notifier_register(
  780. WCD_SPI_CTL_SERVICE_ID_V01,
  781. WCD_SPI_CTL_SERVICE_VERS_V01,
  782. WCD_SPI_CTL_INS_ID,
  783. &ac->nb);
  784. if (ret) {
  785. dev_err(&pdev->dev,
  786. "%s: qmi_svc_event_notifier_register failed %d\n",
  787. __func__, ret);
  788. goto destroy_wq;
  789. }
  790. return 0;
  791. destroy_wq:
  792. destroy_workqueue(ac->qmi_wq);
  793. dev_set_drvdata(&pdev->dev, NULL);
  794. deinit_procfs:
  795. wcd_spi_ac_procfs_deinit(ac);
  796. mutex_destroy(&ac->status_lock);
  797. mutex_destroy(&ac->state_lock);
  798. mutex_destroy(&ac->svc_lock);
  799. unreg_chardev:
  800. wcd_spi_ac_unreg_chardev(ac);
  801. return ret;
  802. }
  803. static int wcd_spi_ac_remove(struct platform_device *pdev)
  804. {
  805. struct wcd_spi_ac_priv *ac;
  806. ac = dev_get_drvdata(&pdev->dev);
  807. qmi_svc_event_notifier_unregister(
  808. WCD_SPI_CTL_SERVICE_ID_V01,
  809. WCD_SPI_CTL_SERVICE_VERS_V01,
  810. WCD_SPI_CTL_INS_ID,
  811. &ac->nb);
  812. if (ac->qmi_wq)
  813. destroy_workqueue(ac->qmi_wq);
  814. wcd_spi_ac_unreg_chardev(ac);
  815. wcd_spi_ac_procfs_deinit(ac);
  816. mutex_destroy(&ac->status_lock);
  817. mutex_destroy(&ac->state_lock);
  818. mutex_destroy(&ac->svc_lock);
  819. return 0;
  820. }
  821. static const struct of_device_id wcd_spi_ac_of_match[] = {
  822. { .compatible = "qcom,wcd-spi-ac" },
  823. { },
  824. };
  825. MODULE_DEVICE_TABLE(of, wcd_spi_ac_of_match);
  826. static struct platform_driver wcd_spi_ac_driver = {
  827. .driver = {
  828. .name = "qcom,wcd-spi-ac",
  829. .of_match_table = wcd_spi_ac_of_match,
  830. .suppress_bind_attrs = true,
  831. },
  832. .probe = wcd_spi_ac_probe,
  833. .remove = wcd_spi_ac_remove,
  834. };
  835. module_platform_driver(wcd_spi_ac_driver);
  836. MODULE_DESCRIPTION("WCD SPI access control driver");
  837. MODULE_LICENSE("GPL v2");