ipa_odl.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729
  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. ret = ipa3_setup_sys_pipe(ipa_odl_ep_cfg,
  274. &ipa3_odl_ctx->odl_client_hdl);
  275. return ret;
  276. }
  277. /**
  278. * ipa3_odl_register_pm - Register odl client for PM
  279. *
  280. * This function will register 1 client with IPA PM to represent odl
  281. * in clock scaling calculation:
  282. * - "ODL" - this client will be activated when pipe connected
  283. */
  284. static int ipa3_odl_register_pm(void)
  285. {
  286. int result = 0;
  287. struct ipa_pm_register_params pm_reg;
  288. memset(&pm_reg, 0, sizeof(pm_reg));
  289. pm_reg.name = "ODL";
  290. pm_reg.group = IPA_PM_GROUP_DEFAULT;
  291. pm_reg.skip_clk_vote = true;
  292. result = ipa_pm_register(&pm_reg, &ipa3_odl_ctx->odl_pm_hdl);
  293. if (result) {
  294. IPAERR("failed to create IPA PM client %d\n", result);
  295. return result;
  296. }
  297. return result;
  298. }
  299. int ipa3_odl_pipe_open(void)
  300. {
  301. int ret = 0;
  302. struct ipa_ep_cfg_holb holb_cfg;
  303. if (!ipa3_odl_ctx->odl_state.adpl_open) {
  304. IPAERR("adpl pipe not configured\n");
  305. return 0;
  306. }
  307. memset(&holb_cfg, 0, sizeof(holb_cfg));
  308. holb_cfg.tmr_val = 0;
  309. holb_cfg.en = 1;
  310. ipa3_cfg_ep_holb_by_client(IPA_CLIENT_USB_DPL_CONS, &holb_cfg);
  311. ret = ipa_setup_odl_pipe();
  312. if (ret) {
  313. IPAERR(" Setup endpoint config failed\n");
  314. ipa3_odl_ctx->odl_state.adpl_open = false;
  315. goto fail;
  316. }
  317. ipa3_cfg_ep_holb_by_client(IPA_CLIENT_ODL_DPL_CONS, &holb_cfg);
  318. ipa3_odl_ctx->odl_state.odl_ep_setup = true;
  319. IPADBG("Setup endpoint config success\n");
  320. ipa3_odl_ctx->stats.odl_drop_pkt = 0;
  321. atomic_set(&ipa3_odl_ctx->stats.numer_in_queue, 0);
  322. ipa3_odl_ctx->stats.odl_rx_pkt = 0;
  323. ipa3_odl_ctx->stats.odl_tx_diag_pkt = 0;
  324. /*
  325. * Send signal to ipa_odl_ctl_fops_read,
  326. * to send ODL ep open notification
  327. */
  328. ipa3_odl_ctx->odl_ctl_msg_wq_flag = true;
  329. IPADBG("Wake up odl ctl\n");
  330. wake_up_interruptible(&odl_ctl_msg_wq);
  331. if (ipa3_odl_ctx->odl_state.odl_disconnected)
  332. ipa3_odl_ctx->odl_state.odl_disconnected = false;
  333. fail:
  334. return ret;
  335. }
  336. static int ipa_adpl_open(struct inode *inode, struct file *filp)
  337. {
  338. int ret = 0;
  339. IPADBG("Called the function :\n");
  340. if (ipa3_odl_ctx->odl_state.odl_init &&
  341. !ipa3_odl_ctx->odl_state.adpl_open) {
  342. /* Activate ipa_pm*/
  343. ret = ipa_pm_activate_sync(ipa3_odl_ctx->odl_pm_hdl);
  344. if (ret)
  345. IPAERR("failed to activate pm\n");
  346. ipa3_odl_ctx->odl_state.adpl_open = true;
  347. ret = ipa3_odl_pipe_open();
  348. } else {
  349. IPAERR("Before odl init trying to open adpl pipe\n");
  350. print_ipa_odl_state_bit_mask();
  351. ret = -ENODEV;
  352. }
  353. return ret;
  354. }
  355. static int ipa_adpl_release(struct inode *inode, struct file *filp)
  356. {
  357. int ret = 0;
  358. /* Deactivate ipa_pm */
  359. ret = ipa_pm_deactivate_sync(ipa3_odl_ctx->odl_pm_hdl);
  360. if (ret)
  361. IPAERR("failed to activate pm\n");
  362. ipa3_odl_pipe_cleanup(false);
  363. return ret;
  364. }
  365. void ipa3_odl_pipe_cleanup(bool is_ssr)
  366. {
  367. bool ipa_odl_opened = false;
  368. struct ipa_ep_cfg_holb holb_cfg;
  369. if (!ipa3_odl_ctx->odl_state.adpl_open) {
  370. IPAERR("adpl pipe not configured\n");
  371. return;
  372. }
  373. if (ipa3_odl_ctx->odl_state.odl_open)
  374. ipa_odl_opened = true;
  375. memset(&ipa3_odl_ctx->odl_state, 0, sizeof(ipa3_odl_ctx->odl_state));
  376. /*Since init will not be done again*/
  377. ipa3_odl_ctx->odl_state.odl_init = true;
  378. memset(&holb_cfg, 0, sizeof(holb_cfg));
  379. holb_cfg.tmr_val = 0;
  380. holb_cfg.en = 0;
  381. ipa3_cfg_ep_holb_by_client(IPA_CLIENT_USB_DPL_CONS, &holb_cfg);
  382. ipa3_teardown_sys_pipe(ipa3_odl_ctx->odl_client_hdl);
  383. ipa3_odl_ctx->odl_client_hdl = -1;
  384. /*Assume QTI will never close this node once opened*/
  385. if (ipa_odl_opened)
  386. ipa3_odl_ctx->odl_state.odl_open = true;
  387. /*Assume DIAG will not close this node in SSR case*/
  388. if (is_ssr)
  389. ipa3_odl_ctx->odl_state.adpl_open = true;
  390. else
  391. ipa3_odl_ctx->odl_state.adpl_open = false;
  392. ipa3_odl_ctx->odl_state.odl_disconnected = true;
  393. ipa3_odl_ctx->odl_state.odl_ep_setup = false;
  394. ipa3_odl_ctx->odl_state.aggr_byte_limit_sent = false;
  395. ipa3_odl_ctx->odl_state.odl_connected = false;
  396. /*
  397. * Send signal to ipa_odl_ctl_fops_read,
  398. * to send ODL ep close notification
  399. */
  400. ipa3_odl_ctx->odl_ctl_msg_wq_flag = true;
  401. ipa3_odl_ctx->stats.odl_drop_pkt = 0;
  402. atomic_set(&ipa3_odl_ctx->stats.numer_in_queue, 0);
  403. ipa3_odl_ctx->stats.odl_rx_pkt = 0;
  404. ipa3_odl_ctx->stats.odl_tx_diag_pkt = 0;
  405. IPADBG("Wake up odl ctl\n");
  406. wake_up_interruptible(&odl_ctl_msg_wq);
  407. }
  408. /**
  409. * ipa_adpl_read() - read message from IPA device
  410. * @filp: [in] file pointer
  411. * @buf: [out] buffer to read into
  412. * @count: [in] size of above buffer
  413. * @f_pos: [inout] file position
  414. *
  415. * User-space should continually read from /dev/ipa_adpl,
  416. * read will block when there are no messages to read.
  417. * Upon return, user-space should read
  418. * Buffer supplied must be big enough to
  419. * hold the data.
  420. *
  421. * Returns: how many bytes copied to buffer
  422. *
  423. * Note: Should not be called from atomic context
  424. */
  425. static ssize_t ipa_adpl_read(struct file *filp, char __user *buf, size_t count,
  426. loff_t *f_pos)
  427. {
  428. int ret = 0;
  429. char __user *start = buf;
  430. struct ipa3_push_msg_odl *msg;
  431. while (1) {
  432. IPADBG_LOW("Writing message to adpl pipe\n");
  433. if (!ipa3_odl_ctx->odl_state.odl_open)
  434. break;
  435. mutex_lock(&ipa3_odl_ctx->adpl_msg_lock);
  436. msg = NULL;
  437. if (!list_empty(&ipa3_odl_ctx->adpl_msg_list)) {
  438. msg = list_first_entry(&ipa3_odl_ctx->adpl_msg_list,
  439. struct ipa3_push_msg_odl, link);
  440. list_del(&msg->link);
  441. if (atomic_read(&ipa3_odl_ctx->stats.numer_in_queue))
  442. atomic_dec(&ipa3_odl_ctx->stats.numer_in_queue);
  443. }
  444. mutex_unlock(&ipa3_odl_ctx->adpl_msg_lock);
  445. if (msg != NULL) {
  446. if (msg->len > count) {
  447. IPAERR("Message length greater than count\n");
  448. kfree(msg->buff);
  449. kfree(msg);
  450. msg = NULL;
  451. ret = -EAGAIN;
  452. break;
  453. }
  454. if (msg->buff) {
  455. if (copy_to_user(buf, msg->buff,
  456. msg->len)) {
  457. ret = -EFAULT;
  458. kfree(msg->buff);
  459. kfree(msg);
  460. msg = NULL;
  461. ret = -EAGAIN;
  462. break;
  463. }
  464. buf += msg->len;
  465. count -= msg->len;
  466. kfree(msg->buff);
  467. }
  468. IPA_STATS_INC_CNT(ipa3_odl_ctx->stats.odl_tx_diag_pkt);
  469. kfree(msg);
  470. msg = NULL;
  471. } else {
  472. ret = -EAGAIN;
  473. break;
  474. }
  475. ret = -EAGAIN;
  476. if (filp->f_flags & O_NONBLOCK)
  477. break;
  478. ret = -EINTR;
  479. if (signal_pending(current))
  480. break;
  481. if (start != buf)
  482. break;
  483. }
  484. if (start != buf && ret != -EFAULT)
  485. ret = buf - start;
  486. return ret;
  487. }
  488. static long ipa_adpl_ioctl(struct file *filp,
  489. unsigned int cmd, unsigned long arg)
  490. {
  491. struct odl_agg_pipe_info odl_pipe_info;
  492. int retval = 0;
  493. if (!ipa3_odl_ctx->odl_state.odl_connected) {
  494. IPAERR("ODL config in progress not allowed ioctl\n");
  495. print_ipa_odl_state_bit_mask();
  496. retval = -ENODEV;
  497. goto fail;
  498. }
  499. IPADBG("Calling adpl ioctl\n");
  500. switch (cmd) {
  501. case IPA_IOC_ODL_GET_AGG_BYTE_LIMIT:
  502. odl_pipe_info.agg_byte_limit =
  503. ipa3_odl_ctx->odl_sys_param.ipa_ep_cfg.aggr.aggr_byte_limit;
  504. if (copy_to_user((void __user *)arg, &odl_pipe_info,
  505. sizeof(odl_pipe_info))) {
  506. retval = -EFAULT;
  507. goto fail;
  508. }
  509. ipa3_odl_ctx->odl_state.aggr_byte_limit_sent = true;
  510. break;
  511. default:
  512. retval = -ENOIOCTLCMD;
  513. print_ipa_odl_state_bit_mask();
  514. break;
  515. }
  516. fail:
  517. return retval;
  518. }
  519. static const struct file_operations ipa_odl_ctl_fops = {
  520. .owner = THIS_MODULE,
  521. .open = ipa_odl_ctl_fops_open,
  522. .release = ipa_odl_ctl_fops_release,
  523. .read = ipa_odl_ctl_fops_read,
  524. .unlocked_ioctl = ipa_odl_ctl_fops_ioctl,
  525. .poll = ipa_odl_ctl_fops_poll,
  526. };
  527. static const struct file_operations ipa_adpl_fops = {
  528. .owner = THIS_MODULE,
  529. .open = ipa_adpl_open,
  530. .release = ipa_adpl_release,
  531. .read = ipa_adpl_read,
  532. .unlocked_ioctl = ipa_adpl_ioctl,
  533. };
  534. int ipa_odl_init(void)
  535. {
  536. int result = 0;
  537. struct cdev *cdev;
  538. int loop = 0;
  539. struct ipa3_odl_char_device_context *odl_cdev;
  540. ipa3_odl_ctx = kzalloc(sizeof(*ipa3_odl_ctx), GFP_KERNEL);
  541. if (!ipa3_odl_ctx) {
  542. result = -ENOMEM;
  543. goto fail_mem_ctx;
  544. }
  545. odl_cdev = ipa3_odl_ctx->odl_cdev;
  546. INIT_LIST_HEAD(&ipa3_odl_ctx->adpl_msg_list);
  547. mutex_init(&ipa3_odl_ctx->adpl_msg_lock);
  548. odl_cdev[loop].class = class_create(THIS_MODULE, "ipa_adpl");
  549. if (IS_ERR(odl_cdev[loop].class)) {
  550. IPAERR("Error: odl_cdev->class NULL\n");
  551. result = -ENODEV;
  552. goto create_char_dev0_fail;
  553. }
  554. result = alloc_chrdev_region(&odl_cdev[loop].dev_num, 0, 1, "ipa_adpl");
  555. if (result) {
  556. IPAERR("alloc_chrdev_region error for ipa adpl pipe\n");
  557. result = -ENODEV;
  558. goto alloc_chrdev0_region_fail;
  559. }
  560. odl_cdev[loop].dev = device_create(odl_cdev[loop].class, NULL,
  561. odl_cdev[loop].dev_num, ipa3_ctx, "ipa_adpl");
  562. if (IS_ERR(odl_cdev[loop].dev)) {
  563. IPAERR("device_create err:%ld\n", PTR_ERR(odl_cdev[loop].dev));
  564. result = PTR_ERR(odl_cdev[loop].dev);
  565. goto device0_create_fail;
  566. }
  567. cdev = &odl_cdev[loop].cdev;
  568. cdev_init(cdev, &ipa_adpl_fops);
  569. cdev->owner = THIS_MODULE;
  570. cdev->ops = &ipa_adpl_fops;
  571. result = cdev_add(cdev, odl_cdev[loop].dev_num, 1);
  572. if (result) {
  573. IPAERR("cdev_add err=%d\n", -result);
  574. goto cdev0_add_fail;
  575. }
  576. loop++;
  577. odl_cdev[loop].class = class_create(THIS_MODULE, "ipa_odl_ctl");
  578. if (IS_ERR(odl_cdev[loop].class)) {
  579. IPAERR("Error: odl_cdev->class NULL\n");
  580. result = -ENODEV;
  581. goto create_char_dev1_fail;
  582. }
  583. result = alloc_chrdev_region(&odl_cdev[loop].dev_num, 0, 1,
  584. "ipa_odl_ctl");
  585. if (result) {
  586. IPAERR("alloc_chrdev_region error for ipa odl ctl pipe\n");
  587. goto alloc_chrdev1_region_fail;
  588. }
  589. odl_cdev[loop].dev = device_create(odl_cdev[loop].class, NULL,
  590. odl_cdev[loop].dev_num, ipa3_ctx, "ipa_odl_ctl");
  591. if (IS_ERR(odl_cdev[loop].dev)) {
  592. IPAERR("device_create err:%ld\n", PTR_ERR(odl_cdev[loop].dev));
  593. result = PTR_ERR(odl_cdev[loop].dev);
  594. goto device1_create_fail;
  595. }
  596. cdev = &odl_cdev[loop].cdev;
  597. cdev_init(cdev, &ipa_odl_ctl_fops);
  598. cdev->owner = THIS_MODULE;
  599. cdev->ops = &ipa_odl_ctl_fops;
  600. result = cdev_add(cdev, odl_cdev[loop].dev_num, 1);
  601. if (result) {
  602. IPAERR(":cdev_add err=%d\n", -result);
  603. goto cdev1_add_fail;
  604. }
  605. ipa3_odl_ctx->odl_state.odl_init = true;
  606. /* register ipa_pm */
  607. result = ipa3_odl_register_pm();
  608. if (result) {
  609. IPAWANERR("ipa3_odl_register_pm failed, ret: %d\n",
  610. result);
  611. }
  612. return 0;
  613. cdev1_add_fail:
  614. device_destroy(odl_cdev[1].class, odl_cdev[1].dev_num);
  615. device1_create_fail:
  616. unregister_chrdev_region(odl_cdev[1].dev_num, 1);
  617. alloc_chrdev1_region_fail:
  618. class_destroy(odl_cdev[1].class);
  619. create_char_dev1_fail:
  620. cdev0_add_fail:
  621. device_destroy(odl_cdev[0].class, odl_cdev[0].dev_num);
  622. device0_create_fail:
  623. unregister_chrdev_region(odl_cdev[0].dev_num, 1);
  624. alloc_chrdev0_region_fail:
  625. class_destroy(odl_cdev[0].class);
  626. create_char_dev0_fail:
  627. kfree(ipa3_odl_ctx);
  628. fail_mem_ctx:
  629. return result;
  630. }