/* * Copyright (C) 2015 Martine Lenders * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ /** * @{ * * @file */ #include #include #include "rbuf.h" #include "net/ipv6/hdr.h" #include "net/gnrc.h" #include "net/gnrc/ipv6/netif.h" #include "net/gnrc/sixlowpan.h" #include "net/gnrc/sixlowpan/frag.h" #include "net/sixlowpan.h" #include "thread.h" #include "timex.h" #include "xtimer.h" #include "utlist.h" #define ENABLE_DEBUG (0) #include "debug.h" #ifndef RBUF_INT_SIZE #define RBUF_INT_SIZE (GNRC_IPV6_NETIF_DEFAULT_MTU * RBUF_SIZE / 127) #endif static rbuf_int_t rbuf_int[RBUF_INT_SIZE]; static rbuf_t rbuf[RBUF_SIZE]; #if ENABLE_DEBUG static char l2addr_str[3 * RBUF_L2ADDR_MAX_LEN]; #endif /* ------------------------------------ * internal function definitions * ------------------------------------*/ /* checks whether start and end are in given interval i */ static inline bool _rbuf_int_in(rbuf_int_t *i, uint16_t start, uint16_t end); /* gets a free entry from interval buffer */ static rbuf_int_t *_rbuf_int_get_free(void); /* remove entry from reassembly buffer */ static void _rbuf_rem(rbuf_t *entry); /* update interval buffer of entry */ static bool _rbuf_update_ints(rbuf_t *entry, uint16_t offset, size_t frag_size); /* checks timeouts and removes entries if necessary (oldest if full) */ static void _rbuf_gc(void); /* gets an entry identified by its tupel */ static rbuf_t *_rbuf_get(const void *src, size_t src_len, const void *dst, size_t dst_len, size_t size, uint16_t tag); void rbuf_add(gnrc_netif_hdr_t *netif_hdr, gnrc_pktsnip_t *pkt, size_t frag_size, size_t offset) { rbuf_t *entry; /* cppcheck is clearly wrong here */ /* cppcheck-suppress variableScope */ unsigned int data_offset = 0; sixlowpan_frag_t *frag = pkt->data; rbuf_int_t *ptr; uint8_t *data = ((uint8_t *)pkt->data) + sizeof(sixlowpan_frag_t); _rbuf_gc(); entry = _rbuf_get(gnrc_netif_hdr_get_src_addr(netif_hdr), netif_hdr->src_l2addr_len, gnrc_netif_hdr_get_dst_addr(netif_hdr), netif_hdr->dst_l2addr_len, byteorder_ntohs(frag->disp_size) & SIXLOWPAN_FRAG_SIZE_MASK, byteorder_ntohs(frag->tag)); if (entry == NULL) { DEBUG("6lo rbuf: reassembly buffer full.\n"); return; } ptr = entry->ints; /* dispatches in the first fragment are ignored */ if (offset == 0) { if (data[0] == SIXLOWPAN_UNCOMP) { data++; /* skip 6LoWPAN dispatch */ frag_size--; } #ifdef MODULE_GNRC_SIXLOWPAN_IPHC else if (sixlowpan_iphc_is(data)) { size_t iphc_len; iphc_len = gnrc_sixlowpan_iphc_decode(entry->pkt, pkt, entry->pkt->size, sizeof(sixlowpan_frag_t)); if (iphc_len == 0) { DEBUG("6lo rfrag: could not decode IPHC dispatch\n"); gnrc_pktbuf_release(entry->pkt); _rbuf_rem(entry); return; } data += iphc_len; /* take remaining data as data */ frag_size -= iphc_len; /* and reduce frag size by IPHC dispatch length */ frag_size += sizeof(ipv6_hdr_t); /* but add IPv6 header length */ data_offset += sizeof(ipv6_hdr_t); /* start copying after IPv6 header */ } #endif } else { data++; /* FRAGN header is one byte longer (offset) */ } if ((offset + frag_size) > entry->pkt->size) { DEBUG("6lo rfrag: fragment too big for resulting datagram, discarding datagram\n"); gnrc_pktbuf_release(entry->pkt); _rbuf_rem(entry); return; } while (ptr != NULL) { if (_rbuf_int_in(ptr, offset, offset + frag_size - 1)) { DEBUG("6lo rfrag: overlapping or same intervals, discarding datagram\n"); gnrc_pktbuf_release(entry->pkt); _rbuf_rem(entry); return; } ptr = ptr->next; } if (_rbuf_update_ints(entry, offset, frag_size)) { DEBUG("6lo rbuf: add fragment data\n"); entry->cur_size += (uint16_t)frag_size; memcpy(((uint8_t *)entry->pkt->data) + offset + data_offset, data, frag_size - data_offset); } if (entry->cur_size == entry->pkt->size) { gnrc_pktsnip_t *netif = gnrc_netif_hdr_build(entry->src, entry->src_len, entry->dst, entry->dst_len); if (netif == NULL) { DEBUG("6lo rbuf: error allocating netif header\n"); gnrc_pktbuf_release(entry->pkt); _rbuf_rem(entry); return; } /* copy the transmit information of the latest fragment into the newly * created header to have some link_layer information. The link_layer * info of the previous fragments is discarded. */ gnrc_netif_hdr_t *new_netif_hdr = netif->data; new_netif_hdr->if_pid = netif_hdr->if_pid; new_netif_hdr->flags = netif_hdr->flags; new_netif_hdr->lqi = netif_hdr->lqi; new_netif_hdr->rssi = netif_hdr->rssi; LL_APPEND(entry->pkt, netif); if (!gnrc_netapi_dispatch_receive(GNRC_NETTYPE_IPV6, GNRC_NETREG_DEMUX_CTX_ALL, entry->pkt)) { DEBUG("6lo rbuf: No receivers for this packet found\n"); gnrc_pktbuf_release(entry->pkt); } _rbuf_rem(entry); } } static inline bool _rbuf_int_in(rbuf_int_t *i, uint16_t start, uint16_t end) { return (((i->start < start) && (start <= i->end)) || ((start < i->start) && (i->start <= end)) || ((i->start == start) && (i->end == end))); } static rbuf_int_t *_rbuf_int_get_free(void) { for (unsigned int i = 0; i < RBUF_INT_SIZE; i++) { if (rbuf_int[i].end == 0) { /* start must be smaller than end anyways*/ return rbuf_int + i; } } return NULL; } static void _rbuf_rem(rbuf_t *entry) { while (entry->ints != NULL) { rbuf_int_t *next = entry->ints->next; entry->ints->start = 0; entry->ints->end = 0; entry->ints->next = NULL; entry->ints = next; } entry->pkt = NULL; } static bool _rbuf_update_ints(rbuf_t *entry, uint16_t offset, size_t frag_size) { rbuf_int_t *new; uint16_t end = (uint16_t)(offset + frag_size - 1); new = _rbuf_int_get_free(); if (new == NULL) { DEBUG("6lo rfrag: no space left in rbuf interval buffer.\n"); return false; } new->start = offset; new->end = end; DEBUG("6lo rfrag: add interval (%" PRIu16 ", %" PRIu16 ") to entry (%s, ", new->start, new->end, gnrc_netif_addr_to_str(l2addr_str, sizeof(l2addr_str), entry->src, entry->src_len)); DEBUG("%s, %u, %u)\n", gnrc_netif_addr_to_str(l2addr_str, sizeof(l2addr_str), entry->dst, entry->dst_len), (unsigned)entry->pkt->size, entry->tag); LL_PREPEND(entry->ints, new); return true; } static void _rbuf_gc(void) { rbuf_t *oldest = NULL; uint32_t now_sec = xtimer_now() / SEC_IN_USEC; unsigned int i; for (i = 0; i < RBUF_SIZE; i++) { if (rbuf[i].pkt == NULL) { /* leave GC early if there is still room */ return; } else if ((rbuf[i].pkt != NULL) && ((now_sec - rbuf[i].arrival) > RBUF_TIMEOUT)) { DEBUG("6lo rfrag: entry (%s, ", gnrc_netif_addr_to_str(l2addr_str, sizeof(l2addr_str), rbuf[i].src, rbuf[i].src_len)); DEBUG("%s, %u, %u) timed out\n", gnrc_netif_addr_to_str(l2addr_str, sizeof(l2addr_str), rbuf[i].dst, rbuf[i].dst_len), (unsigned)rbuf[i].pkt->size, rbuf[i].tag); gnrc_pktbuf_release(rbuf[i].pkt); _rbuf_rem(&(rbuf[i])); } else if ((oldest == NULL) || (rbuf[i].arrival < oldest->arrival)) { oldest = &(rbuf[i]); } } if ((i >= RBUF_SIZE) && (oldest != NULL) && (oldest->pkt != NULL)) { DEBUG("6lo rfrag: reassembly buffer full, remove oldest entry\n"); gnrc_pktbuf_release(oldest->pkt); _rbuf_rem(oldest); } } static rbuf_t *_rbuf_get(const void *src, size_t src_len, const void *dst, size_t dst_len, size_t size, uint16_t tag) { rbuf_t *res = NULL; uint32_t now_sec = xtimer_now() / SEC_IN_USEC; for (unsigned int i = 0; i < RBUF_SIZE; i++) { /* check first if entry already available */ if ((rbuf[i].pkt != NULL) && (rbuf[i].pkt->size == size) && (rbuf[i].tag == tag) && (rbuf[i].src_len == src_len) && (rbuf[i].dst_len == dst_len) && (memcmp(rbuf[i].src, src, src_len) == 0) && (memcmp(rbuf[i].dst, dst, dst_len) == 0)) { DEBUG("6lo rfrag: entry %p (%s, ", (void *)(&rbuf[i]), gnrc_netif_addr_to_str(l2addr_str, sizeof(l2addr_str), rbuf[i].src, rbuf[i].src_len)); DEBUG("%s, %u, %u) found\n", gnrc_netif_addr_to_str(l2addr_str, sizeof(l2addr_str), rbuf[i].dst, rbuf[i].dst_len), (unsigned)rbuf[i].pkt->size, rbuf[i].tag); rbuf[i].arrival = now_sec; return &(rbuf[i]); } /* if there is a free spot: remember it */ if ((res == NULL) && (rbuf[i].pkt == NULL)) { res = &(rbuf[i]); } } if (res != NULL) { /* entry not in buffer but found empty spot */ res->pkt = gnrc_pktbuf_add(NULL, NULL, size, GNRC_NETTYPE_SIXLOWPAN); if (res->pkt == NULL) { DEBUG("6lo rfrag: can not allocate reassembly buffer space.\n"); return NULL; } *((uint64_t *)res->pkt->data) = 0; /* clean first few bytes for later * look-ups */ res->arrival = now_sec; memcpy(res->src, src, src_len); memcpy(res->dst, dst, dst_len); res->src_len = src_len; res->dst_len = dst_len; res->tag = tag; res->cur_size = 0; DEBUG("6lo rfrag: entry %p (%s, ", (void *)res, gnrc_netif_addr_to_str(l2addr_str, sizeof(l2addr_str), res->src, res->src_len)); DEBUG("%s, %u, %u) created\n", gnrc_netif_addr_to_str(l2addr_str, sizeof(l2addr_str), res->dst, res->dst_len), (unsigned)res->pkt->size, res->tag); } return res; } /** @} */