From 6d72cdb5956ddeadc584bd8b108a83e830b5ee07 Mon Sep 17 00:00:00 2001 From: Martine Lenders Date: Wed, 15 Jun 2016 15:07:09 +0200 Subject: [PATCH 1/2] gnrc_ipv6: copy user flags from old netif headers Upper layers might want to utilize the flags (e.g. to tell 6LoWPAN to elide UDP checksums). This change allows for this by copying non-addressing related flags to the address resolution generated netif header from the user generated netif header. --- sys/net/gnrc/network_layer/ipv6/gnrc_ipv6.c | 75 ++++++++++----------- 1 file changed, 34 insertions(+), 41 deletions(-) diff --git a/sys/net/gnrc/network_layer/ipv6/gnrc_ipv6.c b/sys/net/gnrc/network_layer/ipv6/gnrc_ipv6.c index cccecba502..4e0ae63d37 100644 --- a/sys/net/gnrc/network_layer/ipv6/gnrc_ipv6.c +++ b/sys/net/gnrc/network_layer/ipv6/gnrc_ipv6.c @@ -393,34 +393,47 @@ static void _send_to_iface(kernel_pid_t iface, gnrc_pktsnip_t *pkt) } } -/* functions for sending */ -static void _send_unicast(kernel_pid_t iface, uint8_t *dst_l2addr, - uint16_t dst_l2addr_len, gnrc_pktsnip_t *pkt) +static gnrc_pktsnip_t *_create_netif_hdr(uint8_t *dst_l2addr, + uint16_t dst_l2addr_len, + gnrc_pktsnip_t *pkt) { - gnrc_pktsnip_t *netif; - - if (pkt->type == GNRC_NETTYPE_NETIF) { - /* great: someone already added a netif_hdr_t we assume it's wrong - * to keep it simple - * XXX: alternative would be to check if gnrc_netif_hdr_t::dst_l2addr_len - * is long enough and only then to throw away the header. This causes - * to much overhead IMHO */ - DEBUG("ipv6: removed old interface header\n"); - pkt = gnrc_pktbuf_remove_snip(pkt, pkt); - } - - DEBUG("ipv6: add interface header to packet\n"); - netif = gnrc_netif_hdr_build(NULL, 0, dst_l2addr, dst_l2addr_len); + gnrc_pktsnip_t *netif = gnrc_netif_hdr_build(NULL, 0, dst_l2addr, dst_l2addr_len); if (netif == NULL) { DEBUG("ipv6: error on interface header allocation, dropping packet\n"); gnrc_pktbuf_release(pkt); - return; + return NULL; + } + + if (pkt->type == GNRC_NETTYPE_NETIF) { + /* remove old netif header, since checking it for correctness would + * cause to much overhead. + * netif header might have been allocated by some higher layer either + * to set a sending interface or some flags. Interface was already + * copied using iface parameter, so we only need to copy the flags + * (minus the broadcast/multicast flags) */ + DEBUG("ipv6: copy old interface header flags\n"); + gnrc_netif_hdr_t *netif_new = netif->data, *netif_old = pkt->data; + netif_new->flags = netif_old->flags & \ + ~(GNRC_NETIF_HDR_FLAGS_BROADCAST | GNRC_NETIF_HDR_FLAGS_MULTICAST); + DEBUG("ipv6: removed old interface header\n"); + pkt = gnrc_pktbuf_remove_snip(pkt, pkt); } /* add netif to front of the pkt list */ LL_PREPEND(pkt, netif); + return pkt; +} + +/* functions for sending */ +static void _send_unicast(kernel_pid_t iface, uint8_t *dst_l2addr, + uint16_t dst_l2addr_len, gnrc_pktsnip_t *pkt) +{ + DEBUG("ipv6: add interface header to packet\n"); + if ((pkt = _create_netif_hdr(dst_l2addr, dst_l2addr_len, pkt)) == NULL) { + return; + } DEBUG("ipv6: send unicast over interface %" PRIkernel_pid "\n", iface); /* and send to interface */ #ifdef MODULE_NETSTATS_IPV6 @@ -521,14 +534,12 @@ static void _send_multicast(kernel_pid_t iface, gnrc_pktsnip_t *pkt, #if GNRC_NETIF_NUMOF > 1 - /* netif header not present: send over all interfaces */ + /* interface not given: send over all interfaces */ if (iface == KERNEL_PID_UNDEF) { - assert(pkt == ipv6); /* send packet to link layer */ gnrc_pktbuf_hold(pkt, ifnum - 1); for (size_t i = 0; i < ifnum; i++) { - gnrc_pktsnip_t *netif; if (prep_hdr) { /* need to get second write access (duplication) to fill IPv6 * header interface-local */ @@ -564,24 +575,14 @@ static void _send_multicast(kernel_pid_t iface, gnrc_pktsnip_t *pkt, } } - /* allocate interface header */ - netif = gnrc_netif_hdr_build(NULL, 0, NULL, 0); - - if (netif == NULL) { - DEBUG("ipv6: error on interface header allocation, " - "dropping packet\n"); - gnrc_pktbuf_release(ipv6); + if ((ipv6 = _create_netif_hdr(NULL, 0, ipv6)) == NULL) { return; } - LL_PREPEND(ipv6, netif); - _send_multicast_over_iface(ifs[i], ipv6); } } else { - /* iface != KERNEL_PID_UNDEF implies that netif header is present */ - assert(pkt != ipv6); if (prep_hdr) { if (_fill_ipv6_hdr(iface, ipv6, payload) < 0) { /* error on filling up header */ @@ -595,20 +596,12 @@ static void _send_multicast(kernel_pid_t iface, gnrc_pktsnip_t *pkt, #else /* GNRC_NETIF_NUMOF */ (void)ifnum; /* not used in this build branch */ if (iface == KERNEL_PID_UNDEF) { - gnrc_pktsnip_t *netif; iface = ifs[0]; /* allocate interface header */ - netif = gnrc_netif_hdr_build(NULL, 0, NULL, 0); - - if (netif == NULL) { - DEBUG("ipv6: error on interface header allocation, " - "dropping packet\n"); - gnrc_pktbuf_release(pkt); + if ((pkt = _create_netif_hdr(NULL, 0, pkt)) == NULL) { return; } - - LL_PREPEND(pkt, netif); } if (prep_hdr) { From f844008a12102651c04bfe86d674c00991a23985 Mon Sep 17 00:00:00 2001 From: Martine Lenders Date: Mon, 4 Jul 2016 16:34:04 +0200 Subject: [PATCH 2/2] gnrc_netif_hdr: print flags --- sys/net/gnrc/netif/hdr/gnrc_netif_hdr_print.c | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/sys/net/gnrc/netif/hdr/gnrc_netif_hdr_print.c b/sys/net/gnrc/netif/hdr/gnrc_netif_hdr_print.c index 75c8e90d52..fb56102037 100644 --- a/sys/net/gnrc/netif/hdr/gnrc_netif_hdr_print.c +++ b/sys/net/gnrc/netif/hdr/gnrc_netif_hdr_print.c @@ -25,6 +25,22 @@ void gnrc_netif_hdr_print(gnrc_netif_hdr_t *hdr) printf("if_pid: %" PRIkernel_pid " ", hdr->if_pid); printf("rssi: %" PRIu8 " ", hdr->rssi); printf("lqi: %" PRIu8 "\n", hdr->lqi); + printf("flags: "); + + if (hdr->flags) { + if (hdr->flags & GNRC_NETIF_HDR_FLAGS_BROADCAST) { + printf("BROADCAST "); + } + + if (hdr->flags & GNRC_NETIF_HDR_FLAGS_MULTICAST) { + printf("MULTICAST "); + } + puts(""); + } + else { + puts("0x0"); + } + if (hdr->src_l2addr_len > 0) { printf("src_l2addr: %s\n",