ipa_odl.c 18 KB

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