ipa_odl.c 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2018-2019, The Linux Foundation. All rights reserved.
  4. */
  5. #include "ipa_i.h"
  6. #include "ipa_odl.h"
  7. #include <linux/msm_ipa.h>
  8. #include <linux/sched/signal.h>
  9. #include <linux/poll.h>
  10. struct ipa_odl_context *ipa3_odl_ctx;
  11. static DECLARE_WAIT_QUEUE_HEAD(odl_ctl_msg_wq);
  12. static void print_ipa_odl_state_bit_mask(void)
  13. {
  14. IPADBG("ipa3_odl_ctx->odl_state.odl_init --> %d\n",
  15. ipa3_odl_ctx->odl_state.odl_init);
  16. IPADBG("ipa3_odl_ctx->odl_state.odl_open --> %d\n",
  17. ipa3_odl_ctx->odl_state.odl_open);
  18. IPADBG("ipa3_odl_ctx->odl_state.adpl_open --> %d\n",
  19. ipa3_odl_ctx->odl_state.adpl_open);
  20. IPADBG("ipa3_odl_ctx->odl_state.aggr_byte_limit_sent --> %d\n",
  21. ipa3_odl_ctx->odl_state.aggr_byte_limit_sent);
  22. IPADBG("ipa3_odl_ctx->odl_state.odl_ep_setup --> %d\n",
  23. ipa3_odl_ctx->odl_state.odl_ep_setup);
  24. IPADBG("ipa3_odl_ctx->odl_state.odl_setup_done_sent --> %d\n",
  25. ipa3_odl_ctx->odl_state.odl_setup_done_sent);
  26. IPADBG("ipa3_odl_ctx->odl_state.odl_ep_info_sent --> %d\n",
  27. ipa3_odl_ctx->odl_state.odl_ep_info_sent);
  28. IPADBG("ipa3_odl_ctx->odl_state.odl_connected --> %d\n",
  29. ipa3_odl_ctx->odl_state.odl_connected);
  30. IPADBG("ipa3_odl_ctx->odl_state.odl_disconnected --> %d\n\n",
  31. ipa3_odl_ctx->odl_state.odl_disconnected);
  32. }
  33. static int ipa_odl_ctl_fops_open(struct inode *inode, struct file *filp)
  34. {
  35. int ret = 0;
  36. if (ipa3_odl_ctx->odl_state.odl_init) {
  37. ipa3_odl_ctx->odl_state.odl_open = true;
  38. } else {
  39. IPAERR("Before odl init trying to open odl ctl pipe\n");
  40. print_ipa_odl_state_bit_mask();
  41. ret = -ENODEV;
  42. }
  43. return ret;
  44. }
  45. static int ipa_odl_ctl_fops_release(struct inode *inode, struct file *filp)
  46. {
  47. IPADBG("QTI closed ipa_odl_ctl node\n");
  48. ipa3_odl_ctx->odl_state.odl_open = false;
  49. return 0;
  50. }
  51. /**
  52. * ipa_odl_ctl_fops_read() - read message from IPA ODL device
  53. * @filp: [in] file pointer
  54. * @buf: [out] buffer to read into
  55. * @count: [in] size of above buffer
  56. * @f_pos: [inout] file position
  57. *
  58. * Uer-space should continuously read from /dev/ipa_odl_ctl,
  59. * read will block when there are no messages to read.
  60. * Upon return, user-space should read the u32 data from the
  61. * start of the buffer.
  62. *
  63. * 0 --> ODL disconnected.
  64. * 1 --> ODL connected.
  65. *
  66. * Buffer supplied must be big enough to
  67. * hold the message of size u32.
  68. *
  69. * Returns: how many bytes copied to buffer
  70. *
  71. * Note: Should not be called from atomic context
  72. */
  73. static ssize_t ipa_odl_ctl_fops_read(struct file *filp, char __user *buf,
  74. size_t count, loff_t *f_pos)
  75. {
  76. char __user *start;
  77. u8 data;
  78. int ret = 0;
  79. static bool old_state;
  80. bool new_state = false;
  81. start = buf;
  82. ipa3_odl_ctx->odl_ctl_msg_wq_flag = false;
  83. if (!ipa3_odl_ctx->odl_state.adpl_open &&
  84. !ipa3_odl_ctx->odl_state.odl_disconnected) {
  85. IPADBG("Failed to send data odl pipe already disconnected\n");
  86. ret = -EFAULT;
  87. goto send_failed;
  88. }
  89. if (ipa3_odl_ctx->odl_state.odl_ep_setup)
  90. new_state = true;
  91. else if (ipa3_odl_ctx->odl_state.odl_disconnected)
  92. new_state = false;
  93. else {
  94. IPADBG("Failed to send data odl already running\n");
  95. ret = -EFAULT;
  96. goto send_failed;
  97. }
  98. if (old_state != new_state) {
  99. old_state = new_state;
  100. if (new_state)
  101. data = 1;
  102. else if (!new_state)
  103. data = 0;
  104. if (copy_to_user(buf, &data,
  105. sizeof(data))) {
  106. IPADBG("Cpoying data to user failed\n");
  107. ret = -EFAULT;
  108. goto send_failed;
  109. }
  110. buf += sizeof(data);
  111. if (data == 1)
  112. ipa3_odl_ctx->odl_state.odl_setup_done_sent =
  113. true;
  114. }
  115. if (start != buf && ret != -EFAULT)
  116. ret = buf - start;
  117. send_failed:
  118. return ret;
  119. }
  120. static unsigned int ipa_odl_ctl_fops_poll(struct file *file, poll_table *wait)
  121. {
  122. unsigned int mask = 0;
  123. poll_wait(file, &odl_ctl_msg_wq, wait);
  124. if (ipa3_odl_ctx->odl_ctl_msg_wq_flag) {
  125. IPADBG("Sending read mask to odl control pipe\n");
  126. mask |= POLLIN | POLLRDNORM;
  127. }
  128. return mask;
  129. }
  130. static long ipa_odl_ctl_fops_ioctl(struct file *filp, unsigned int cmd,
  131. unsigned long arg)
  132. {
  133. struct ipa_odl_ep_info ep_info = {0};
  134. struct ipa_odl_modem_config status;
  135. int retval = 0;
  136. IPADBG("Calling odl ioctl cmd = %d\n", cmd);
  137. if (!ipa3_odl_ctx->odl_state.odl_setup_done_sent) {
  138. IPAERR("Before complete the odl setup trying calling ioctl\n");
  139. print_ipa_odl_state_bit_mask();
  140. retval = -ENODEV;
  141. goto fail;
  142. }
  143. switch (cmd) {
  144. case IPA_IOC_ODL_QUERY_ADAPL_EP_INFO:
  145. /* Send ep_info to user APP */
  146. ep_info.ep_type = ODL_EP_TYPE_HSUSB;
  147. ep_info.peripheral_iface_id = ODL_EP_PERIPHERAL_IFACE_ID;
  148. ep_info.cons_pipe_num = -1;
  149. ep_info.prod_pipe_num =
  150. ipa3_odl_ctx->odl_client_hdl;
  151. if (copy_to_user((void __user *)arg, &ep_info,
  152. sizeof(ep_info))) {
  153. retval = -EFAULT;
  154. goto fail;
  155. }
  156. ipa3_odl_ctx->odl_state.odl_ep_info_sent = true;
  157. break;
  158. case IPA_IOC_ODL_QUERY_MODEM_CONFIG:
  159. IPADBG("Received the IPA_IOC_ODL_QUERY_MODEM_CONFIG :\n");
  160. if (copy_from_user(&status, (const void __user *)arg,
  161. sizeof(status))) {
  162. retval = -EFAULT;
  163. break;
  164. }
  165. if (status.config_status == CONFIG_SUCCESS)
  166. ipa3_odl_ctx->odl_state.odl_connected = true;
  167. IPADBG("status.config_status = %d odl_connected = %d\n",
  168. status.config_status, ipa3_odl_ctx->odl_state.odl_connected);
  169. break;
  170. default:
  171. retval = -ENOIOCTLCMD;
  172. break;
  173. }
  174. fail:
  175. return retval;
  176. }
  177. static void delete_first_node(void)
  178. {
  179. struct ipa3_push_msg_odl *msg;
  180. if (!list_empty(&ipa3_odl_ctx->adpl_msg_list)) {
  181. msg = list_first_entry(&ipa3_odl_ctx->adpl_msg_list,
  182. struct ipa3_push_msg_odl, link);
  183. if (msg) {
  184. list_del(&msg->link);
  185. kfree(msg->buff);
  186. kfree(msg);
  187. ipa3_odl_ctx->stats.odl_drop_pkt++;
  188. if (atomic_read(&ipa3_odl_ctx->stats.numer_in_queue))
  189. atomic_dec(&ipa3_odl_ctx->stats.numer_in_queue);
  190. }
  191. } else {
  192. IPADBG("List Empty\n");
  193. }
  194. }
  195. int ipa3_send_adpl_msg(unsigned long skb_data)
  196. {
  197. struct ipa3_push_msg_odl *msg;
  198. struct sk_buff *skb = (struct sk_buff *)skb_data;
  199. void *data;
  200. IPADBG_LOW("Processing DPL data\n");
  201. msg = kzalloc(sizeof(struct ipa3_push_msg_odl), GFP_KERNEL);
  202. if (msg == NULL) {
  203. IPADBG("Memory allocation failed\n");
  204. return -ENOMEM;
  205. }
  206. data = kmemdup(skb->data, skb->len, GFP_KERNEL);
  207. if (data == NULL) {
  208. kfree(msg);
  209. return -ENOMEM;
  210. }
  211. memcpy(data, skb->data, skb->len);
  212. msg->buff = data;
  213. msg->len = skb->len;
  214. mutex_lock(&ipa3_odl_ctx->adpl_msg_lock);
  215. if (atomic_read(&ipa3_odl_ctx->stats.numer_in_queue) >=
  216. MAX_QUEUE_TO_ODL)
  217. delete_first_node();
  218. list_add_tail(&msg->link, &ipa3_odl_ctx->adpl_msg_list);
  219. atomic_inc(&ipa3_odl_ctx->stats.numer_in_queue);
  220. mutex_unlock(&ipa3_odl_ctx->adpl_msg_lock);
  221. IPA_STATS_INC_CNT(ipa3_odl_ctx->stats.odl_rx_pkt);
  222. return 0;
  223. }
  224. /**
  225. * odl_ipa_packet_receive_notify() - Rx notify
  226. *
  227. * @priv: driver context
  228. * @evt: event type
  229. * @data: data provided with event
  230. *
  231. * IPA will pass a packet to the Linux network stack with skb->data
  232. */
  233. static void odl_ipa_packet_receive_notify(void *priv,
  234. enum ipa_dp_evt_type evt,
  235. unsigned long data)
  236. {
  237. IPADBG_LOW("Rx packet was received\n");
  238. if (evt == IPA_RECEIVE)
  239. ipa3_send_adpl_msg(data);
  240. else
  241. IPAERR("Invalid evt %d received in wan_ipa_receive\n", evt);
  242. }
  243. int ipa_setup_odl_pipe(void)
  244. {
  245. struct ipa_sys_connect_params *ipa_odl_ep_cfg;
  246. int ret;
  247. ipa_odl_ep_cfg = &ipa3_odl_ctx->odl_sys_param;
  248. IPADBG("Setting up the odl endpoint\n");
  249. ipa_odl_ep_cfg->ipa_ep_cfg.cfg.cs_offload_en = IPA_ENABLE_CS_OFFLOAD_DL;
  250. ipa_odl_ep_cfg->ipa_ep_cfg.aggr.aggr_en = IPA_ENABLE_AGGR;
  251. ipa_odl_ep_cfg->ipa_ep_cfg.aggr.aggr_hard_byte_limit_en = 1;
  252. ipa_odl_ep_cfg->ipa_ep_cfg.aggr.aggr = IPA_GENERIC;
  253. ipa_odl_ep_cfg->ipa_ep_cfg.aggr.aggr_byte_limit =
  254. IPA_ODL_AGGR_BYTE_LIMIT;
  255. ipa_odl_ep_cfg->ipa_ep_cfg.aggr.aggr_pkt_limit = 0;
  256. ipa_odl_ep_cfg->ipa_ep_cfg.hdr.hdr_len = 4;
  257. ipa_odl_ep_cfg->ipa_ep_cfg.hdr.hdr_ofst_metadata_valid = 1;
  258. ipa_odl_ep_cfg->ipa_ep_cfg.hdr.hdr_ofst_metadata = 1;
  259. ipa_odl_ep_cfg->ipa_ep_cfg.hdr.hdr_ofst_pkt_size_valid = 1;
  260. ipa_odl_ep_cfg->ipa_ep_cfg.hdr.hdr_ofst_pkt_size = 2;
  261. ipa_odl_ep_cfg->ipa_ep_cfg.hdr_ext.hdr_total_len_or_pad_valid = true;
  262. ipa_odl_ep_cfg->ipa_ep_cfg.hdr_ext.hdr_total_len_or_pad = 0;
  263. ipa_odl_ep_cfg->ipa_ep_cfg.hdr_ext.hdr_payload_len_inc_padding = true;
  264. ipa_odl_ep_cfg->ipa_ep_cfg.hdr_ext.hdr_total_len_or_pad_offset = 0;
  265. ipa_odl_ep_cfg->ipa_ep_cfg.hdr_ext.hdr_little_endian = 0;
  266. ipa_odl_ep_cfg->ipa_ep_cfg.metadata_mask.metadata_mask = 0xFF000000;
  267. ipa_odl_ep_cfg->client = IPA_CLIENT_ODL_DPL_CONS;
  268. ipa_odl_ep_cfg->notify = odl_ipa_packet_receive_notify;
  269. ipa_odl_ep_cfg->napi_obj = NULL;
  270. ipa_odl_ep_cfg->desc_fifo_sz = IPA_ODL_RX_RING_SIZE *
  271. IPA_FIFO_ELEMENT_SIZE;
  272. ipa3_odl_ctx->odl_client_hdl = -1;
  273. /* For MHIP, ODL functionality is DMA. So bypass aggregation, checksum
  274. * offload, hdr_len.
  275. */
  276. if (ipa3_ctx->platform_type == IPA_PLAT_TYPE_APQ &&
  277. ipa3_is_mhip_offload_enabled()) {
  278. IPADBG("MHIP enabled: bypass aggr + csum offload for ODL");
  279. ipa_odl_ep_cfg->ipa_ep_cfg.aggr.aggr_en = IPA_BYPASS_AGGR;
  280. ipa_odl_ep_cfg->ipa_ep_cfg.cfg.cs_offload_en =
  281. IPA_DISABLE_CS_OFFLOAD;
  282. ipa_odl_ep_cfg->ipa_ep_cfg.hdr.hdr_len = 0;
  283. }
  284. ret = ipa3_setup_sys_pipe(ipa_odl_ep_cfg,
  285. &ipa3_odl_ctx->odl_client_hdl);
  286. return ret;
  287. }
  288. /**
  289. * ipa3_odl_register_pm - Register odl client for PM
  290. *
  291. * This function will register 1 client with IPA PM to represent odl
  292. * in clock scaling calculation:
  293. * - "ODL" - this client will be activated when pipe connected
  294. */
  295. static int ipa3_odl_register_pm(void)
  296. {
  297. int result = 0;
  298. struct ipa_pm_register_params pm_reg;
  299. memset(&pm_reg, 0, sizeof(pm_reg));
  300. pm_reg.name = "ODL";
  301. pm_reg.group = IPA_PM_GROUP_DEFAULT;
  302. pm_reg.skip_clk_vote = true;
  303. result = ipa_pm_register(&pm_reg, &ipa3_odl_ctx->odl_pm_hdl);
  304. if (result) {
  305. IPAERR("failed to create IPA PM client %d\n", result);
  306. return result;
  307. }
  308. return result;
  309. }
  310. int ipa3_odl_pipe_open(void)
  311. {
  312. int ret = 0;
  313. struct ipa_ep_cfg_holb holb_cfg;
  314. if (!ipa3_odl_ctx->odl_state.adpl_open) {
  315. IPAERR("adpl pipe not configured\n");
  316. return 0;
  317. }
  318. memset(&holb_cfg, 0, sizeof(holb_cfg));
  319. holb_cfg.tmr_val = 0;
  320. holb_cfg.en = 1;
  321. ipa3_cfg_ep_holb_by_client(IPA_CLIENT_USB_DPL_CONS, &holb_cfg);
  322. ret = ipa_setup_odl_pipe();
  323. if (ret) {
  324. IPAERR(" Setup endpoint config failed\n");
  325. ipa3_odl_ctx->odl_state.adpl_open = false;
  326. goto fail;
  327. }
  328. ipa3_cfg_ep_holb_by_client(IPA_CLIENT_ODL_DPL_CONS, &holb_cfg);
  329. ipa3_odl_ctx->odl_state.odl_ep_setup = true;
  330. IPADBG("Setup endpoint config success\n");
  331. ipa3_odl_ctx->stats.odl_drop_pkt = 0;
  332. atomic_set(&ipa3_odl_ctx->stats.numer_in_queue, 0);
  333. ipa3_odl_ctx->stats.odl_rx_pkt = 0;
  334. ipa3_odl_ctx->stats.odl_tx_diag_pkt = 0;
  335. /*
  336. * Send signal to ipa_odl_ctl_fops_read,
  337. * to send ODL ep open notification
  338. */
  339. if (ipa3_is_mhip_offload_enabled()) {
  340. IPADBG("MHIP is enabled, continue\n");
  341. ipa3_odl_ctx->odl_state.odl_open = true;
  342. ipa3_odl_ctx->odl_state.odl_setup_done_sent = true;
  343. ipa3_odl_ctx->odl_state.odl_ep_info_sent = true;
  344. ipa3_odl_ctx->odl_state.odl_connected = true;
  345. ipa3_odl_ctx->odl_state.odl_disconnected = false;
  346. /* Enable ADPL over ODL for MPM */
  347. ret = ipa3_mpm_enable_adpl_over_odl(true);
  348. if (ret) {
  349. IPAERR("mpm failed to enable ADPL over ODL %d\n", ret);
  350. return ret;
  351. }
  352. } else {
  353. ipa3_odl_ctx->odl_ctl_msg_wq_flag = true;
  354. IPAERR("Wake up odl ctl\n");
  355. wake_up_interruptible(&odl_ctl_msg_wq);
  356. if (ipa3_odl_ctx->odl_state.odl_disconnected)
  357. ipa3_odl_ctx->odl_state.odl_disconnected = false;
  358. }
  359. fail:
  360. return ret;
  361. }
  362. static int ipa_adpl_open(struct inode *inode, struct file *filp)
  363. {
  364. int ret = 0;
  365. IPADBG("Called the function :\n");
  366. if (ipa3_odl_ctx->odl_state.odl_init &&
  367. !ipa3_odl_ctx->odl_state.adpl_open) {
  368. /* Activate ipa_pm*/
  369. ret = ipa_pm_activate_sync(ipa3_odl_ctx->odl_pm_hdl);
  370. if (ret)
  371. IPAERR("failed to activate pm\n");
  372. ipa3_odl_ctx->odl_state.adpl_open = true;
  373. ret = ipa3_odl_pipe_open();
  374. } else {
  375. IPAERR("Before odl init trying to open adpl pipe\n");
  376. print_ipa_odl_state_bit_mask();
  377. ret = -ENODEV;
  378. }
  379. return ret;
  380. }
  381. static int ipa_adpl_release(struct inode *inode, struct file *filp)
  382. {
  383. int ret = 0;
  384. /* Deactivate ipa_pm */
  385. ret = ipa_pm_deactivate_sync(ipa3_odl_ctx->odl_pm_hdl);
  386. if (ret)
  387. IPAERR("failed to activate pm\n");
  388. ipa3_odl_pipe_cleanup(false);
  389. /* Disable ADPL over ODL for MPM */
  390. if (ipa3_is_mhip_offload_enabled()) {
  391. ret = ipa3_mpm_enable_adpl_over_odl(false);
  392. if (ret)
  393. IPAERR("mpm failed to disable ADPL over ODL\n");
  394. }
  395. return ret;
  396. }
  397. void ipa3_odl_pipe_cleanup(bool is_ssr)
  398. {
  399. bool ipa_odl_opened = false;
  400. struct ipa_ep_cfg_holb holb_cfg;
  401. if (!ipa3_odl_ctx->odl_state.adpl_open) {
  402. IPAERR("adpl pipe not configured\n");
  403. return;
  404. }
  405. if (ipa3_odl_ctx->odl_state.odl_open)
  406. ipa_odl_opened = true;
  407. memset(&ipa3_odl_ctx->odl_state, 0, sizeof(ipa3_odl_ctx->odl_state));
  408. /*Since init will not be done again*/
  409. ipa3_odl_ctx->odl_state.odl_init = true;
  410. memset(&holb_cfg, 0, sizeof(holb_cfg));
  411. holb_cfg.tmr_val = 0;
  412. holb_cfg.en = 0;
  413. ipa3_cfg_ep_holb_by_client(IPA_CLIENT_USB_DPL_CONS, &holb_cfg);
  414. ipa3_teardown_sys_pipe(ipa3_odl_ctx->odl_client_hdl);
  415. ipa3_odl_ctx->odl_client_hdl = -1;
  416. /*Assume QTI will never close this node once opened*/
  417. if (ipa_odl_opened)
  418. ipa3_odl_ctx->odl_state.odl_open = true;
  419. /*Assume DIAG will not close this node in SSR case*/
  420. if (is_ssr)
  421. ipa3_odl_ctx->odl_state.adpl_open = true;
  422. else
  423. ipa3_odl_ctx->odl_state.adpl_open = false;
  424. ipa3_odl_ctx->odl_state.odl_disconnected = true;
  425. ipa3_odl_ctx->odl_state.odl_ep_setup = false;
  426. ipa3_odl_ctx->odl_state.aggr_byte_limit_sent = false;
  427. ipa3_odl_ctx->odl_state.odl_connected = false;
  428. /*
  429. * Send signal to ipa_odl_ctl_fops_read,
  430. * to send ODL ep close notification
  431. */
  432. ipa3_odl_ctx->odl_ctl_msg_wq_flag = true;
  433. ipa3_odl_ctx->stats.odl_drop_pkt = 0;
  434. atomic_set(&ipa3_odl_ctx->stats.numer_in_queue, 0);
  435. ipa3_odl_ctx->stats.odl_rx_pkt = 0;
  436. ipa3_odl_ctx->stats.odl_tx_diag_pkt = 0;
  437. IPADBG("Wake up odl ctl\n");
  438. wake_up_interruptible(&odl_ctl_msg_wq);
  439. }
  440. /**
  441. * ipa_adpl_read() - read message from IPA device
  442. * @filp: [in] file pointer
  443. * @buf: [out] buffer to read into
  444. * @count: [in] size of above buffer
  445. * @f_pos: [inout] file position
  446. *
  447. * User-space should continually read from /dev/ipa_adpl,
  448. * read will block when there are no messages to read.
  449. * Upon return, user-space should read
  450. * Buffer supplied must be big enough to
  451. * hold the data.
  452. *
  453. * Returns: how many bytes copied to buffer
  454. *
  455. * Note: Should not be called from atomic context
  456. */
  457. static ssize_t ipa_adpl_read(struct file *filp, char __user *buf, size_t count,
  458. loff_t *f_pos)
  459. {
  460. int ret = 0;
  461. char __user *start = buf;
  462. struct ipa3_push_msg_odl *msg;
  463. while (1) {
  464. IPADBG_LOW("Writing message to adpl pipe\n");
  465. if (!ipa3_odl_ctx->odl_state.odl_open)
  466. break;
  467. mutex_lock(&ipa3_odl_ctx->adpl_msg_lock);
  468. msg = NULL;
  469. if (!list_empty(&ipa3_odl_ctx->adpl_msg_list)) {
  470. msg = list_first_entry(&ipa3_odl_ctx->adpl_msg_list,
  471. struct ipa3_push_msg_odl, link);
  472. list_del(&msg->link);
  473. if (atomic_read(&ipa3_odl_ctx->stats.numer_in_queue))
  474. atomic_dec(&ipa3_odl_ctx->stats.numer_in_queue);
  475. }
  476. mutex_unlock(&ipa3_odl_ctx->adpl_msg_lock);
  477. if (msg != NULL) {
  478. if (msg->len > count) {
  479. IPAERR("Message length greater than count\n");
  480. kfree(msg->buff);
  481. kfree(msg);
  482. msg = NULL;
  483. ret = -EAGAIN;
  484. break;
  485. }
  486. if (msg->buff) {
  487. if (copy_to_user(buf, msg->buff,
  488. msg->len)) {
  489. ret = -EFAULT;
  490. kfree(msg->buff);
  491. kfree(msg);
  492. msg = NULL;
  493. ret = -EAGAIN;
  494. break;
  495. }
  496. buf += msg->len;
  497. count -= msg->len;
  498. kfree(msg->buff);
  499. }
  500. IPA_STATS_INC_CNT(ipa3_odl_ctx->stats.odl_tx_diag_pkt);
  501. kfree(msg);
  502. msg = NULL;
  503. } else {
  504. ret = -EAGAIN;
  505. break;
  506. }
  507. ret = -EAGAIN;
  508. if (filp->f_flags & O_NONBLOCK)
  509. break;
  510. ret = -EINTR;
  511. if (signal_pending(current))
  512. break;
  513. if (start != buf)
  514. break;
  515. }
  516. if (start != buf && ret != -EFAULT)
  517. ret = buf - start;
  518. return ret;
  519. }
  520. static long ipa_adpl_ioctl(struct file *filp,
  521. unsigned int cmd, unsigned long arg)
  522. {
  523. struct odl_agg_pipe_info odl_pipe_info;
  524. int retval = 0;
  525. if (!ipa3_odl_ctx->odl_state.odl_connected) {
  526. IPAERR("ODL config in progress not allowed ioctl\n");
  527. print_ipa_odl_state_bit_mask();
  528. retval = -ENODEV;
  529. goto fail;
  530. }
  531. IPADBG("Calling adpl ioctl\n");
  532. switch (cmd) {
  533. case IPA_IOC_ODL_GET_AGG_BYTE_LIMIT:
  534. odl_pipe_info.agg_byte_limit =
  535. ipa3_odl_ctx->odl_sys_param.ipa_ep_cfg.aggr.aggr_byte_limit;
  536. if (copy_to_user((void __user *)arg, &odl_pipe_info,
  537. sizeof(odl_pipe_info))) {
  538. retval = -EFAULT;
  539. goto fail;
  540. }
  541. ipa3_odl_ctx->odl_state.aggr_byte_limit_sent = true;
  542. break;
  543. default:
  544. retval = -ENOIOCTLCMD;
  545. print_ipa_odl_state_bit_mask();
  546. break;
  547. }
  548. fail:
  549. return retval;
  550. }
  551. static const struct file_operations ipa_odl_ctl_fops = {
  552. .owner = THIS_MODULE,
  553. .open = ipa_odl_ctl_fops_open,
  554. .release = ipa_odl_ctl_fops_release,
  555. .read = ipa_odl_ctl_fops_read,
  556. .unlocked_ioctl = ipa_odl_ctl_fops_ioctl,
  557. .poll = ipa_odl_ctl_fops_poll,
  558. };
  559. static const struct file_operations ipa_adpl_fops = {
  560. .owner = THIS_MODULE,
  561. .open = ipa_adpl_open,
  562. .release = ipa_adpl_release,
  563. .read = ipa_adpl_read,
  564. .unlocked_ioctl = ipa_adpl_ioctl,
  565. };
  566. int ipa_odl_init(void)
  567. {
  568. int result = 0;
  569. struct cdev *cdev;
  570. int loop = 0;
  571. struct ipa3_odl_char_device_context *odl_cdev;
  572. ipa3_odl_ctx = kzalloc(sizeof(*ipa3_odl_ctx), GFP_KERNEL);
  573. if (!ipa3_odl_ctx) {
  574. result = -ENOMEM;
  575. goto fail_mem_ctx;
  576. }
  577. odl_cdev = ipa3_odl_ctx->odl_cdev;
  578. INIT_LIST_HEAD(&ipa3_odl_ctx->adpl_msg_list);
  579. mutex_init(&ipa3_odl_ctx->adpl_msg_lock);
  580. odl_cdev[loop].class = class_create(THIS_MODULE, "ipa_adpl");
  581. if (IS_ERR(odl_cdev[loop].class)) {
  582. IPAERR("Error: odl_cdev->class NULL\n");
  583. result = -ENODEV;
  584. goto create_char_dev0_fail;
  585. }
  586. result = alloc_chrdev_region(&odl_cdev[loop].dev_num, 0, 1, "ipa_adpl");
  587. if (result) {
  588. IPAERR("alloc_chrdev_region error for ipa adpl pipe\n");
  589. result = -ENODEV;
  590. goto alloc_chrdev0_region_fail;
  591. }
  592. odl_cdev[loop].dev = device_create(odl_cdev[loop].class, NULL,
  593. odl_cdev[loop].dev_num, ipa3_ctx, "ipa_adpl");
  594. if (IS_ERR(odl_cdev[loop].dev)) {
  595. IPAERR("device_create err:%ld\n", PTR_ERR(odl_cdev[loop].dev));
  596. result = PTR_ERR(odl_cdev[loop].dev);
  597. goto device0_create_fail;
  598. }
  599. cdev = &odl_cdev[loop].cdev;
  600. cdev_init(cdev, &ipa_adpl_fops);
  601. cdev->owner = THIS_MODULE;
  602. cdev->ops = &ipa_adpl_fops;
  603. result = cdev_add(cdev, odl_cdev[loop].dev_num, 1);
  604. if (result) {
  605. IPAERR("cdev_add err=%d\n", -result);
  606. goto cdev0_add_fail;
  607. }
  608. loop++;
  609. odl_cdev[loop].class = class_create(THIS_MODULE, "ipa_odl_ctl");
  610. if (IS_ERR(odl_cdev[loop].class)) {
  611. IPAERR("Error: odl_cdev->class NULL\n");
  612. result = -ENODEV;
  613. goto create_char_dev1_fail;
  614. }
  615. result = alloc_chrdev_region(&odl_cdev[loop].dev_num, 0, 1,
  616. "ipa_odl_ctl");
  617. if (result) {
  618. IPAERR("alloc_chrdev_region error for ipa odl ctl pipe\n");
  619. goto alloc_chrdev1_region_fail;
  620. }
  621. odl_cdev[loop].dev = device_create(odl_cdev[loop].class, NULL,
  622. odl_cdev[loop].dev_num, ipa3_ctx, "ipa_odl_ctl");
  623. if (IS_ERR(odl_cdev[loop].dev)) {
  624. IPAERR("device_create err:%ld\n", PTR_ERR(odl_cdev[loop].dev));
  625. result = PTR_ERR(odl_cdev[loop].dev);
  626. goto device1_create_fail;
  627. }
  628. cdev = &odl_cdev[loop].cdev;
  629. cdev_init(cdev, &ipa_odl_ctl_fops);
  630. cdev->owner = THIS_MODULE;
  631. cdev->ops = &ipa_odl_ctl_fops;
  632. result = cdev_add(cdev, odl_cdev[loop].dev_num, 1);
  633. if (result) {
  634. IPAERR(":cdev_add err=%d\n", -result);
  635. goto cdev1_add_fail;
  636. }
  637. ipa3_odl_ctx->odl_state.odl_init = true;
  638. /* register ipa_pm */
  639. result = ipa3_odl_register_pm();
  640. if (result) {
  641. IPAWANERR("ipa3_odl_register_pm failed, ret: %d\n",
  642. result);
  643. }
  644. return 0;
  645. cdev1_add_fail:
  646. device_destroy(odl_cdev[1].class, odl_cdev[1].dev_num);
  647. device1_create_fail:
  648. unregister_chrdev_region(odl_cdev[1].dev_num, 1);
  649. alloc_chrdev1_region_fail:
  650. class_destroy(odl_cdev[1].class);
  651. create_char_dev1_fail:
  652. cdev0_add_fail:
  653. device_destroy(odl_cdev[0].class, odl_cdev[0].dev_num);
  654. device0_create_fail:
  655. unregister_chrdev_region(odl_cdev[0].dev_num, 1);
  656. alloc_chrdev0_region_fail:
  657. class_destroy(odl_cdev[0].class);
  658. create_char_dev0_fail:
  659. kfree(ipa3_odl_ctx);
  660. fail_mem_ctx:
  661. return result;
  662. }
  663. bool ipa3_is_odl_connected(void)
  664. {
  665. return ipa3_odl_ctx->odl_state.odl_connected;
  666. }