1 /*
2  * Copyright (c) 2016 Intel Corporation
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 /**
8  * @file
9  * @brief 802.15.4 6LoWPAN fragment handler implementation
10  */
11 
12 #include <zephyr/logging/log.h>
13 LOG_MODULE_REGISTER(net_ieee802154_6lo_fragment, CONFIG_NET_L2_IEEE802154_LOG_LEVEL);
14 
15 #include "6lo.h"
16 #include "6lo_private.h"
17 #include "ieee802154_6lo_fragment.h"
18 #include "net_private.h"
19 
20 #include <errno.h>
21 
22 #include <zephyr/net/net_core.h>
23 #include <zephyr/net/net_if.h>
24 #include <zephyr/net/net_pkt.h>
25 #include <zephyr/net/net_stats.h>
26 #include <zephyr/net/udp.h>
27 
28 #define NET_FRAG_DISPATCH_MASK 0xF8
29 #define NET_FRAG_OFFSET_POS    (NET_6LO_FRAG_DATAGRAM_SIZE_LEN + NET_6LO_FRAG_DATAGRAM_OFFSET_LEN)
30 
31 #define BUF_TIMEOUT K_MSEC(50)
32 
33 #define FRAG_REASSEMBLY_TIMEOUT K_SECONDS(CONFIG_NET_L2_IEEE802154_REASSEMBLY_TIMEOUT)
34 #define REASS_CACHE_SIZE	CONFIG_NET_L2_IEEE802154_FRAGMENT_REASS_CACHE_SIZE
35 
36 /**
37  *  Reassemble cache : Depends on cache size it used for reassemble
38  *  IPv6 packets simultaneously.
39  */
40 struct frag_cache {
41 	struct k_work_delayable timer; /* Reassemble timer */
42 	struct net_pkt *pkt;	       /* Reassemble packet */
43 	uint16_t size;		       /* Datagram size */
44 	uint16_t tag;		       /* Datagram tag */
45 	bool used;
46 };
47 
48 static struct frag_cache cache[REASS_CACHE_SIZE];
49 
50 /**
51  *  RFC 4944, section 5.3
52  *  If an entire payload (e.g., IPv6) datagram fits within a single 802.15.4
53  *  frame, it is unfragmented and the LoWPAN encapsulation should not contain
54  *  a fragmentation header.  If the datagram does not fit within a single
55  *  IEEE 802.15.4 frame, it SHALL be broken into link fragments.  As the
56  *  fragment offset can only express multiples of eight bytes, all link
57  *  fragments for a datagram except the last one MUST be multiples of eight
58  *  bytes in length.
59  *
60  *  RFC 7668, section 3 (IPv6 over Bluetooth Low Energy)
61  *  Functionality is comprised of link-local IPv6 addresses and stateless
62  *  IPv6 address autoconfiguration, Neighbor Discovery, and header compression
63  *  Fragmentation features from 6LoWPAN standards are not used due to Bluetooth
64  *  LE's link-layer fragmentation support.
65  */
66 
67 /**
68  *                     1                   2                   3
69  *   0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
70  *  +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
71  *  |1 1 0 0 0|    datagram_size    |         datagram_tag          |
72  *  +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
73  *
74  *                     1                   2                   3
75  *   0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
76  *  +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
77  *  |1 1 0 0 0|    datagram_size    |         datagram_tag          |
78  *  +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
79  *  |datagram_offset|
80  *  +-+-+-+-+-+-+-+-+
81  */
82 
set_datagram_size(uint8_t * ptr,uint16_t size)83 static inline void set_datagram_size(uint8_t *ptr, uint16_t size)
84 {
85 	ptr[0] |= ((size & 0x7FF) >> 8);
86 	ptr[1] = (uint8_t)size;
87 }
88 
set_datagram_tag(uint8_t * ptr,uint16_t tag)89 static inline void set_datagram_tag(uint8_t *ptr, uint16_t tag)
90 {
91 	ptr[0] = tag >> 8;
92 	ptr[1] = (uint8_t)tag;
93 }
94 
set_up_frag_hdr(struct net_buf * frag,uint16_t size,uint8_t offset)95 static inline void set_up_frag_hdr(struct net_buf *frag, uint16_t size, uint8_t offset)
96 {
97 	static uint16_t datagram_tag;
98 
99 	bool is_first_frag = !offset;
100 	uint8_t pos = frag->len;
101 
102 	if (is_first_frag) {
103 		datagram_tag++;
104 		net_buf_add(frag, NET_6LO_FRAG1_HDR_LEN);
105 		frag->data[pos] = NET_6LO_DISPATCH_FRAG1;
106 	} else {
107 		net_buf_add(frag, NET_6LO_FRAGN_HDR_LEN);
108 		frag->data[pos] = NET_6LO_DISPATCH_FRAGN;
109 	}
110 
111 	set_datagram_size(frag->data + pos, size);
112 	pos += NET_6LO_FRAG_DATAGRAM_SIZE_LEN;
113 
114 	set_datagram_tag(frag->data + pos, datagram_tag);
115 	pos += NET_6LO_FRAG_DATAGRAM_OFFSET_LEN;
116 
117 	if (!is_first_frag) {
118 		frag->data[pos] = offset;
119 	}
120 }
121 
calc_payload_capacity(struct net_buf * frag)122 static inline uint8_t calc_payload_capacity(struct net_buf *frag)
123 {
124 	uint8_t capacity = frag->size - frag->len;
125 
126 	return (capacity & 0xF8);
127 }
128 
copy_data(struct ieee802154_6lo_fragment_ctx * ctx,struct net_buf * frame_buf,uint8_t capacity)129 static inline uint8_t copy_data(struct ieee802154_6lo_fragment_ctx *ctx, struct net_buf *frame_buf,
130 				uint8_t capacity)
131 {
132 	uint8_t remainder = ctx->buf->len - (ctx->pos - ctx->buf->data);
133 	uint8_t move = MIN(remainder, capacity);
134 
135 	memcpy(frame_buf->data + frame_buf->len, ctx->pos, move);
136 	net_buf_add(frame_buf, move);
137 
138 	return move;
139 }
140 
update_fragment_ctx(struct ieee802154_6lo_fragment_ctx * ctx,uint8_t moved)141 static inline void update_fragment_ctx(struct ieee802154_6lo_fragment_ctx *ctx, uint8_t moved)
142 {
143 	uint8_t remainder = (ctx->buf->len - (ctx->pos - ctx->buf->data));
144 	bool next_frag = moved == remainder;
145 
146 	if (next_frag) {
147 		ctx->buf = ctx->buf->frags;
148 		if (ctx->buf) {
149 			ctx->pos = ctx->buf->data;
150 		}
151 	} else {
152 		ctx->pos += moved;
153 	}
154 }
155 
156 /**
157  *  ch  : compressed (IPv6) header(s)
158  *  fh  : fragment header (dispatch + size + tag + [offset])
159  *  p   : payload (first fragment holds IPv6 hdr as payload)
160  *  e   : empty space
161  *  ll  : link layer
162  *
163  *  Input frame_buf to ieee802154_6lo_fragment() looks like below
164  *
165  *  | ll |
166  *
167  *  After fragment creation, frame_buf will look like:
168  *
169  *  | ll + fh + p + e |
170  *
171  *  p being taken from current pkt buffer and position.
172  *
173  *  Space in every fragment is because fragment payload should be multiple
174  *  of 8 octets (we have predefined packets at compile time, data packet mtu
175  *  is set already).
176  *
177  *  If it's the first fragment being created, fh will not own any offset
178  *  (so it will be 1 byte smaller)
179  */
ieee802154_6lo_fragment(struct ieee802154_6lo_fragment_ctx * ctx,struct net_buf * frame_buf,bool iphc)180 struct net_buf *ieee802154_6lo_fragment(struct ieee802154_6lo_fragment_ctx *ctx,
181 					struct net_buf *frame_buf, bool iphc)
182 {
183 	uint8_t capacity;
184 
185 	set_up_frag_hdr(frame_buf, ctx->pkt_size, ctx->offset);
186 
187 	capacity = calc_payload_capacity(frame_buf);
188 	ctx->processed += capacity;
189 
190 	bool is_first_frag = !ctx->offset;
191 
192 	if (is_first_frag) {
193 		/* First fragment needs to take into account 6lo */
194 		if (iphc) {
195 			capacity -= ctx->hdr_diff;
196 		} else {
197 			/* Adding IPv6 dispatch header */
198 			capacity += 1U;
199 		}
200 	}
201 
202 	while (capacity && ctx->buf) {
203 		uint8_t moved = copy_data(ctx, frame_buf, capacity);
204 
205 		update_fragment_ctx(ctx, moved);
206 		capacity -= moved;
207 	}
208 
209 	ctx->offset = ctx->processed >> 3;
210 
211 	return ctx->buf;
212 }
213 
get_datagram_type(uint8_t * ptr)214 static inline uint8_t get_datagram_type(uint8_t *ptr)
215 {
216 	return ptr[0] & NET_FRAG_DISPATCH_MASK;
217 }
218 
get_datagram_size(uint8_t * ptr)219 static inline uint16_t get_datagram_size(uint8_t *ptr)
220 {
221 	return ((ptr[0] & 0x1F) << 8) | ptr[1];
222 }
223 
get_datagram_tag(uint8_t * ptr)224 static inline uint16_t get_datagram_tag(uint8_t *ptr)
225 {
226 	return (ptr[0] << 8) | ptr[1];
227 }
228 
update_protocol_header_lengths(struct net_pkt * pkt,uint16_t size)229 static void update_protocol_header_lengths(struct net_pkt *pkt, uint16_t size)
230 {
231 	NET_PKT_DATA_ACCESS_DEFINE(ipv6_access, struct net_ipv6_hdr);
232 	struct net_ipv6_hdr *ipv6;
233 
234 	ipv6 = (struct net_ipv6_hdr *)net_pkt_get_data(pkt, &ipv6_access);
235 	if (!ipv6) {
236 		NET_ERR("Could not get IPv6 header");
237 		return;
238 	}
239 
240 	net_pkt_set_ip_hdr_len(pkt, NET_IPV6H_LEN);
241 	ipv6->len = htons(size - NET_IPV6H_LEN);
242 
243 	net_pkt_set_data(pkt, &ipv6_access);
244 
245 	if (ipv6->nexthdr == IPPROTO_UDP) {
246 		NET_PKT_DATA_ACCESS_DEFINE(udp_access, struct net_udp_hdr);
247 		struct net_udp_hdr *udp;
248 
249 		udp = (struct net_udp_hdr *)net_pkt_get_data(pkt, &udp_access);
250 		if (udp) {
251 			udp->len = htons(size - NET_IPV6H_LEN);
252 			net_pkt_set_data(pkt, &udp_access);
253 		} else {
254 			NET_ERR("Could not get UDP header");
255 		}
256 	}
257 }
258 
clear_reass_cache(uint16_t size,uint16_t tag)259 static inline void clear_reass_cache(uint16_t size, uint16_t tag)
260 {
261 	uint8_t i;
262 
263 	for (i = 0U; i < REASS_CACHE_SIZE; i++) {
264 		if (!(cache[i].size == size && cache[i].tag == tag)) {
265 			continue;
266 		}
267 
268 		if (cache[i].pkt) {
269 			net_pkt_unref(cache[i].pkt);
270 		}
271 
272 		cache[i].pkt = NULL;
273 		cache[i].size = 0U;
274 		cache[i].tag = 0U;
275 		cache[i].used = false;
276 		k_work_cancel_delayable(&cache[i].timer);
277 	}
278 }
279 
280 /**
281  *  If the reassembly not completed within reassembly timeout discard
282  *  the whole packet.
283  */
reass_timeout(struct k_work * work)284 static void reass_timeout(struct k_work *work)
285 {
286 	struct k_work_delayable *dwork = k_work_delayable_from_work(work);
287 	struct frag_cache *fcache = CONTAINER_OF(dwork, struct frag_cache, timer);
288 
289 	if (fcache->pkt) {
290 		net_pkt_unref(fcache->pkt);
291 	}
292 
293 	fcache->pkt = NULL;
294 	fcache->size = 0U;
295 	fcache->tag = 0U;
296 	fcache->used = false;
297 }
298 
299 /**
300  *  Upon reception of first fragment with respective of size and tag
301  *  create a new cache. If number of unused cache are out then
302  *  discard the fragments.
303  */
set_reass_cache(struct net_pkt * pkt,uint16_t size,uint16_t tag)304 static inline struct frag_cache *set_reass_cache(struct net_pkt *pkt, uint16_t size, uint16_t tag)
305 {
306 	int i;
307 
308 	for (i = 0; i < REASS_CACHE_SIZE; i++) {
309 		if (cache[i].used) {
310 			continue;
311 		}
312 
313 		cache[i].pkt = pkt;
314 		cache[i].size = size;
315 		cache[i].tag = tag;
316 		cache[i].used = true;
317 
318 		k_work_init_delayable(&cache[i].timer, reass_timeout);
319 		k_work_reschedule(&cache[i].timer, FRAG_REASSEMBLY_TIMEOUT);
320 		return &cache[i];
321 	}
322 
323 	return NULL;
324 }
325 
326 /**
327  *  Return cache if it matches with size and tag of stored caches,
328  *  otherwise return NULL.
329  */
get_reass_cache(uint16_t size,uint16_t tag)330 static inline struct frag_cache *get_reass_cache(uint16_t size, uint16_t tag)
331 {
332 	uint8_t i;
333 
334 	for (i = 0U; i < REASS_CACHE_SIZE; i++) {
335 		if (cache[i].used) {
336 			if (cache[i].size == size && cache[i].tag == tag) {
337 				return &cache[i];
338 			}
339 		}
340 	}
341 
342 	return NULL;
343 }
344 
fragment_append(struct net_pkt * pkt,struct net_buf * frag)345 static inline void fragment_append(struct net_pkt *pkt, struct net_buf *frag)
346 {
347 	if (get_datagram_type(frag->data) == NET_6LO_DISPATCH_FRAG1) {
348 		/* Always make sure first fragment is inserted first
349 		 * This will be useful for fragment_cached_pkt_len()
350 		 */
351 		frag->frags = pkt->buffer;
352 		pkt->buffer = frag;
353 	} else {
354 		net_pkt_append_buffer(pkt, frag);
355 	}
356 }
357 
fragment_cached_pkt_len(struct net_pkt * pkt)358 static inline size_t fragment_cached_pkt_len(struct net_pkt *pkt)
359 {
360 	size_t len = 0U;
361 	struct net_buf *frag;
362 	int hdr_diff;
363 	uint8_t *data;
364 
365 	frag = pkt->buffer;
366 	while (frag) {
367 		uint16_t frag_hdr_len = NET_6LO_FRAGN_HDR_LEN;
368 
369 		if (get_datagram_type(frag->data) == NET_6LO_DISPATCH_FRAG1) {
370 			frag_hdr_len = NET_6LO_FRAG1_HDR_LEN;
371 		}
372 
373 		len += frag->len - frag_hdr_len;
374 
375 		frag = frag->frags;
376 	}
377 
378 	/* 6lo assumes that fragment header has been removed,
379 	 * and in our side we assume first buffer is always the first fragment.
380 	 */
381 	data = pkt->buffer->data;
382 	pkt->buffer->data += NET_6LO_FRAG1_HDR_LEN;
383 
384 	hdr_diff = net_6lo_uncompress_hdr_diff(pkt);
385 
386 	pkt->buffer->data = data;
387 
388 	if (hdr_diff == INT_MAX) {
389 		return 0;
390 	}
391 
392 	return len + hdr_diff;
393 }
394 
fragment_offset(struct net_buf * frag)395 static inline uint16_t fragment_offset(struct net_buf *frag)
396 {
397 	if (get_datagram_type(frag->data) == NET_6LO_DISPATCH_FRAG1) {
398 		return 0;
399 	}
400 
401 	return ((uint16_t)frag->data[NET_FRAG_OFFSET_POS] << 3);
402 }
403 
fragment_move_back(struct net_pkt * pkt,struct net_buf * frag,struct net_buf * stop)404 static void fragment_move_back(struct net_pkt *pkt, struct net_buf *frag, struct net_buf *stop)
405 {
406 	struct net_buf *prev, *current;
407 
408 	prev = NULL;
409 	current = pkt->buffer;
410 
411 	while (current && current != stop) {
412 		if (fragment_offset(frag) < fragment_offset(current)) {
413 			if (prev) {
414 				prev->frags = frag;
415 			}
416 
417 			frag->frags = current;
418 			break;
419 		}
420 
421 		prev = current;
422 		current = current->frags;
423 	}
424 }
425 
fragment_remove_headers(struct net_pkt * pkt)426 static inline void fragment_remove_headers(struct net_pkt *pkt)
427 {
428 	struct net_buf *frag;
429 
430 	frag = pkt->buffer;
431 	while (frag) {
432 		uint16_t frag_hdr_len = NET_6LO_FRAGN_HDR_LEN;
433 
434 		if (get_datagram_type(frag->data) == NET_6LO_DISPATCH_FRAG1) {
435 			frag_hdr_len = NET_6LO_FRAG1_HDR_LEN;
436 		}
437 
438 		memmove(frag->data, frag->data + frag_hdr_len, frag->len - frag_hdr_len);
439 		frag->len -= frag_hdr_len;
440 
441 		frag = frag->frags;
442 	}
443 }
444 
fragment_reconstruct_packet(struct net_pkt * pkt)445 static inline void fragment_reconstruct_packet(struct net_pkt *pkt)
446 {
447 	struct net_buf *prev, *current, *next;
448 
449 	prev = NULL;
450 	current = pkt->buffer;
451 
452 	while (current) {
453 		next = current->frags;
454 
455 		if (!prev || (fragment_offset(prev) > fragment_offset(current))) {
456 			prev = current;
457 		} else {
458 			fragment_move_back(pkt, current, prev);
459 		}
460 
461 		current = next;
462 	}
463 
464 	/* Let's remove now useless fragmentation headers */
465 	fragment_remove_headers(pkt);
466 }
467 
fragment_packet_valid(struct net_pkt * pkt)468 static inline bool fragment_packet_valid(struct net_pkt *pkt)
469 {
470 	return (get_datagram_type(pkt->buffer->data) == NET_6LO_DISPATCH_FRAG1);
471 }
472 
473 /**
474  *  Parse size and tag from the fragment, check if we have any cache
475  *  related to it. If not create a new cache.
476  *  Remove the fragmentation header and uncompress IPv6 and related headers.
477  *  Cache Rx part of fragment along with data buf for the first fragment
478  *  in the cache, remaining fragments just cache data fragment, unref
479  *  RX pkt. So in both the cases caller can assume packet is consumed.
480  */
fragment_add_to_cache(struct net_pkt * pkt)481 static inline enum net_verdict fragment_add_to_cache(struct net_pkt *pkt)
482 {
483 	bool first_frag = false;
484 	struct frag_cache *fcache;
485 	struct net_buf *frag;
486 	uint16_t size;
487 	uint16_t tag;
488 	uint8_t type;
489 
490 	frag = pkt->buffer;
491 	type = get_datagram_type(frag->data);
492 
493 	if ((type == NET_6LO_DISPATCH_FRAG1 && frag->len < NET_6LO_FRAG1_HDR_LEN) ||
494 	    (type == NET_6LO_DISPATCH_FRAGN && frag->len < NET_6LO_FRAGN_HDR_LEN)) {
495 		NET_ERR("Fragment too short (%u): fragment dropped", frag->len);
496 		return NET_DROP;
497 	}
498 
499 	/* Parse total size of packet */
500 	size = get_datagram_size(frag->data);
501 
502 	/* Parse the datagram tag */
503 	tag = get_datagram_tag(frag->data + NET_6LO_FRAG_DATAGRAM_SIZE_LEN);
504 
505 	/* If there are no fragments in the cache means this frag
506 	 * is the first one. So cache Rx pkt otherwise not.
507 	 */
508 	pkt->buffer = NULL;
509 
510 	fcache = get_reass_cache(size, tag);
511 	if (!fcache) {
512 		fcache = set_reass_cache(pkt, size, tag);
513 		if (!fcache) {
514 			NET_ERR("Could not allocate fragment cache, consider increasing "
515 				"CONFIG_NET_L2_IEEE802154_FRAGMENT_REASS_CACHE_SIZE: packet "
516 				"dropped");
517 			pkt->buffer = frag;
518 			return NET_DROP;
519 		}
520 
521 		first_frag = true;
522 	}
523 
524 	fragment_append(fcache->pkt, frag);
525 
526 	if (fragment_cached_pkt_len(fcache->pkt) == fcache->size) {
527 		/* All fragments received - reassemble packet. */
528 
529 		if (!first_frag) {
530 			/* Assign buffer back to input packet. */
531 			pkt->buffer = fcache->pkt->buffer;
532 			fcache->pkt->buffer = NULL;
533 		} else {
534 			/* In case pkt == fcache->pkt, we don't want
535 			 * to unref it while clearing the cache.
536 			 */
537 			fcache->pkt = NULL;
538 		}
539 
540 		clear_reass_cache(size, tag);
541 
542 		if (!fragment_packet_valid(pkt)) {
543 			NET_ERR("Invalid fragment type: packet dropped");
544 			return NET_DROP;
545 		}
546 
547 		fragment_reconstruct_packet(pkt);
548 
549 		if (!net_6lo_uncompress(pkt)) {
550 			NET_ERR("Invalid 6LoWPAN header: packet dropped");
551 			return NET_DROP;
552 		}
553 
554 		net_pkt_cursor_init(pkt);
555 
556 		update_protocol_header_lengths(pkt, size);
557 
558 		net_pkt_cursor_init(pkt);
559 
560 		NET_DBG("All fragments received and reassembled");
561 
562 		return NET_CONTINUE;
563 	}
564 
565 	/* Unref Rx part of original packet */
566 	if (!first_frag) {
567 		net_pkt_unref(pkt);
568 	}
569 
570 	return NET_OK;
571 }
572 
ieee802154_6lo_reassemble(struct net_pkt * pkt)573 enum net_verdict ieee802154_6lo_reassemble(struct net_pkt *pkt)
574 {
575 	if (!pkt || !pkt->buffer) {
576 		NET_WARN("Empty payload: packet dropped");
577 		return NET_DROP;
578 	}
579 
580 	if (get_datagram_type(pkt->buffer->data) >= NET_6LO_DISPATCH_FRAG1) {
581 		return fragment_add_to_cache(pkt);
582 	} else {
583 		NET_DBG("No frag dispatch (%02x)", pkt->buffer->data[0]);
584 		/* Received unfragmented packet, uncompress */
585 		if (net_6lo_uncompress(pkt)) {
586 			return NET_CONTINUE;
587 		}
588 
589 		NET_ERR("Invalid header: packet dropped");
590 	}
591 
592 	return NET_DROP;
593 }
594