odu_bridge.c 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2014-2019, The Linux Foundation. All rights reserved.
  4. */
  5. #include <linux/debugfs.h>
  6. #include <linux/export.h>
  7. #include <linux/fs.h>
  8. #include <linux/if_ether.h>
  9. #include <linux/ioctl.h>
  10. #include <linux/kernel.h>
  11. #include <linux/module.h>
  12. #include <linux/msm_ipa.h>
  13. #include <linux/mutex.h>
  14. #include <linux/skbuff.h>
  15. #include <linux/types.h>
  16. #include <linux/ipv6.h>
  17. #include <net/addrconf.h>
  18. #include <linux/ipa.h>
  19. #include <linux/cdev.h>
  20. #include <linux/ipa_odu_bridge.h>
  21. #include "../ipa_common_i.h"
  22. #include "../ipa_v3/ipa_pm.h"
  23. #define ODU_BRIDGE_DRV_NAME "odu_ipa_bridge"
  24. #define ODU_BRIDGE_DBG(fmt, args...) \
  25. do { \
  26. pr_debug(ODU_BRIDGE_DRV_NAME " %s:%d " fmt, \
  27. __func__, __LINE__, ## args); \
  28. IPA_IPC_LOGGING(ipa_get_ipc_logbuf(), \
  29. ODU_BRIDGE_DRV_NAME " %s:%d " fmt, ## args); \
  30. IPA_IPC_LOGGING(ipa_get_ipc_logbuf_low(), \
  31. ODU_BRIDGE_DRV_NAME " %s:%d " fmt, ## args); \
  32. } while (0)
  33. #define ODU_BRIDGE_DBG_LOW(fmt, args...) \
  34. do { \
  35. pr_debug(ODU_BRIDGE_DRV_NAME " %s:%d " fmt, \
  36. __func__, __LINE__, ## args); \
  37. IPA_IPC_LOGGING(ipa_get_ipc_logbuf_low(), \
  38. ODU_BRIDGE_DRV_NAME " %s:%d " fmt, ## args); \
  39. } while (0)
  40. #define ODU_BRIDGE_ERR(fmt, args...) \
  41. do { \
  42. pr_err(ODU_BRIDGE_DRV_NAME " %s:%d " fmt, \
  43. __func__, __LINE__, ## args); \
  44. IPA_IPC_LOGGING(ipa_get_ipc_logbuf(), \
  45. ODU_BRIDGE_DRV_NAME " %s:%d " fmt, ## args); \
  46. IPA_IPC_LOGGING(ipa_get_ipc_logbuf_low(), \
  47. ODU_BRIDGE_DRV_NAME " %s:%d " fmt, ## args); \
  48. } while (0)
  49. #define ODU_BRIDGE_FUNC_ENTRY() \
  50. ODU_BRIDGE_DBG_LOW("ENTRY\n")
  51. #define ODU_BRIDGE_FUNC_EXIT() \
  52. ODU_BRIDGE_DBG_LOW("EXIT\n")
  53. #define ODU_BRIDGE_IS_QMI_ADDR(daddr) \
  54. (memcmp(&(daddr), &odu_bridge_ctx->llv6_addr, sizeof((daddr))) \
  55. == 0)
  56. #define ODU_BRIDGE_IPV4_HDR_NAME "odu_br_ipv4"
  57. #define ODU_BRIDGE_IPV6_HDR_NAME "odu_br_ipv6"
  58. #define IPA_ODU_SYS_DESC_FIFO_SZ 0x800
  59. #ifdef CONFIG_COMPAT
  60. #define ODU_BRIDGE_IOC_SET_LLV6_ADDR32 _IOW(ODU_BRIDGE_IOC_MAGIC, \
  61. ODU_BRIDGE_IOCTL_SET_LLV6_ADDR, \
  62. compat_uptr_t)
  63. #endif
  64. #define IPA_ODU_VER_CHECK() \
  65. do { \
  66. ret = 0;\
  67. if (ipa_get_hw_type() == IPA_HW_None) { \
  68. pr_err("IPA HW is unknown\n"); \
  69. ret = -EFAULT; \
  70. } \
  71. else if (ipa_get_hw_type() < IPA_HW_v3_0) \
  72. ret = 1; \
  73. } while (0)
  74. /**
  75. * struct stats - driver statistics, viewable using debugfs
  76. * @num_ul_packets: number of packets bridged in uplink direction
  77. * @num_dl_packets: number of packets bridged in downink direction
  78. * bridge
  79. * @num_lan_packets: number of packets bridged to APPS on bridge mode
  80. */
  81. struct stats {
  82. u64 num_ul_packets;
  83. u64 num_dl_packets;
  84. u64 num_lan_packets;
  85. };
  86. /**
  87. * struct odu_bridge_ctx - ODU bridge driver context information
  88. * @class: kernel class pointer
  89. * @dev_num: kernel device number
  90. * @dev: kernel device struct pointer
  91. * @cdev: kernel character device struct
  92. * @netdev_name: network interface name
  93. * @device_ethaddr: network interface ethernet address
  94. * @priv: client's private data. to be used in client's callbacks
  95. * @tx_dp_notify: client callback for handling IPA ODU_PROD callback
  96. * @send_dl_skb: client callback for sending skb in downlink direction
  97. * @stats: statistics, how many packets were transmitted using the SW bridge
  98. * @is_conencted: is bridge connected ?
  99. * @is_suspended: is bridge suspended ?
  100. * @mode: ODU mode (router/bridge)
  101. * @lock: for the initialization, connect and disconnect synchronization
  102. * @llv6_addr: link local IPv6 address of ODU network interface
  103. * @odu_br_ipv4_hdr_hdl: handle for partial ipv4 ethernet header
  104. * @odu_br_ipv6_hdr_hdl: handle for partial ipv6 ethernet header
  105. * @odu_prod_hdl: handle for IPA_CLIENT_ODU_PROD pipe
  106. * @odu_emb_cons_hdl: handle for IPA_CLIENT_ODU_EMB_CONS pipe
  107. * @odu_teth_cons_hdl: handle for IPA_CLIENT_ODU_TETH_CONS pipe
  108. * @wakeup_request: client callback to wakeup
  109. */
  110. struct odu_bridge_ctx {
  111. struct class *class;
  112. dev_t dev_num;
  113. struct device *dev;
  114. struct cdev cdev;
  115. char netdev_name[IPA_RESOURCE_NAME_MAX];
  116. u8 device_ethaddr[ETH_ALEN];
  117. void *priv;
  118. ipa_notify_cb tx_dp_notify;
  119. int (*send_dl_skb)(void *priv, struct sk_buff *skb);
  120. struct stats stats;
  121. bool is_connected;
  122. bool is_suspended;
  123. enum odu_bridge_mode mode;
  124. struct mutex lock;
  125. struct in6_addr llv6_addr;
  126. uint32_t odu_br_ipv4_hdr_hdl;
  127. uint32_t odu_br_ipv6_hdr_hdl;
  128. u32 odu_prod_hdl;
  129. u32 odu_emb_cons_hdl;
  130. u32 odu_teth_cons_hdl;
  131. u32 ipa_sys_desc_size;
  132. void *logbuf;
  133. void *logbuf_low;
  134. void (*wakeup_request)(void *cl_priv);
  135. u32 pm_hdl;
  136. };
  137. static struct odu_bridge_ctx *odu_bridge_ctx;
  138. #ifdef CONFIG_DEBUG_FS
  139. #define ODU_MAX_MSG_LEN 512
  140. static char dbg_buff[ODU_MAX_MSG_LEN];
  141. #endif
  142. static void odu_bridge_emb_cons_cb(void *priv, enum ipa_dp_evt_type evt,
  143. unsigned long data)
  144. {
  145. ODU_BRIDGE_FUNC_ENTRY();
  146. if (evt != IPA_RECEIVE) {
  147. ODU_BRIDGE_ERR("unexpected event\n");
  148. WARN_ON(1);
  149. return;
  150. }
  151. odu_bridge_ctx->send_dl_skb(priv, (struct sk_buff *)data);
  152. odu_bridge_ctx->stats.num_dl_packets++;
  153. ODU_BRIDGE_FUNC_EXIT();
  154. }
  155. static void odu_bridge_teth_cons_cb(void *priv, enum ipa_dp_evt_type evt,
  156. unsigned long data)
  157. {
  158. struct ipv6hdr *ipv6hdr;
  159. struct sk_buff *skb = (struct sk_buff *)data;
  160. struct sk_buff *skb_copied;
  161. ODU_BRIDGE_FUNC_ENTRY();
  162. if (evt != IPA_RECEIVE) {
  163. ODU_BRIDGE_ERR("unexpected event\n");
  164. WARN_ON(1);
  165. return;
  166. }
  167. ipv6hdr = (struct ipv6hdr *)(skb->data + ETH_HLEN);
  168. if (ipv6hdr->version == 6 &&
  169. ipv6_addr_is_multicast(&ipv6hdr->daddr)) {
  170. ODU_BRIDGE_DBG_LOW("Multicast pkt, send to APPS and adapter\n");
  171. skb_copied = skb_clone(skb, GFP_KERNEL);
  172. if (skb_copied) {
  173. odu_bridge_ctx->tx_dp_notify(odu_bridge_ctx->priv,
  174. IPA_RECEIVE,
  175. (unsigned long) skb_copied);
  176. odu_bridge_ctx->stats.num_lan_packets++;
  177. } else {
  178. ODU_BRIDGE_ERR("No memory\n");
  179. }
  180. }
  181. odu_bridge_ctx->send_dl_skb(priv, skb);
  182. odu_bridge_ctx->stats.num_dl_packets++;
  183. ODU_BRIDGE_FUNC_EXIT();
  184. }
  185. static int odu_bridge_connect_router(void)
  186. {
  187. struct ipa_sys_connect_params odu_prod_params;
  188. struct ipa_sys_connect_params odu_emb_cons_params;
  189. int res;
  190. ODU_BRIDGE_FUNC_ENTRY();
  191. memset(&odu_prod_params, 0, sizeof(odu_prod_params));
  192. memset(&odu_emb_cons_params, 0, sizeof(odu_emb_cons_params));
  193. /* configure RX (ODU->IPA) EP */
  194. odu_prod_params.client = IPA_CLIENT_ODU_PROD;
  195. odu_prod_params.ipa_ep_cfg.hdr.hdr_len = ETH_HLEN;
  196. odu_prod_params.ipa_ep_cfg.nat.nat_en = IPA_SRC_NAT;
  197. odu_prod_params.desc_fifo_sz = odu_bridge_ctx->ipa_sys_desc_size;
  198. odu_prod_params.priv = odu_bridge_ctx->priv;
  199. odu_prod_params.notify = odu_bridge_ctx->tx_dp_notify;
  200. res = ipa_setup_sys_pipe(&odu_prod_params,
  201. &odu_bridge_ctx->odu_prod_hdl);
  202. if (res) {
  203. ODU_BRIDGE_ERR("fail to setup sys pipe ODU_PROD %d\n", res);
  204. goto fail_odu_prod;
  205. }
  206. /* configure TX (IPA->ODU) EP */
  207. odu_emb_cons_params.client = IPA_CLIENT_ODU_EMB_CONS;
  208. odu_emb_cons_params.ipa_ep_cfg.hdr.hdr_len = ETH_HLEN;
  209. odu_emb_cons_params.ipa_ep_cfg.nat.nat_en = IPA_BYPASS_NAT;
  210. odu_emb_cons_params.desc_fifo_sz = odu_bridge_ctx->ipa_sys_desc_size;
  211. odu_emb_cons_params.priv = odu_bridge_ctx->priv;
  212. odu_emb_cons_params.notify = odu_bridge_emb_cons_cb;
  213. res = ipa_setup_sys_pipe(&odu_emb_cons_params,
  214. &odu_bridge_ctx->odu_emb_cons_hdl);
  215. if (res) {
  216. ODU_BRIDGE_ERR("fail to setup sys pipe ODU_EMB_CONS %d\n", res);
  217. goto fail_odu_emb_cons;
  218. }
  219. ODU_BRIDGE_DBG("odu_prod_hdl = %d, odu_emb_cons_hdl = %d\n",
  220. odu_bridge_ctx->odu_prod_hdl, odu_bridge_ctx->odu_emb_cons_hdl);
  221. ODU_BRIDGE_FUNC_EXIT();
  222. return 0;
  223. fail_odu_emb_cons:
  224. ipa_teardown_sys_pipe(odu_bridge_ctx->odu_prod_hdl);
  225. odu_bridge_ctx->odu_prod_hdl = 0;
  226. fail_odu_prod:
  227. return res;
  228. }
  229. static int odu_bridge_connect_bridge(void)
  230. {
  231. struct ipa_sys_connect_params odu_prod_params;
  232. struct ipa_sys_connect_params odu_emb_cons_params;
  233. struct ipa_sys_connect_params odu_teth_cons_params;
  234. int res;
  235. ODU_BRIDGE_FUNC_ENTRY();
  236. memset(&odu_prod_params, 0, sizeof(odu_prod_params));
  237. memset(&odu_emb_cons_params, 0, sizeof(odu_emb_cons_params));
  238. /* configure RX (ODU->IPA) EP */
  239. odu_prod_params.client = IPA_CLIENT_ODU_PROD;
  240. odu_prod_params.desc_fifo_sz = IPA_ODU_SYS_DESC_FIFO_SZ;
  241. odu_prod_params.priv = odu_bridge_ctx->priv;
  242. odu_prod_params.notify = odu_bridge_ctx->tx_dp_notify;
  243. odu_prod_params.skip_ep_cfg = true;
  244. res = ipa_setup_sys_pipe(&odu_prod_params,
  245. &odu_bridge_ctx->odu_prod_hdl);
  246. if (res) {
  247. ODU_BRIDGE_ERR("fail to setup sys pipe ODU_PROD %d\n", res);
  248. goto fail_odu_prod;
  249. }
  250. /* configure TX tethered (IPA->ODU) EP */
  251. odu_teth_cons_params.client = IPA_CLIENT_ODU_TETH_CONS;
  252. odu_teth_cons_params.desc_fifo_sz = IPA_ODU_SYS_DESC_FIFO_SZ;
  253. odu_teth_cons_params.priv = odu_bridge_ctx->priv;
  254. odu_teth_cons_params.notify = odu_bridge_teth_cons_cb;
  255. odu_teth_cons_params.skip_ep_cfg = true;
  256. res = ipa_setup_sys_pipe(&odu_teth_cons_params,
  257. &odu_bridge_ctx->odu_teth_cons_hdl);
  258. if (res) {
  259. ODU_BRIDGE_ERR("fail to setup sys pipe ODU_TETH_CONS %d\n",
  260. res);
  261. goto fail_odu_teth_cons;
  262. }
  263. /* configure TX embedded(IPA->ODU) EP */
  264. odu_emb_cons_params.client = IPA_CLIENT_ODU_EMB_CONS;
  265. odu_emb_cons_params.ipa_ep_cfg.hdr.hdr_len = ETH_HLEN;
  266. odu_emb_cons_params.ipa_ep_cfg.nat.nat_en = IPA_BYPASS_NAT;
  267. odu_emb_cons_params.desc_fifo_sz = IPA_ODU_SYS_DESC_FIFO_SZ;
  268. odu_emb_cons_params.priv = odu_bridge_ctx->priv;
  269. odu_emb_cons_params.notify = odu_bridge_emb_cons_cb;
  270. res = ipa_setup_sys_pipe(&odu_emb_cons_params,
  271. &odu_bridge_ctx->odu_emb_cons_hdl);
  272. if (res) {
  273. ODU_BRIDGE_ERR("fail to setup sys pipe ODU_EMB_CONS %d\n", res);
  274. goto fail_odu_emb_cons;
  275. }
  276. ODU_BRIDGE_DBG_LOW("odu_prod_hdl = %d, odu_emb_cons_hdl = %d\n",
  277. odu_bridge_ctx->odu_prod_hdl, odu_bridge_ctx->odu_emb_cons_hdl);
  278. ODU_BRIDGE_DBG_LOW("odu_teth_cons_hdl = %d\n",
  279. odu_bridge_ctx->odu_teth_cons_hdl);
  280. ODU_BRIDGE_FUNC_EXIT();
  281. return 0;
  282. fail_odu_emb_cons:
  283. ipa_teardown_sys_pipe(odu_bridge_ctx->odu_teth_cons_hdl);
  284. odu_bridge_ctx->odu_teth_cons_hdl = 0;
  285. fail_odu_teth_cons:
  286. ipa_teardown_sys_pipe(odu_bridge_ctx->odu_prod_hdl);
  287. odu_bridge_ctx->odu_prod_hdl = 0;
  288. fail_odu_prod:
  289. return res;
  290. }
  291. static int odu_bridge_disconnect_router(void)
  292. {
  293. int res;
  294. ODU_BRIDGE_FUNC_ENTRY();
  295. res = ipa_teardown_sys_pipe(odu_bridge_ctx->odu_prod_hdl);
  296. if (res)
  297. ODU_BRIDGE_ERR("teardown ODU PROD failed\n");
  298. odu_bridge_ctx->odu_prod_hdl = 0;
  299. res = ipa_teardown_sys_pipe(odu_bridge_ctx->odu_emb_cons_hdl);
  300. if (res)
  301. ODU_BRIDGE_ERR("teardown ODU EMB CONS failed\n");
  302. odu_bridge_ctx->odu_emb_cons_hdl = 0;
  303. ODU_BRIDGE_FUNC_EXIT();
  304. return 0;
  305. }
  306. static int odu_bridge_disconnect_bridge(void)
  307. {
  308. int res;
  309. ODU_BRIDGE_FUNC_ENTRY();
  310. res = ipa_teardown_sys_pipe(odu_bridge_ctx->odu_prod_hdl);
  311. if (res)
  312. ODU_BRIDGE_ERR("teardown ODU PROD failed\n");
  313. odu_bridge_ctx->odu_prod_hdl = 0;
  314. res = ipa_teardown_sys_pipe(odu_bridge_ctx->odu_teth_cons_hdl);
  315. if (res)
  316. ODU_BRIDGE_ERR("teardown ODU TETH CONS failed\n");
  317. odu_bridge_ctx->odu_teth_cons_hdl = 0;
  318. res = ipa_teardown_sys_pipe(odu_bridge_ctx->odu_emb_cons_hdl);
  319. if (res)
  320. ODU_BRIDGE_ERR("teardown ODU EMB CONS failed\n");
  321. odu_bridge_ctx->odu_emb_cons_hdl = 0;
  322. return 0;
  323. }
  324. /**
  325. * odu_bridge_disconnect() - Disconnect odu bridge
  326. *
  327. * Disconnect all pipes
  328. *
  329. * Return codes: 0- success, error otherwise
  330. */
  331. int odu_bridge_disconnect(void)
  332. {
  333. int res;
  334. ODU_BRIDGE_FUNC_ENTRY();
  335. if (!odu_bridge_ctx) {
  336. ODU_BRIDGE_ERR("Not initialized\n");
  337. return -EFAULT;
  338. }
  339. if (!odu_bridge_ctx->is_connected) {
  340. ODU_BRIDGE_ERR("Not connected\n");
  341. return -EFAULT;
  342. }
  343. mutex_lock(&odu_bridge_ctx->lock);
  344. if (odu_bridge_ctx->mode == ODU_BRIDGE_MODE_ROUTER) {
  345. res = odu_bridge_disconnect_router();
  346. if (res) {
  347. ODU_BRIDGE_ERR("disconnect_router failed %d\n", res);
  348. goto out;
  349. }
  350. } else {
  351. res = odu_bridge_disconnect_bridge();
  352. if (res) {
  353. ODU_BRIDGE_ERR("disconnect_bridge failed %d\n", res);
  354. goto out;
  355. }
  356. }
  357. odu_bridge_ctx->is_connected = false;
  358. res = 0;
  359. out:
  360. mutex_unlock(&odu_bridge_ctx->lock);
  361. ODU_BRIDGE_FUNC_EXIT();
  362. return res;
  363. }
  364. EXPORT_SYMBOL(odu_bridge_disconnect);
  365. /**
  366. * odu_bridge_connect() - Connect odu bridge.
  367. *
  368. * Call to the mode-specific connect function for connection IPA pipes
  369. * Return codes: 0: success
  370. * -EINVAL: invalid parameters
  371. * -EPERM: Operation not permitted as the bridge is already
  372. * connected
  373. */
  374. int odu_bridge_connect(void)
  375. {
  376. int res;
  377. ODU_BRIDGE_FUNC_ENTRY();
  378. if (!odu_bridge_ctx) {
  379. ODU_BRIDGE_ERR("Not initialized\n");
  380. return -EFAULT;
  381. }
  382. if (odu_bridge_ctx->is_connected) {
  383. ODU_BRIDGE_ERR("already connected\n");
  384. return -EFAULT;
  385. }
  386. mutex_lock(&odu_bridge_ctx->lock);
  387. if (odu_bridge_ctx->mode == ODU_BRIDGE_MODE_ROUTER) {
  388. res = odu_bridge_connect_router();
  389. if (res) {
  390. ODU_BRIDGE_ERR("connect_router failed\n");
  391. goto bail;
  392. }
  393. } else {
  394. res = odu_bridge_connect_bridge();
  395. if (res) {
  396. ODU_BRIDGE_ERR("connect_bridge failed\n");
  397. goto bail;
  398. }
  399. }
  400. odu_bridge_ctx->is_connected = true;
  401. res = 0;
  402. bail:
  403. mutex_unlock(&odu_bridge_ctx->lock);
  404. ODU_BRIDGE_FUNC_EXIT();
  405. return res;
  406. }
  407. EXPORT_SYMBOL(odu_bridge_connect);
  408. /**
  409. * odu_bridge_set_mode() - Set bridge mode to Router/Bridge
  410. * @mode: mode to be set
  411. */
  412. static int odu_bridge_set_mode(enum odu_bridge_mode mode)
  413. {
  414. int res;
  415. ODU_BRIDGE_FUNC_ENTRY();
  416. if (mode < 0 || mode >= ODU_BRIDGE_MODE_MAX) {
  417. ODU_BRIDGE_ERR("Unsupported mode: %d\n", mode);
  418. return -EFAULT;
  419. }
  420. ODU_BRIDGE_DBG_LOW("setting mode: %d\n", mode);
  421. mutex_lock(&odu_bridge_ctx->lock);
  422. if (odu_bridge_ctx->mode == mode) {
  423. ODU_BRIDGE_DBG_LOW("same mode\n");
  424. res = 0;
  425. goto bail;
  426. }
  427. if (odu_bridge_ctx->is_connected) {
  428. /* first disconnect the old configuration */
  429. if (odu_bridge_ctx->mode == ODU_BRIDGE_MODE_ROUTER) {
  430. res = odu_bridge_disconnect_router();
  431. if (res) {
  432. ODU_BRIDGE_ERR("disconnect_router failed\n");
  433. goto bail;
  434. }
  435. } else {
  436. res = odu_bridge_disconnect_bridge();
  437. if (res) {
  438. ODU_BRIDGE_ERR("disconnect_bridge failed\n");
  439. goto bail;
  440. }
  441. }
  442. /* connect the new configuration */
  443. if (mode == ODU_BRIDGE_MODE_ROUTER) {
  444. res = odu_bridge_connect_router();
  445. if (res) {
  446. ODU_BRIDGE_ERR("connect_router failed\n");
  447. goto bail;
  448. }
  449. } else {
  450. res = odu_bridge_connect_bridge();
  451. if (res) {
  452. ODU_BRIDGE_ERR("connect_bridge failed\n");
  453. goto bail;
  454. }
  455. }
  456. }
  457. odu_bridge_ctx->mode = mode;
  458. res = 0;
  459. bail:
  460. mutex_unlock(&odu_bridge_ctx->lock);
  461. ODU_BRIDGE_FUNC_EXIT();
  462. return res;
  463. };
  464. /**
  465. * odu_bridge_set_llv6_addr() - Set link local ipv6 address
  466. * @llv6_addr: odu network interface link local address
  467. *
  468. * This function sets the link local ipv6 address provided by IOCTL
  469. */
  470. static int odu_bridge_set_llv6_addr(struct in6_addr *llv6_addr)
  471. {
  472. struct in6_addr llv6_addr_host;
  473. ODU_BRIDGE_FUNC_ENTRY();
  474. llv6_addr_host.s6_addr32[0] = ntohl(llv6_addr->s6_addr32[0]);
  475. llv6_addr_host.s6_addr32[1] = ntohl(llv6_addr->s6_addr32[1]);
  476. llv6_addr_host.s6_addr32[2] = ntohl(llv6_addr->s6_addr32[2]);
  477. llv6_addr_host.s6_addr32[3] = ntohl(llv6_addr->s6_addr32[3]);
  478. memcpy(&odu_bridge_ctx->llv6_addr, &llv6_addr_host,
  479. sizeof(odu_bridge_ctx->llv6_addr));
  480. ODU_BRIDGE_DBG_LOW("LLV6 addr: %pI6c\n", &odu_bridge_ctx->llv6_addr);
  481. ODU_BRIDGE_FUNC_EXIT();
  482. return 0;
  483. };
  484. static long odu_bridge_ioctl(struct file *filp,
  485. unsigned int cmd,
  486. unsigned long arg)
  487. {
  488. int res = 0;
  489. struct in6_addr llv6_addr;
  490. ODU_BRIDGE_DBG("cmd=%x nr=%d\n", cmd, _IOC_NR(cmd));
  491. if ((_IOC_TYPE(cmd) != ODU_BRIDGE_IOC_MAGIC) ||
  492. (_IOC_NR(cmd) >= ODU_BRIDGE_IOCTL_MAX)) {
  493. ODU_BRIDGE_ERR("Invalid ioctl\n");
  494. return -ENOIOCTLCMD;
  495. }
  496. switch (cmd) {
  497. case ODU_BRIDGE_IOC_SET_MODE:
  498. ODU_BRIDGE_DBG("ODU_BRIDGE_IOC_SET_MODE ioctl called\n");
  499. res = odu_bridge_set_mode(arg);
  500. if (res) {
  501. ODU_BRIDGE_ERR("Error, res = %d\n", res);
  502. break;
  503. }
  504. break;
  505. case ODU_BRIDGE_IOC_SET_LLV6_ADDR:
  506. ODU_BRIDGE_DBG("ODU_BRIDGE_IOC_SET_LLV6_ADDR ioctl called\n");
  507. res = copy_from_user(&llv6_addr,
  508. (struct in6_addr *)arg,
  509. sizeof(llv6_addr));
  510. if (res) {
  511. ODU_BRIDGE_ERR("Error, res = %d\n", res);
  512. res = -EFAULT;
  513. break;
  514. }
  515. res = odu_bridge_set_llv6_addr(&llv6_addr);
  516. if (res) {
  517. ODU_BRIDGE_ERR("Error, res = %d\n", res);
  518. break;
  519. }
  520. break;
  521. default:
  522. ODU_BRIDGE_ERR("Unknown ioctl: %d\n", cmd);
  523. WARN_ON(1);
  524. }
  525. return res;
  526. }
  527. #ifdef CONFIG_COMPAT
  528. static long compat_odu_bridge_ioctl(struct file *file,
  529. unsigned int cmd, unsigned long arg)
  530. {
  531. switch (cmd) {
  532. case ODU_BRIDGE_IOC_SET_LLV6_ADDR32:
  533. cmd = ODU_BRIDGE_IOC_SET_LLV6_ADDR;
  534. break;
  535. case ODU_BRIDGE_IOC_SET_MODE:
  536. break;
  537. default:
  538. return -ENOIOCTLCMD;
  539. }
  540. return odu_bridge_ioctl(file, cmd, (unsigned long)compat_ptr(arg));
  541. }
  542. #endif
  543. #ifdef CONFIG_DEBUG_FS
  544. static struct dentry *dent;
  545. static struct dentry *dfile_stats;
  546. static struct dentry *dfile_mode;
  547. static ssize_t odu_debugfs_stats(struct file *file,
  548. char __user *ubuf,
  549. size_t count,
  550. loff_t *ppos)
  551. {
  552. int nbytes = 0;
  553. nbytes += scnprintf(&dbg_buff[nbytes],
  554. ODU_MAX_MSG_LEN - nbytes,
  555. "UL packets: %lld\n",
  556. odu_bridge_ctx->stats.num_ul_packets);
  557. nbytes += scnprintf(&dbg_buff[nbytes],
  558. ODU_MAX_MSG_LEN - nbytes,
  559. "DL packets: %lld\n",
  560. odu_bridge_ctx->stats.num_dl_packets);
  561. nbytes += scnprintf(&dbg_buff[nbytes],
  562. ODU_MAX_MSG_LEN - nbytes,
  563. "LAN packets: %lld\n",
  564. odu_bridge_ctx->stats.num_lan_packets);
  565. return simple_read_from_buffer(ubuf, count, ppos, dbg_buff, nbytes);
  566. }
  567. static ssize_t odu_debugfs_hw_bridge_mode_write(struct file *file,
  568. const char __user *ubuf,
  569. size_t count,
  570. loff_t *ppos)
  571. {
  572. unsigned long missing;
  573. enum odu_bridge_mode mode;
  574. if (count >= sizeof(dbg_buff))
  575. return -EFAULT;
  576. missing = copy_from_user(dbg_buff, ubuf, count);
  577. if (missing)
  578. return -EFAULT;
  579. if (count > 0)
  580. dbg_buff[count-1] = '\0';
  581. if (strcmp(dbg_buff, "router") == 0) {
  582. mode = ODU_BRIDGE_MODE_ROUTER;
  583. } else if (strcmp(dbg_buff, "bridge") == 0) {
  584. mode = ODU_BRIDGE_MODE_BRIDGE;
  585. } else {
  586. ODU_BRIDGE_ERR("Bad mode, got %s,\n"
  587. "Use <router> or <bridge>.\n", dbg_buff);
  588. return count;
  589. }
  590. odu_bridge_set_mode(mode);
  591. return count;
  592. }
  593. static ssize_t odu_debugfs_hw_bridge_mode_read(struct file *file,
  594. char __user *ubuf,
  595. size_t count,
  596. loff_t *ppos)
  597. {
  598. int nbytes = 0;
  599. switch (odu_bridge_ctx->mode) {
  600. case ODU_BRIDGE_MODE_ROUTER:
  601. nbytes += scnprintf(&dbg_buff[nbytes],
  602. ODU_MAX_MSG_LEN - nbytes,
  603. "router\n");
  604. break;
  605. case ODU_BRIDGE_MODE_BRIDGE:
  606. nbytes += scnprintf(&dbg_buff[nbytes],
  607. ODU_MAX_MSG_LEN - nbytes,
  608. "bridge\n");
  609. break;
  610. default:
  611. nbytes += scnprintf(&dbg_buff[nbytes],
  612. ODU_MAX_MSG_LEN - nbytes,
  613. "mode error\n");
  614. break;
  615. }
  616. return simple_read_from_buffer(ubuf, count, ppos, dbg_buff, nbytes);
  617. }
  618. const struct file_operations odu_stats_ops = {
  619. .read = odu_debugfs_stats,
  620. };
  621. const struct file_operations odu_hw_bridge_mode_ops = {
  622. .read = odu_debugfs_hw_bridge_mode_read,
  623. .write = odu_debugfs_hw_bridge_mode_write,
  624. };
  625. static void odu_debugfs_init(void)
  626. {
  627. const mode_t read_only_mode = 0444;
  628. const mode_t read_write_mode = 0666;
  629. dent = debugfs_create_dir("odu_ipa_bridge", 0);
  630. if (IS_ERR(dent)) {
  631. ODU_BRIDGE_ERR("fail to create folder odu_ipa_bridge\n");
  632. return;
  633. }
  634. dfile_stats =
  635. debugfs_create_file("stats", read_only_mode, dent,
  636. 0, &odu_stats_ops);
  637. if (!dfile_stats || IS_ERR(dfile_stats)) {
  638. ODU_BRIDGE_ERR("fail to create file stats\n");
  639. goto fail;
  640. }
  641. dfile_mode =
  642. debugfs_create_file("mode", read_write_mode,
  643. dent, 0, &odu_hw_bridge_mode_ops);
  644. if (!dfile_mode ||
  645. IS_ERR(dfile_mode)) {
  646. ODU_BRIDGE_ERR("fail to create file dfile_mode\n");
  647. goto fail;
  648. }
  649. return;
  650. fail:
  651. debugfs_remove_recursive(dent);
  652. }
  653. static void odu_debugfs_destroy(void)
  654. {
  655. debugfs_remove_recursive(dent);
  656. }
  657. #else
  658. static void odu_debugfs_init(void) {}
  659. static void odu_debugfs_destroy(void) {}
  660. #endif /* CONFIG_DEBUG_FS */
  661. static const struct file_operations odu_bridge_drv_fops = {
  662. .owner = THIS_MODULE,
  663. .unlocked_ioctl = odu_bridge_ioctl,
  664. #ifdef CONFIG_COMPAT
  665. .compat_ioctl = compat_odu_bridge_ioctl,
  666. #endif
  667. };
  668. /**
  669. * odu_bridge_tx_dp() - Send skb to ODU bridge
  670. * @skb: skb to send
  671. * @metadata: metadata on packet
  672. *
  673. * This function handles uplink packet.
  674. * In Router Mode:
  675. * packet is sent directly to IPA.
  676. * In Router Mode:
  677. * packet is classified if it should arrive to network stack.
  678. * QMI IP packet should arrive to APPS network stack
  679. * IPv6 Multicast packet should arrive to APPS network stack and Q6
  680. *
  681. * Return codes: 0- success, error otherwise
  682. */
  683. int odu_bridge_tx_dp(struct sk_buff *skb, struct ipa_tx_meta *metadata)
  684. {
  685. struct sk_buff *skb_copied = NULL;
  686. struct ipv6hdr *ipv6hdr;
  687. int res;
  688. ODU_BRIDGE_FUNC_ENTRY();
  689. switch (odu_bridge_ctx->mode) {
  690. case ODU_BRIDGE_MODE_ROUTER:
  691. /* Router mode - pass skb to IPA */
  692. res = ipa_tx_dp(IPA_CLIENT_ODU_PROD, skb, metadata);
  693. if (res) {
  694. ODU_BRIDGE_DBG("tx dp failed %d\n", res);
  695. goto out;
  696. }
  697. odu_bridge_ctx->stats.num_ul_packets++;
  698. goto out;
  699. case ODU_BRIDGE_MODE_BRIDGE:
  700. ipv6hdr = (struct ipv6hdr *)(skb->data + ETH_HLEN);
  701. if (ipv6hdr->version == 6 &&
  702. ODU_BRIDGE_IS_QMI_ADDR(ipv6hdr->daddr)) {
  703. ODU_BRIDGE_DBG_LOW("QMI packet\n");
  704. skb_copied = skb_clone(skb, GFP_KERNEL);
  705. if (!skb_copied) {
  706. ODU_BRIDGE_ERR("No memory\n");
  707. return -ENOMEM;
  708. }
  709. odu_bridge_ctx->tx_dp_notify(odu_bridge_ctx->priv,
  710. IPA_RECEIVE,
  711. (unsigned long)skb_copied);
  712. odu_bridge_ctx->tx_dp_notify(odu_bridge_ctx->priv,
  713. IPA_WRITE_DONE,
  714. (unsigned long)skb);
  715. odu_bridge_ctx->stats.num_ul_packets++;
  716. odu_bridge_ctx->stats.num_lan_packets++;
  717. res = 0;
  718. goto out;
  719. }
  720. if (ipv6hdr->version == 6 &&
  721. ipv6_addr_is_multicast(&ipv6hdr->daddr)) {
  722. ODU_BRIDGE_DBG_LOW(
  723. "Multicast pkt, send to APPS and IPA\n");
  724. skb_copied = skb_clone(skb, GFP_KERNEL);
  725. if (!skb_copied) {
  726. ODU_BRIDGE_ERR("No memory\n");
  727. return -ENOMEM;
  728. }
  729. res = ipa_tx_dp(IPA_CLIENT_ODU_PROD, skb, metadata);
  730. if (res) {
  731. ODU_BRIDGE_DBG("tx dp failed %d\n", res);
  732. dev_kfree_skb(skb_copied);
  733. goto out;
  734. }
  735. odu_bridge_ctx->tx_dp_notify(odu_bridge_ctx->priv,
  736. IPA_RECEIVE,
  737. (unsigned long)skb_copied);
  738. odu_bridge_ctx->stats.num_ul_packets++;
  739. odu_bridge_ctx->stats.num_lan_packets++;
  740. goto out;
  741. }
  742. res = ipa_tx_dp(IPA_CLIENT_ODU_PROD, skb, metadata);
  743. if (res) {
  744. ODU_BRIDGE_DBG("tx dp failed %d\n", res);
  745. goto out;
  746. }
  747. odu_bridge_ctx->stats.num_ul_packets++;
  748. goto out;
  749. default:
  750. ODU_BRIDGE_ERR("Unsupported mode: %d\n", odu_bridge_ctx->mode);
  751. WARN_ON(1);
  752. res = -EFAULT;
  753. }
  754. out:
  755. ODU_BRIDGE_FUNC_EXIT();
  756. return res;
  757. }
  758. EXPORT_SYMBOL(odu_bridge_tx_dp);
  759. static int odu_bridge_add_hdrs(void)
  760. {
  761. struct ipa_ioc_add_hdr *hdrs;
  762. struct ipa_hdr_add *ipv4_hdr;
  763. struct ipa_hdr_add *ipv6_hdr;
  764. struct ethhdr *eth_ipv4;
  765. struct ethhdr *eth_ipv6;
  766. int res;
  767. ODU_BRIDGE_FUNC_ENTRY();
  768. hdrs = kzalloc(sizeof(*hdrs) + sizeof(*ipv4_hdr) + sizeof(*ipv6_hdr),
  769. GFP_KERNEL);
  770. if (!hdrs) {
  771. ODU_BRIDGE_ERR("no mem\n");
  772. res = -ENOMEM;
  773. goto out;
  774. }
  775. ipv4_hdr = &hdrs->hdr[0];
  776. eth_ipv4 = (struct ethhdr *)(ipv4_hdr->hdr);
  777. ipv6_hdr = &hdrs->hdr[1];
  778. eth_ipv6 = (struct ethhdr *)(ipv6_hdr->hdr);
  779. strlcpy(ipv4_hdr->name, ODU_BRIDGE_IPV4_HDR_NAME,
  780. IPA_RESOURCE_NAME_MAX);
  781. memcpy(eth_ipv4->h_source, odu_bridge_ctx->device_ethaddr, ETH_ALEN);
  782. eth_ipv4->h_proto = htons(ETH_P_IP);
  783. ipv4_hdr->hdr_len = ETH_HLEN;
  784. ipv4_hdr->is_partial = 1;
  785. ipv4_hdr->is_eth2_ofst_valid = 1;
  786. ipv4_hdr->eth2_ofst = 0;
  787. strlcpy(ipv6_hdr->name, ODU_BRIDGE_IPV6_HDR_NAME,
  788. IPA_RESOURCE_NAME_MAX);
  789. memcpy(eth_ipv6->h_source, odu_bridge_ctx->device_ethaddr, ETH_ALEN);
  790. eth_ipv6->h_proto = htons(ETH_P_IPV6);
  791. ipv6_hdr->hdr_len = ETH_HLEN;
  792. ipv6_hdr->is_partial = 1;
  793. ipv6_hdr->is_eth2_ofst_valid = 1;
  794. ipv6_hdr->eth2_ofst = 0;
  795. hdrs->commit = 1;
  796. hdrs->num_hdrs = 2;
  797. res = ipa_add_hdr(hdrs);
  798. if (res) {
  799. ODU_BRIDGE_ERR("Fail on Header-Insertion(%d)\n", res);
  800. goto out_free_mem;
  801. }
  802. if (ipv4_hdr->status) {
  803. ODU_BRIDGE_ERR("Fail on Header-Insertion ipv4(%d)\n",
  804. ipv4_hdr->status);
  805. res = ipv4_hdr->status;
  806. goto out_free_mem;
  807. }
  808. if (ipv6_hdr->status) {
  809. ODU_BRIDGE_ERR("Fail on Header-Insertion ipv6(%d)\n",
  810. ipv6_hdr->status);
  811. res = ipv6_hdr->status;
  812. goto out_free_mem;
  813. }
  814. odu_bridge_ctx->odu_br_ipv4_hdr_hdl = ipv4_hdr->hdr_hdl;
  815. odu_bridge_ctx->odu_br_ipv6_hdr_hdl = ipv6_hdr->hdr_hdl;
  816. res = 0;
  817. out_free_mem:
  818. kfree(hdrs);
  819. out:
  820. ODU_BRIDGE_FUNC_EXIT();
  821. return res;
  822. }
  823. static void odu_bridge_del_hdrs(void)
  824. {
  825. struct ipa_ioc_del_hdr *del_hdr;
  826. struct ipa_hdr_del *ipv4;
  827. struct ipa_hdr_del *ipv6;
  828. int result;
  829. del_hdr = kzalloc(sizeof(*del_hdr) + sizeof(*ipv4) +
  830. sizeof(*ipv6), GFP_KERNEL);
  831. if (!del_hdr)
  832. return;
  833. del_hdr->commit = 1;
  834. del_hdr->num_hdls = 2;
  835. ipv4 = &del_hdr->hdl[0];
  836. ipv4->hdl = odu_bridge_ctx->odu_br_ipv4_hdr_hdl;
  837. ipv6 = &del_hdr->hdl[1];
  838. ipv6->hdl = odu_bridge_ctx->odu_br_ipv6_hdr_hdl;
  839. result = ipa_del_hdr(del_hdr);
  840. if (result || ipv4->status || ipv6->status)
  841. ODU_BRIDGE_ERR("ipa_del_hdr failed");
  842. kfree(del_hdr);
  843. }
  844. /**
  845. * odu_bridge_register_properties() - set Tx/Rx properties for ipacm
  846. *
  847. * Register the network interface interface with Tx and Rx properties
  848. * Tx properties are for data flowing from IPA to adapter, they
  849. * have Header-Insertion properties both for Ipv4 and Ipv6 Ethernet framing.
  850. * Rx properties are for data flowing from adapter to IPA, they have
  851. * simple rule which always "hit".
  852. *
  853. */
  854. static int odu_bridge_register_properties(void)
  855. {
  856. struct ipa_tx_intf tx_properties = {0};
  857. struct ipa_ioc_tx_intf_prop properties[2] = { {0}, {0} };
  858. struct ipa_ioc_tx_intf_prop *ipv4_property;
  859. struct ipa_ioc_tx_intf_prop *ipv6_property;
  860. struct ipa_ioc_rx_intf_prop rx_ioc_properties[2] = { {0}, {0} };
  861. struct ipa_rx_intf rx_properties = {0};
  862. struct ipa_ioc_rx_intf_prop *rx_ipv4_property;
  863. struct ipa_ioc_rx_intf_prop *rx_ipv6_property;
  864. int res = 0;
  865. ODU_BRIDGE_FUNC_ENTRY();
  866. tx_properties.prop = properties;
  867. ipv4_property = &tx_properties.prop[0];
  868. ipv4_property->ip = IPA_IP_v4;
  869. ipv4_property->dst_pipe = IPA_CLIENT_ODU_EMB_CONS;
  870. ipv4_property->hdr_l2_type = IPA_HDR_L2_ETHERNET_II;
  871. strlcpy(ipv4_property->hdr_name, ODU_BRIDGE_IPV4_HDR_NAME,
  872. IPA_RESOURCE_NAME_MAX);
  873. ipv6_property = &tx_properties.prop[1];
  874. ipv6_property->ip = IPA_IP_v6;
  875. ipv6_property->dst_pipe = IPA_CLIENT_ODU_EMB_CONS;
  876. ipv6_property->hdr_l2_type = IPA_HDR_L2_ETHERNET_II;
  877. strlcpy(ipv6_property->hdr_name, ODU_BRIDGE_IPV6_HDR_NAME,
  878. IPA_RESOURCE_NAME_MAX);
  879. tx_properties.num_props = 2;
  880. rx_properties.prop = rx_ioc_properties;
  881. rx_ipv4_property = &rx_properties.prop[0];
  882. rx_ipv4_property->ip = IPA_IP_v4;
  883. rx_ipv4_property->attrib.attrib_mask = 0;
  884. rx_ipv4_property->src_pipe = IPA_CLIENT_ODU_PROD;
  885. rx_ipv4_property->hdr_l2_type = IPA_HDR_L2_ETHERNET_II;
  886. rx_ipv6_property = &rx_properties.prop[1];
  887. rx_ipv6_property->ip = IPA_IP_v6;
  888. rx_ipv6_property->attrib.attrib_mask = 0;
  889. rx_ipv6_property->src_pipe = IPA_CLIENT_ODU_PROD;
  890. rx_ipv6_property->hdr_l2_type = IPA_HDR_L2_ETHERNET_II;
  891. rx_properties.num_props = 2;
  892. res = ipa_register_intf(odu_bridge_ctx->netdev_name, &tx_properties,
  893. &rx_properties);
  894. if (res) {
  895. ODU_BRIDGE_ERR("fail on Tx/Rx properties registration %d\n",
  896. res);
  897. }
  898. ODU_BRIDGE_FUNC_EXIT();
  899. return res;
  900. }
  901. static void odu_bridge_deregister_properties(void)
  902. {
  903. int res;
  904. ODU_BRIDGE_FUNC_ENTRY();
  905. res = ipa_deregister_intf(odu_bridge_ctx->netdev_name);
  906. if (res)
  907. ODU_BRIDGE_ERR("Fail on Tx prop deregister %d\n", res);
  908. ODU_BRIDGE_FUNC_EXIT();
  909. }
  910. /**
  911. * odu_bridge_init() - Initialize the ODU bridge driver
  912. * @params: initialization parameters
  913. *
  914. * This function initialize all bridge internal data and register odu bridge to
  915. * kernel for IOCTL and debugfs.
  916. * Header addition and properties are registered to IPA driver.
  917. *
  918. * Return codes: 0: success,
  919. * -EINVAL - Bad parameter
  920. * Other negative value - Failure
  921. */
  922. int odu_bridge_init(struct odu_bridge_params *params)
  923. {
  924. int res;
  925. ODU_BRIDGE_FUNC_ENTRY();
  926. if (!params) {
  927. ODU_BRIDGE_ERR("null pointer params\n");
  928. return -EINVAL;
  929. }
  930. if (!params->netdev_name) {
  931. ODU_BRIDGE_ERR("null pointer params->netdev_name\n");
  932. return -EINVAL;
  933. }
  934. if (!params->tx_dp_notify) {
  935. ODU_BRIDGE_ERR("null pointer params->tx_dp_notify\n");
  936. return -EINVAL;
  937. }
  938. if (!params->send_dl_skb) {
  939. ODU_BRIDGE_ERR("null pointer params->send_dl_skb\n");
  940. return -EINVAL;
  941. }
  942. if (odu_bridge_ctx) {
  943. ODU_BRIDGE_ERR("Already initialized\n");
  944. return -EFAULT;
  945. }
  946. if (!ipa_is_ready()) {
  947. ODU_BRIDGE_ERR("IPA is not ready\n");
  948. return -EFAULT;
  949. }
  950. ODU_BRIDGE_DBG("device_ethaddr=%pM\n", params->device_ethaddr);
  951. odu_bridge_ctx = kzalloc(sizeof(*odu_bridge_ctx), GFP_KERNEL);
  952. if (!odu_bridge_ctx) {
  953. ODU_BRIDGE_ERR("kzalloc err.\n");
  954. return -ENOMEM;
  955. }
  956. odu_bridge_ctx->class = class_create(THIS_MODULE, ODU_BRIDGE_DRV_NAME);
  957. if (!odu_bridge_ctx->class) {
  958. ODU_BRIDGE_ERR("Class_create err.\n");
  959. res = -ENODEV;
  960. goto fail_class_create;
  961. }
  962. res = alloc_chrdev_region(&odu_bridge_ctx->dev_num, 0, 1,
  963. ODU_BRIDGE_DRV_NAME);
  964. if (res) {
  965. ODU_BRIDGE_ERR("alloc_chrdev_region err.\n");
  966. res = -ENODEV;
  967. goto fail_alloc_chrdev_region;
  968. }
  969. odu_bridge_ctx->dev = device_create(odu_bridge_ctx->class, NULL,
  970. odu_bridge_ctx->dev_num, odu_bridge_ctx, ODU_BRIDGE_DRV_NAME);
  971. if (IS_ERR(odu_bridge_ctx->dev)) {
  972. ODU_BRIDGE_ERR(":device_create err.\n");
  973. res = -ENODEV;
  974. goto fail_device_create;
  975. }
  976. cdev_init(&odu_bridge_ctx->cdev, &odu_bridge_drv_fops);
  977. odu_bridge_ctx->cdev.owner = THIS_MODULE;
  978. odu_bridge_ctx->cdev.ops = &odu_bridge_drv_fops;
  979. res = cdev_add(&odu_bridge_ctx->cdev, odu_bridge_ctx->dev_num, 1);
  980. if (res) {
  981. ODU_BRIDGE_ERR(":cdev_add err=%d\n", -res);
  982. res = -ENODEV;
  983. goto fail_cdev_add;
  984. }
  985. odu_debugfs_init();
  986. strlcpy(odu_bridge_ctx->netdev_name, params->netdev_name,
  987. IPA_RESOURCE_NAME_MAX);
  988. odu_bridge_ctx->priv = params->priv;
  989. odu_bridge_ctx->tx_dp_notify = params->tx_dp_notify;
  990. odu_bridge_ctx->send_dl_skb = params->send_dl_skb;
  991. memcpy(odu_bridge_ctx->device_ethaddr, params->device_ethaddr,
  992. ETH_ALEN);
  993. odu_bridge_ctx->ipa_sys_desc_size = params->ipa_desc_size;
  994. odu_bridge_ctx->mode = ODU_BRIDGE_MODE_ROUTER;
  995. mutex_init(&odu_bridge_ctx->lock);
  996. res = odu_bridge_add_hdrs();
  997. if (res) {
  998. ODU_BRIDGE_ERR("fail on odu_bridge_add_hdr %d\n", res);
  999. goto fail_add_hdrs;
  1000. }
  1001. res = odu_bridge_register_properties();
  1002. if (res) {
  1003. ODU_BRIDGE_ERR("fail on register properties %d\n", res);
  1004. goto fail_register_properties;
  1005. }
  1006. ODU_BRIDGE_FUNC_EXIT();
  1007. return 0;
  1008. fail_register_properties:
  1009. odu_bridge_del_hdrs();
  1010. fail_add_hdrs:
  1011. odu_debugfs_destroy();
  1012. fail_cdev_add:
  1013. device_destroy(odu_bridge_ctx->class, odu_bridge_ctx->dev_num);
  1014. fail_device_create:
  1015. unregister_chrdev_region(odu_bridge_ctx->dev_num, 1);
  1016. fail_alloc_chrdev_region:
  1017. class_destroy(odu_bridge_ctx->class);
  1018. fail_class_create:
  1019. kfree(odu_bridge_ctx);
  1020. odu_bridge_ctx = NULL;
  1021. return res;
  1022. }
  1023. EXPORT_SYMBOL(odu_bridge_init);
  1024. /**
  1025. * odu_bridge_cleanup() - De-Initialize the ODU bridge driver
  1026. *
  1027. * Return codes: 0: success,
  1028. * -EINVAL - Bad parameter
  1029. * Other negative value - Failure
  1030. */
  1031. int odu_bridge_cleanup(void)
  1032. {
  1033. ODU_BRIDGE_FUNC_ENTRY();
  1034. if (!odu_bridge_ctx) {
  1035. ODU_BRIDGE_ERR("Not initialized\n");
  1036. return -EFAULT;
  1037. }
  1038. if (odu_bridge_ctx->is_connected) {
  1039. ODU_BRIDGE_ERR("cannot deinit while bridge is conncetd\n");
  1040. return -EFAULT;
  1041. }
  1042. odu_bridge_deregister_properties();
  1043. odu_bridge_del_hdrs();
  1044. odu_debugfs_destroy();
  1045. cdev_del(&odu_bridge_ctx->cdev);
  1046. device_destroy(odu_bridge_ctx->class, odu_bridge_ctx->dev_num);
  1047. unregister_chrdev_region(odu_bridge_ctx->dev_num, 1);
  1048. class_destroy(odu_bridge_ctx->class);
  1049. ipc_log_context_destroy(odu_bridge_ctx->logbuf);
  1050. ipc_log_context_destroy(odu_bridge_ctx->logbuf_low);
  1051. kfree(odu_bridge_ctx);
  1052. odu_bridge_ctx = NULL;
  1053. ODU_BRIDGE_FUNC_EXIT();
  1054. return 0;
  1055. }
  1056. EXPORT_SYMBOL(odu_bridge_cleanup);
  1057. MODULE_LICENSE("GPL v2");
  1058. MODULE_DESCRIPTION("ODU bridge driver");