teth_bridge.c 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2013-2019, The Linux Foundation. All rights reserved.
  4. */
  5. #include <linux/completion.h>
  6. #include <linux/debugfs.h>
  7. #include <linux/export.h>
  8. #include <linux/fs.h>
  9. #include <linux/if_ether.h>
  10. #include <linux/ioctl.h>
  11. #include <linux/kernel.h>
  12. #include <linux/module.h>
  13. #include <linux/msm_ipa.h>
  14. #include <linux/mutex.h>
  15. #include <linux/skbuff.h>
  16. #include <linux/types.h>
  17. #include <linux/ipa.h>
  18. #include <linux/netdevice.h>
  19. #include "ipa_i.h"
  20. #define TETH_BRIDGE_DRV_NAME "ipa_tethering_bridge"
  21. #define TETH_DBG(fmt, args...) \
  22. pr_debug(TETH_BRIDGE_DRV_NAME " %s:%d " fmt, \
  23. __func__, __LINE__, ## args)
  24. #define TETH_DBG_FUNC_ENTRY() \
  25. pr_debug(TETH_BRIDGE_DRV_NAME " %s:%d ENTRY\n", __func__, __LINE__)
  26. #define TETH_DBG_FUNC_EXIT() \
  27. pr_debug(TETH_BRIDGE_DRV_NAME " %s:%d EXIT\n", __func__, __LINE__)
  28. #define TETH_ERR(fmt, args...) \
  29. pr_err(TETH_BRIDGE_DRV_NAME " %s:%d " fmt, __func__, __LINE__, ## args)
  30. /**
  31. * struct ipa3_teth_bridge_ctx - Tethering bridge driver context information
  32. * @class: kernel class pointer
  33. * @dev_num: kernel device number
  34. * @dev: kernel device struct pointer
  35. * @cdev: kernel character device struct
  36. */
  37. struct ipa3_teth_bridge_ctx {
  38. struct class *class;
  39. dev_t dev_num;
  40. struct device *dev;
  41. struct cdev cdev;
  42. u32 modem_pm_hdl;
  43. };
  44. static struct ipa3_teth_bridge_ctx *ipa3_teth_ctx;
  45. /**
  46. * teth_bridge_ipa_cb() - Callback to handle IPA data path events
  47. * @priv - private data
  48. * @evt - event type
  49. * @data - event specific data (usually skb)
  50. *
  51. * This callback is called by IPA driver for exception packets from USB.
  52. * All exception packets are handled by Q6 and should not reach this function.
  53. * Packets will arrive to AP exception pipe only in case where packets are
  54. * sent from USB before Q6 has setup the call.
  55. */
  56. static void teth_bridge_ipa_cb(void *priv, enum ipa_dp_evt_type evt,
  57. unsigned long data)
  58. {
  59. struct sk_buff *skb = (struct sk_buff *)data;
  60. TETH_DBG_FUNC_ENTRY();
  61. if (evt != IPA_RECEIVE) {
  62. TETH_ERR("unexpected event %d\n", evt);
  63. WARN_ON(1);
  64. return;
  65. }
  66. TETH_ERR("Unexpected exception packet from USB, dropping packet\n");
  67. dev_kfree_skb_any(skb);
  68. TETH_DBG_FUNC_EXIT();
  69. }
  70. /**
  71. * ipa3_teth_bridge_init() - Initialize the Tethering bridge driver
  72. * @params - in/out params for USB initialization API (please look at struct
  73. * definition for more info)
  74. *
  75. * USB driver gets a pointer to a callback function (usb_notify_cb) and an
  76. * associated data.
  77. *
  78. * Builds IPA resource manager dependency graph.
  79. *
  80. * Return codes: 0: success,
  81. * -EINVAL - Bad parameter
  82. * Other negative value - Failure
  83. */
  84. int ipa3_teth_bridge_init(struct teth_bridge_init_params *params)
  85. {
  86. TETH_DBG_FUNC_ENTRY();
  87. if (!params) {
  88. TETH_ERR("Bad parameter\n");
  89. TETH_DBG_FUNC_EXIT();
  90. return -EINVAL;
  91. }
  92. params->usb_notify_cb = teth_bridge_ipa_cb;
  93. params->private_data = NULL;
  94. params->skip_ep_cfg = true;
  95. TETH_DBG_FUNC_EXIT();
  96. return 0;
  97. }
  98. /**
  99. * ipa3_teth_bridge_get_pm_hdl() - Get the Tethering bridge Driver pm hdl
  100. *
  101. *
  102. * Return codes: handle
  103. * -EINVAL - Bad parameter
  104. */
  105. int ipa3_teth_bridge_get_pm_hdl(void)
  106. {
  107. TETH_DBG_FUNC_ENTRY();
  108. if (ipa3_teth_ctx->modem_pm_hdl == ~0) {
  109. TETH_ERR("Bad parameter\n");
  110. TETH_DBG_FUNC_EXIT();
  111. return -EINVAL;
  112. }
  113. TETH_DBG("Return pm-handle %d\n", ipa3_teth_ctx->modem_pm_hdl);
  114. TETH_DBG_FUNC_EXIT();
  115. return ipa3_teth_ctx->modem_pm_hdl;
  116. }
  117. /**
  118. * ipa3_teth_bridge_disconnect() - Disconnect tethering bridge module
  119. */
  120. int ipa3_teth_bridge_disconnect(enum ipa_client_type client)
  121. {
  122. int res = 0;
  123. TETH_DBG_FUNC_ENTRY();
  124. res = ipa_pm_deactivate_sync(ipa3_teth_ctx->modem_pm_hdl);
  125. if (res) {
  126. TETH_ERR("fail to deactivate modem %d\n", res);
  127. return res;
  128. }
  129. res = ipa_pm_deregister(ipa3_teth_ctx->modem_pm_hdl);
  130. ipa3_teth_ctx->modem_pm_hdl = ~0;
  131. TETH_DBG_FUNC_EXIT();
  132. return res;
  133. }
  134. /**
  135. * ipa3_teth_bridge_connect() - Connect bridge for a tethered Rmnet / MBIM call
  136. * @connect_params: Connection info
  137. *
  138. * Return codes: 0: success
  139. * -EINVAL: invalid parameters
  140. * -EPERM: Operation not permitted as the bridge is already
  141. * connected
  142. */
  143. int ipa3_teth_bridge_connect(struct teth_bridge_connect_params *connect_params)
  144. {
  145. int res = 0;
  146. struct ipa_pm_register_params reg_params;
  147. memset(&reg_params, 0, sizeof(reg_params));
  148. TETH_DBG_FUNC_ENTRY();
  149. reg_params.name = "MODEM (USB RMNET)";
  150. reg_params.group = IPA_PM_GROUP_MODEM;
  151. reg_params.skip_clk_vote = true;
  152. res = ipa_pm_register(&reg_params,
  153. &ipa3_teth_ctx->modem_pm_hdl);
  154. if (res) {
  155. TETH_ERR("fail to register with PM %d\n", res);
  156. return res;
  157. }
  158. /* vote for turbo */
  159. res = ipa_pm_set_throughput(ipa3_teth_ctx->modem_pm_hdl,
  160. 5200);
  161. res = ipa_pm_activate_sync(ipa3_teth_ctx->modem_pm_hdl);
  162. TETH_DBG_FUNC_EXIT();
  163. return res;
  164. }
  165. static long ipa3_teth_bridge_ioctl(struct file *filp,
  166. unsigned int cmd,
  167. unsigned long arg)
  168. {
  169. IPAERR("No ioctls are supported!\n");
  170. return -ENOIOCTLCMD;
  171. }
  172. static const struct file_operations ipa3_teth_bridge_drv_fops = {
  173. .owner = THIS_MODULE,
  174. .unlocked_ioctl = ipa3_teth_bridge_ioctl,
  175. };
  176. /**
  177. * ipa3_teth_bridge_driver_init() - Initialize tethering bridge driver
  178. *
  179. */
  180. int ipa3_teth_bridge_driver_init(void)
  181. {
  182. int res;
  183. TETH_DBG("Tethering bridge driver init\n");
  184. ipa3_teth_ctx = kzalloc(sizeof(*ipa3_teth_ctx), GFP_KERNEL);
  185. if (!ipa3_teth_ctx)
  186. return -ENOMEM;
  187. ipa3_teth_ctx->class = class_create(THIS_MODULE, TETH_BRIDGE_DRV_NAME);
  188. res = alloc_chrdev_region(&ipa3_teth_ctx->dev_num, 0, 1,
  189. TETH_BRIDGE_DRV_NAME);
  190. if (res) {
  191. TETH_ERR("alloc_chrdev_region err.\n");
  192. res = -ENODEV;
  193. goto fail_alloc_chrdev_region;
  194. }
  195. ipa3_teth_ctx->dev = device_create(ipa3_teth_ctx->class,
  196. NULL,
  197. ipa3_teth_ctx->dev_num,
  198. ipa3_teth_ctx,
  199. TETH_BRIDGE_DRV_NAME);
  200. if (IS_ERR(ipa3_teth_ctx->dev)) {
  201. TETH_ERR(":device_create err.\n");
  202. res = -ENODEV;
  203. goto fail_device_create;
  204. }
  205. cdev_init(&ipa3_teth_ctx->cdev, &ipa3_teth_bridge_drv_fops);
  206. ipa3_teth_ctx->cdev.owner = THIS_MODULE;
  207. ipa3_teth_ctx->cdev.ops = &ipa3_teth_bridge_drv_fops;
  208. res = cdev_add(&ipa3_teth_ctx->cdev, ipa3_teth_ctx->dev_num, 1);
  209. if (res) {
  210. TETH_ERR(":cdev_add err=%d\n", -res);
  211. res = -ENODEV;
  212. goto fail_cdev_add;
  213. }
  214. ipa3_teth_ctx->modem_pm_hdl = ~0;
  215. TETH_DBG("Tethering bridge driver init OK\n");
  216. return 0;
  217. fail_cdev_add:
  218. device_destroy(ipa3_teth_ctx->class, ipa3_teth_ctx->dev_num);
  219. fail_device_create:
  220. unregister_chrdev_region(ipa3_teth_ctx->dev_num, 1);
  221. fail_alloc_chrdev_region:
  222. kfree(ipa3_teth_ctx);
  223. ipa3_teth_ctx = NULL;
  224. return res;
  225. }
  226. MODULE_LICENSE("GPL v2");
  227. MODULE_DESCRIPTION("Tethering bridge driver");