reassembly.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551
  1. // SPDX-License-Identifier: GPL-2.0-or-later
  2. /* 6LoWPAN fragment reassembly
  3. *
  4. * Authors:
  5. * Alexander Aring <[email protected]>
  6. *
  7. * Based on: net/ipv6/reassembly.c
  8. */
  9. #define pr_fmt(fmt) "6LoWPAN: " fmt
  10. #include <linux/net.h>
  11. #include <linux/list.h>
  12. #include <linux/netdevice.h>
  13. #include <linux/random.h>
  14. #include <linux/jhash.h>
  15. #include <linux/skbuff.h>
  16. #include <linux/slab.h>
  17. #include <linux/export.h>
  18. #include <net/ieee802154_netdev.h>
  19. #include <net/6lowpan.h>
  20. #include <net/ipv6_frag.h>
  21. #include <net/inet_frag.h>
  22. #include <net/ip.h>
  23. #include "6lowpan_i.h"
  24. static const char lowpan_frags_cache_name[] = "lowpan-frags";
  25. static struct inet_frags lowpan_frags;
  26. static int lowpan_frag_reasm(struct lowpan_frag_queue *fq, struct sk_buff *skb,
  27. struct sk_buff *prev, struct net_device *ldev);
  28. static void lowpan_frag_init(struct inet_frag_queue *q, const void *a)
  29. {
  30. const struct frag_lowpan_compare_key *key = a;
  31. BUILD_BUG_ON(sizeof(*key) > sizeof(q->key));
  32. memcpy(&q->key, key, sizeof(*key));
  33. }
  34. static void lowpan_frag_expire(struct timer_list *t)
  35. {
  36. struct inet_frag_queue *frag = from_timer(frag, t, timer);
  37. struct frag_queue *fq;
  38. fq = container_of(frag, struct frag_queue, q);
  39. spin_lock(&fq->q.lock);
  40. if (fq->q.flags & INET_FRAG_COMPLETE)
  41. goto out;
  42. inet_frag_kill(&fq->q);
  43. out:
  44. spin_unlock(&fq->q.lock);
  45. inet_frag_put(&fq->q);
  46. }
  47. static inline struct lowpan_frag_queue *
  48. fq_find(struct net *net, const struct lowpan_802154_cb *cb,
  49. const struct ieee802154_addr *src,
  50. const struct ieee802154_addr *dst)
  51. {
  52. struct netns_ieee802154_lowpan *ieee802154_lowpan =
  53. net_ieee802154_lowpan(net);
  54. struct frag_lowpan_compare_key key = {};
  55. struct inet_frag_queue *q;
  56. key.tag = cb->d_tag;
  57. key.d_size = cb->d_size;
  58. key.src = *src;
  59. key.dst = *dst;
  60. q = inet_frag_find(ieee802154_lowpan->fqdir, &key);
  61. if (!q)
  62. return NULL;
  63. return container_of(q, struct lowpan_frag_queue, q);
  64. }
  65. static int lowpan_frag_queue(struct lowpan_frag_queue *fq,
  66. struct sk_buff *skb, u8 frag_type)
  67. {
  68. struct sk_buff *prev_tail;
  69. struct net_device *ldev;
  70. int end, offset, err;
  71. /* inet_frag_queue_* functions use skb->cb; see struct ipfrag_skb_cb
  72. * in inet_fragment.c
  73. */
  74. BUILD_BUG_ON(sizeof(struct lowpan_802154_cb) > sizeof(struct inet_skb_parm));
  75. BUILD_BUG_ON(sizeof(struct lowpan_802154_cb) > sizeof(struct inet6_skb_parm));
  76. if (fq->q.flags & INET_FRAG_COMPLETE)
  77. goto err;
  78. offset = lowpan_802154_cb(skb)->d_offset << 3;
  79. end = lowpan_802154_cb(skb)->d_size;
  80. /* Is this the final fragment? */
  81. if (offset + skb->len == end) {
  82. /* If we already have some bits beyond end
  83. * or have different end, the segment is corrupted.
  84. */
  85. if (end < fq->q.len ||
  86. ((fq->q.flags & INET_FRAG_LAST_IN) && end != fq->q.len))
  87. goto err;
  88. fq->q.flags |= INET_FRAG_LAST_IN;
  89. fq->q.len = end;
  90. } else {
  91. if (end > fq->q.len) {
  92. /* Some bits beyond end -> corruption. */
  93. if (fq->q.flags & INET_FRAG_LAST_IN)
  94. goto err;
  95. fq->q.len = end;
  96. }
  97. }
  98. ldev = skb->dev;
  99. if (ldev)
  100. skb->dev = NULL;
  101. barrier();
  102. prev_tail = fq->q.fragments_tail;
  103. err = inet_frag_queue_insert(&fq->q, skb, offset, end);
  104. if (err)
  105. goto err;
  106. fq->q.stamp = skb->tstamp;
  107. fq->q.mono_delivery_time = skb->mono_delivery_time;
  108. if (frag_type == LOWPAN_DISPATCH_FRAG1)
  109. fq->q.flags |= INET_FRAG_FIRST_IN;
  110. fq->q.meat += skb->len;
  111. add_frag_mem_limit(fq->q.fqdir, skb->truesize);
  112. if (fq->q.flags == (INET_FRAG_FIRST_IN | INET_FRAG_LAST_IN) &&
  113. fq->q.meat == fq->q.len) {
  114. int res;
  115. unsigned long orefdst = skb->_skb_refdst;
  116. skb->_skb_refdst = 0UL;
  117. res = lowpan_frag_reasm(fq, skb, prev_tail, ldev);
  118. skb->_skb_refdst = orefdst;
  119. return res;
  120. }
  121. skb_dst_drop(skb);
  122. return -1;
  123. err:
  124. kfree_skb(skb);
  125. return -1;
  126. }
  127. /* Check if this packet is complete.
  128. *
  129. * It is called with locked fq, and caller must check that
  130. * queue is eligible for reassembly i.e. it is not COMPLETE,
  131. * the last and the first frames arrived and all the bits are here.
  132. */
  133. static int lowpan_frag_reasm(struct lowpan_frag_queue *fq, struct sk_buff *skb,
  134. struct sk_buff *prev_tail, struct net_device *ldev)
  135. {
  136. void *reasm_data;
  137. inet_frag_kill(&fq->q);
  138. reasm_data = inet_frag_reasm_prepare(&fq->q, skb, prev_tail);
  139. if (!reasm_data)
  140. goto out_oom;
  141. inet_frag_reasm_finish(&fq->q, skb, reasm_data, false);
  142. skb->dev = ldev;
  143. skb->tstamp = fq->q.stamp;
  144. fq->q.rb_fragments = RB_ROOT;
  145. fq->q.fragments_tail = NULL;
  146. fq->q.last_run_head = NULL;
  147. return 1;
  148. out_oom:
  149. net_dbg_ratelimited("lowpan_frag_reasm: no memory for reassembly\n");
  150. return -1;
  151. }
  152. static int lowpan_frag_rx_handlers_result(struct sk_buff *skb,
  153. lowpan_rx_result res)
  154. {
  155. switch (res) {
  156. case RX_QUEUED:
  157. return NET_RX_SUCCESS;
  158. case RX_CONTINUE:
  159. /* nobody cared about this packet */
  160. net_warn_ratelimited("%s: received unknown dispatch\n",
  161. __func__);
  162. fallthrough;
  163. default:
  164. /* all others failure */
  165. return NET_RX_DROP;
  166. }
  167. }
  168. static lowpan_rx_result lowpan_frag_rx_h_iphc(struct sk_buff *skb)
  169. {
  170. int ret;
  171. if (!lowpan_is_iphc(*skb_network_header(skb)))
  172. return RX_CONTINUE;
  173. ret = lowpan_iphc_decompress(skb);
  174. if (ret < 0)
  175. return RX_DROP;
  176. return RX_QUEUED;
  177. }
  178. static int lowpan_invoke_frag_rx_handlers(struct sk_buff *skb)
  179. {
  180. lowpan_rx_result res;
  181. #define CALL_RXH(rxh) \
  182. do { \
  183. res = rxh(skb); \
  184. if (res != RX_CONTINUE) \
  185. goto rxh_next; \
  186. } while (0)
  187. /* likely at first */
  188. CALL_RXH(lowpan_frag_rx_h_iphc);
  189. CALL_RXH(lowpan_rx_h_ipv6);
  190. rxh_next:
  191. return lowpan_frag_rx_handlers_result(skb, res);
  192. #undef CALL_RXH
  193. }
  194. #define LOWPAN_FRAG_DGRAM_SIZE_HIGH_MASK 0x07
  195. #define LOWPAN_FRAG_DGRAM_SIZE_HIGH_SHIFT 8
  196. static int lowpan_get_cb(struct sk_buff *skb, u8 frag_type,
  197. struct lowpan_802154_cb *cb)
  198. {
  199. bool fail;
  200. u8 high = 0, low = 0;
  201. __be16 d_tag = 0;
  202. fail = lowpan_fetch_skb(skb, &high, 1);
  203. fail |= lowpan_fetch_skb(skb, &low, 1);
  204. /* remove the dispatch value and use first three bits as high value
  205. * for the datagram size
  206. */
  207. cb->d_size = (high & LOWPAN_FRAG_DGRAM_SIZE_HIGH_MASK) <<
  208. LOWPAN_FRAG_DGRAM_SIZE_HIGH_SHIFT | low;
  209. fail |= lowpan_fetch_skb(skb, &d_tag, 2);
  210. cb->d_tag = ntohs(d_tag);
  211. if (frag_type == LOWPAN_DISPATCH_FRAGN) {
  212. fail |= lowpan_fetch_skb(skb, &cb->d_offset, 1);
  213. } else {
  214. skb_reset_network_header(skb);
  215. cb->d_offset = 0;
  216. /* check if datagram_size has ipv6hdr on FRAG1 */
  217. fail |= cb->d_size < sizeof(struct ipv6hdr);
  218. /* check if we can dereference the dispatch value */
  219. fail |= !skb->len;
  220. }
  221. if (unlikely(fail))
  222. return -EIO;
  223. return 0;
  224. }
  225. int lowpan_frag_rcv(struct sk_buff *skb, u8 frag_type)
  226. {
  227. struct lowpan_frag_queue *fq;
  228. struct net *net = dev_net(skb->dev);
  229. struct lowpan_802154_cb *cb = lowpan_802154_cb(skb);
  230. struct ieee802154_hdr hdr = {};
  231. int err;
  232. if (ieee802154_hdr_peek_addrs(skb, &hdr) < 0)
  233. goto err;
  234. err = lowpan_get_cb(skb, frag_type, cb);
  235. if (err < 0)
  236. goto err;
  237. if (frag_type == LOWPAN_DISPATCH_FRAG1) {
  238. err = lowpan_invoke_frag_rx_handlers(skb);
  239. if (err == NET_RX_DROP)
  240. goto err;
  241. }
  242. if (cb->d_size > IPV6_MIN_MTU) {
  243. net_warn_ratelimited("lowpan_frag_rcv: datagram size exceeds MTU\n");
  244. goto err;
  245. }
  246. fq = fq_find(net, cb, &hdr.source, &hdr.dest);
  247. if (fq != NULL) {
  248. int ret;
  249. spin_lock(&fq->q.lock);
  250. ret = lowpan_frag_queue(fq, skb, frag_type);
  251. spin_unlock(&fq->q.lock);
  252. inet_frag_put(&fq->q);
  253. return ret;
  254. }
  255. err:
  256. kfree_skb(skb);
  257. return -1;
  258. }
  259. #ifdef CONFIG_SYSCTL
  260. static struct ctl_table lowpan_frags_ns_ctl_table[] = {
  261. {
  262. .procname = "6lowpanfrag_high_thresh",
  263. .maxlen = sizeof(unsigned long),
  264. .mode = 0644,
  265. .proc_handler = proc_doulongvec_minmax,
  266. },
  267. {
  268. .procname = "6lowpanfrag_low_thresh",
  269. .maxlen = sizeof(unsigned long),
  270. .mode = 0644,
  271. .proc_handler = proc_doulongvec_minmax,
  272. },
  273. {
  274. .procname = "6lowpanfrag_time",
  275. .maxlen = sizeof(int),
  276. .mode = 0644,
  277. .proc_handler = proc_dointvec_jiffies,
  278. },
  279. { }
  280. };
  281. /* secret interval has been deprecated */
  282. static int lowpan_frags_secret_interval_unused;
  283. static struct ctl_table lowpan_frags_ctl_table[] = {
  284. {
  285. .procname = "6lowpanfrag_secret_interval",
  286. .data = &lowpan_frags_secret_interval_unused,
  287. .maxlen = sizeof(int),
  288. .mode = 0644,
  289. .proc_handler = proc_dointvec_jiffies,
  290. },
  291. { }
  292. };
  293. static int __net_init lowpan_frags_ns_sysctl_register(struct net *net)
  294. {
  295. struct ctl_table *table;
  296. struct ctl_table_header *hdr;
  297. struct netns_ieee802154_lowpan *ieee802154_lowpan =
  298. net_ieee802154_lowpan(net);
  299. table = lowpan_frags_ns_ctl_table;
  300. if (!net_eq(net, &init_net)) {
  301. table = kmemdup(table, sizeof(lowpan_frags_ns_ctl_table),
  302. GFP_KERNEL);
  303. if (table == NULL)
  304. goto err_alloc;
  305. /* Don't export sysctls to unprivileged users */
  306. if (net->user_ns != &init_user_ns)
  307. table[0].procname = NULL;
  308. }
  309. table[0].data = &ieee802154_lowpan->fqdir->high_thresh;
  310. table[0].extra1 = &ieee802154_lowpan->fqdir->low_thresh;
  311. table[1].data = &ieee802154_lowpan->fqdir->low_thresh;
  312. table[1].extra2 = &ieee802154_lowpan->fqdir->high_thresh;
  313. table[2].data = &ieee802154_lowpan->fqdir->timeout;
  314. hdr = register_net_sysctl(net, "net/ieee802154/6lowpan", table);
  315. if (hdr == NULL)
  316. goto err_reg;
  317. ieee802154_lowpan->sysctl.frags_hdr = hdr;
  318. return 0;
  319. err_reg:
  320. if (!net_eq(net, &init_net))
  321. kfree(table);
  322. err_alloc:
  323. return -ENOMEM;
  324. }
  325. static void __net_exit lowpan_frags_ns_sysctl_unregister(struct net *net)
  326. {
  327. struct ctl_table *table;
  328. struct netns_ieee802154_lowpan *ieee802154_lowpan =
  329. net_ieee802154_lowpan(net);
  330. table = ieee802154_lowpan->sysctl.frags_hdr->ctl_table_arg;
  331. unregister_net_sysctl_table(ieee802154_lowpan->sysctl.frags_hdr);
  332. if (!net_eq(net, &init_net))
  333. kfree(table);
  334. }
  335. static struct ctl_table_header *lowpan_ctl_header;
  336. static int __init lowpan_frags_sysctl_register(void)
  337. {
  338. lowpan_ctl_header = register_net_sysctl(&init_net,
  339. "net/ieee802154/6lowpan",
  340. lowpan_frags_ctl_table);
  341. return lowpan_ctl_header == NULL ? -ENOMEM : 0;
  342. }
  343. static void lowpan_frags_sysctl_unregister(void)
  344. {
  345. unregister_net_sysctl_table(lowpan_ctl_header);
  346. }
  347. #else
  348. static inline int lowpan_frags_ns_sysctl_register(struct net *net)
  349. {
  350. return 0;
  351. }
  352. static inline void lowpan_frags_ns_sysctl_unregister(struct net *net)
  353. {
  354. }
  355. static inline int __init lowpan_frags_sysctl_register(void)
  356. {
  357. return 0;
  358. }
  359. static inline void lowpan_frags_sysctl_unregister(void)
  360. {
  361. }
  362. #endif
  363. static int __net_init lowpan_frags_init_net(struct net *net)
  364. {
  365. struct netns_ieee802154_lowpan *ieee802154_lowpan =
  366. net_ieee802154_lowpan(net);
  367. int res;
  368. res = fqdir_init(&ieee802154_lowpan->fqdir, &lowpan_frags, net);
  369. if (res < 0)
  370. return res;
  371. ieee802154_lowpan->fqdir->high_thresh = IPV6_FRAG_HIGH_THRESH;
  372. ieee802154_lowpan->fqdir->low_thresh = IPV6_FRAG_LOW_THRESH;
  373. ieee802154_lowpan->fqdir->timeout = IPV6_FRAG_TIMEOUT;
  374. res = lowpan_frags_ns_sysctl_register(net);
  375. if (res < 0)
  376. fqdir_exit(ieee802154_lowpan->fqdir);
  377. return res;
  378. }
  379. static void __net_exit lowpan_frags_pre_exit_net(struct net *net)
  380. {
  381. struct netns_ieee802154_lowpan *ieee802154_lowpan =
  382. net_ieee802154_lowpan(net);
  383. fqdir_pre_exit(ieee802154_lowpan->fqdir);
  384. }
  385. static void __net_exit lowpan_frags_exit_net(struct net *net)
  386. {
  387. struct netns_ieee802154_lowpan *ieee802154_lowpan =
  388. net_ieee802154_lowpan(net);
  389. lowpan_frags_ns_sysctl_unregister(net);
  390. fqdir_exit(ieee802154_lowpan->fqdir);
  391. }
  392. static struct pernet_operations lowpan_frags_ops = {
  393. .init = lowpan_frags_init_net,
  394. .pre_exit = lowpan_frags_pre_exit_net,
  395. .exit = lowpan_frags_exit_net,
  396. };
  397. static u32 lowpan_key_hashfn(const void *data, u32 len, u32 seed)
  398. {
  399. return jhash2(data,
  400. sizeof(struct frag_lowpan_compare_key) / sizeof(u32), seed);
  401. }
  402. static u32 lowpan_obj_hashfn(const void *data, u32 len, u32 seed)
  403. {
  404. const struct inet_frag_queue *fq = data;
  405. return jhash2((const u32 *)&fq->key,
  406. sizeof(struct frag_lowpan_compare_key) / sizeof(u32), seed);
  407. }
  408. static int lowpan_obj_cmpfn(struct rhashtable_compare_arg *arg, const void *ptr)
  409. {
  410. const struct frag_lowpan_compare_key *key = arg->key;
  411. const struct inet_frag_queue *fq = ptr;
  412. return !!memcmp(&fq->key, key, sizeof(*key));
  413. }
  414. static const struct rhashtable_params lowpan_rhash_params = {
  415. .head_offset = offsetof(struct inet_frag_queue, node),
  416. .hashfn = lowpan_key_hashfn,
  417. .obj_hashfn = lowpan_obj_hashfn,
  418. .obj_cmpfn = lowpan_obj_cmpfn,
  419. .automatic_shrinking = true,
  420. };
  421. int __init lowpan_net_frag_init(void)
  422. {
  423. int ret;
  424. lowpan_frags.constructor = lowpan_frag_init;
  425. lowpan_frags.destructor = NULL;
  426. lowpan_frags.qsize = sizeof(struct frag_queue);
  427. lowpan_frags.frag_expire = lowpan_frag_expire;
  428. lowpan_frags.frags_cache_name = lowpan_frags_cache_name;
  429. lowpan_frags.rhash_params = lowpan_rhash_params;
  430. ret = inet_frags_init(&lowpan_frags);
  431. if (ret)
  432. goto out;
  433. ret = lowpan_frags_sysctl_register();
  434. if (ret)
  435. goto err_sysctl;
  436. ret = register_pernet_subsys(&lowpan_frags_ops);
  437. if (ret)
  438. goto err_pernet;
  439. out:
  440. return ret;
  441. err_pernet:
  442. lowpan_frags_sysctl_unregister();
  443. err_sysctl:
  444. inet_frags_fini(&lowpan_frags);
  445. return ret;
  446. }
  447. void lowpan_net_frag_exit(void)
  448. {
  449. lowpan_frags_sysctl_unregister();
  450. unregister_pernet_subsys(&lowpan_frags_ops);
  451. inet_frags_fini(&lowpan_frags);
  452. }