2015-04-14 00:18:36 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 Martine Lenders <mlenders@inf.fu-berlin.de>
|
|
|
|
*
|
|
|
|
* 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 <stdio.h>
|
2015-07-06 17:49:07 +02:00
|
|
|
#include <inttypes.h>
|
2015-04-14 00:18:36 +02:00
|
|
|
|
|
|
|
#include "od.h"
|
2015-08-09 23:53:40 +02:00
|
|
|
#include "net/ipv6/hdr.h"
|
2015-08-17 15:06:44 +02:00
|
|
|
#include "net/sixlowpan.h"
|
2015-04-14 00:18:36 +02:00
|
|
|
|
2015-08-17 15:06:44 +02:00
|
|
|
void sixlowpan_print(uint8_t *data, size_t size)
|
2015-04-14 00:18:36 +02:00
|
|
|
{
|
2015-08-17 15:06:44 +02:00
|
|
|
if (data[0] == SIXLOWPAN_UNCOMP) {
|
2015-04-14 00:18:36 +02:00
|
|
|
puts("Uncompressed IPv6 packet");
|
|
|
|
|
|
|
|
/* might just be the dispatch (or fragmented) so better check */
|
2015-08-09 23:53:40 +02:00
|
|
|
if (size > sizeof(ipv6_hdr_t)) {
|
|
|
|
ipv6_hdr_print((ipv6_hdr_t *)(data + 1));
|
|
|
|
od_hex_dump(data + sizeof(ipv6_hdr_t) + 1,
|
|
|
|
size - sizeof(ipv6_hdr_t) - 1,
|
2015-04-14 00:18:36 +02:00
|
|
|
OD_WIDTH_DEFAULT);
|
|
|
|
}
|
|
|
|
}
|
2015-08-17 15:06:44 +02:00
|
|
|
else if (sixlowpan_nalp(data[0])) {
|
2015-04-14 00:18:36 +02:00
|
|
|
puts("Not a LoWPAN (NALP) frame");
|
|
|
|
od_hex_dump(data, size, OD_WIDTH_DEFAULT);
|
|
|
|
}
|
2015-08-17 15:06:44 +02:00
|
|
|
else if ((data[0] & SIXLOWPAN_FRAG_DISP_MASK) == SIXLOWPAN_FRAG_1_DISP) {
|
|
|
|
sixlowpan_frag_t *hdr = (sixlowpan_frag_t *)data;
|
2015-04-14 00:18:36 +02:00
|
|
|
|
|
|
|
puts("Fragmentation Header (first)");
|
|
|
|
printf("datagram size: %" PRIu16 "\n",
|
2015-08-17 15:06:44 +02:00
|
|
|
byteorder_ntohs(hdr->disp_size) & SIXLOWPAN_FRAG_SIZE_MASK);
|
2015-04-14 00:18:36 +02:00
|
|
|
printf("tag: 0x%" PRIu16 "\n", byteorder_ntohs(hdr->tag));
|
|
|
|
|
|
|
|
/* Print next dispatch */
|
2015-08-17 15:06:44 +02:00
|
|
|
sixlowpan_print(data + sizeof(sixlowpan_frag_t),
|
|
|
|
size - sizeof(sixlowpan_frag_t));
|
2015-04-14 00:18:36 +02:00
|
|
|
}
|
2015-08-17 15:06:44 +02:00
|
|
|
else if ((data[0] & SIXLOWPAN_FRAG_DISP_MASK) == SIXLOWPAN_FRAG_N_DISP) {
|
|
|
|
sixlowpan_frag_n_t *hdr = (sixlowpan_frag_n_t *)data;
|
2015-04-14 00:18:36 +02:00
|
|
|
|
|
|
|
puts("Fragmentation Header (subsequent)");
|
|
|
|
printf("datagram size: %" PRIu16 "\n",
|
2015-08-17 15:06:44 +02:00
|
|
|
byteorder_ntohs(hdr->disp_size) & SIXLOWPAN_FRAG_SIZE_MASK);
|
2015-04-14 00:18:36 +02:00
|
|
|
printf("tag: 0x%" PRIu16 "\n", byteorder_ntohs(hdr->tag));
|
|
|
|
printf("offset: 0x%" PRIu8 "\n", hdr->offset);
|
|
|
|
|
2015-08-17 15:06:44 +02:00
|
|
|
od_hex_dump(data + sizeof(sixlowpan_frag_n_t),
|
|
|
|
size - sizeof(sixlowpan_frag_n_t),
|
2015-04-14 00:18:36 +02:00
|
|
|
OD_WIDTH_DEFAULT);
|
|
|
|
}
|
2015-08-17 15:06:44 +02:00
|
|
|
else if ((data[0] & SIXLOWPAN_IPHC1_DISP_MASK) == SIXLOWPAN_IPHC1_DISP) {
|
|
|
|
uint8_t offset = SIXLOWPAN_IPHC_HDR_LEN;
|
2015-04-14 00:18:36 +02:00
|
|
|
puts("IPHC dispatch");
|
|
|
|
|
2015-08-17 15:06:44 +02:00
|
|
|
switch (data[0] & SIXLOWPAN_IPHC1_TF) {
|
2015-04-14 00:18:36 +02:00
|
|
|
case 0x00:
|
|
|
|
puts("TF: ECN + DSCP + Flow Label (4 bytes)");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x08:
|
|
|
|
puts("TF: ECN + Flow Label (3 bytes)");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x10:
|
|
|
|
puts("TF: ECN + DSCP (1 bytes)");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x18:
|
|
|
|
puts("TF: traffic class and flow label elided");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:06:44 +02:00
|
|
|
switch (data[0] & SIXLOWPAN_IPHC1_NH) {
|
2015-04-14 00:18:36 +02:00
|
|
|
case 0x00:
|
|
|
|
puts("NH: inline");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x04:
|
|
|
|
puts("NH: LOWPAN_NHC");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:06:44 +02:00
|
|
|
switch (data[0] & SIXLOWPAN_IPHC1_HL) {
|
2015-04-14 00:18:36 +02:00
|
|
|
case 0x00:
|
|
|
|
puts("HLIM: inline");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x01:
|
|
|
|
puts("HLIM: 1");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x02:
|
|
|
|
puts("HLIM: 64");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x03:
|
|
|
|
puts("HLIM: 255");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:06:44 +02:00
|
|
|
if (data[1] & SIXLOWPAN_IPHC2_SAC) {
|
2015-04-14 00:18:36 +02:00
|
|
|
printf("Stateful source address compression: ");
|
|
|
|
|
2015-08-17 15:06:44 +02:00
|
|
|
switch (data[1] & SIXLOWPAN_IPHC2_SAM) {
|
2015-04-14 00:18:36 +02:00
|
|
|
case 0x00:
|
|
|
|
puts("unspecified address (::)");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x10:
|
|
|
|
puts("64 bits inline");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x20:
|
|
|
|
puts("16 bits inline");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x40:
|
|
|
|
puts("elided (use L2 address)");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
printf("Stateless source address compression: ");
|
|
|
|
|
2015-08-17 15:06:44 +02:00
|
|
|
switch (data[1] & SIXLOWPAN_IPHC2_SAM) {
|
2015-04-14 00:18:36 +02:00
|
|
|
case 0x00:
|
|
|
|
puts("128 bits inline");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x10:
|
|
|
|
puts("64 bits inline");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x20:
|
|
|
|
puts("16 bits inline");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x40:
|
|
|
|
puts("elided (use L2 address)");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:06:44 +02:00
|
|
|
if (data[1] & SIXLOWPAN_IPHC2_M) {
|
|
|
|
if (data[1] & SIXLOWPAN_IPHC2_DAC) {
|
2015-04-14 00:18:36 +02:00
|
|
|
puts("Stateful destinaton multicast address compression:");
|
|
|
|
|
2015-08-17 15:06:44 +02:00
|
|
|
switch (data[1] & SIXLOWPAN_IPHC2_DAM) {
|
2015-04-14 00:18:36 +02:00
|
|
|
case 0x00:
|
|
|
|
puts(" 48 bits carried inline (Unicast-Prefix-based)");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x01:
|
|
|
|
case 0x02:
|
|
|
|
case 0x03:
|
|
|
|
puts(" reserved");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("Stateless destinaton multicast address compression:");
|
|
|
|
|
2015-08-17 15:06:44 +02:00
|
|
|
switch (data[1] & SIXLOWPAN_IPHC2_DAM) {
|
2015-04-14 00:18:36 +02:00
|
|
|
case 0x00:
|
|
|
|
puts(" 128 bits carried inline");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x01:
|
|
|
|
puts(" 48 bits carried inline");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x02:
|
|
|
|
puts(" 32 bits carried inline");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x03:
|
|
|
|
puts(" 8 bits carried inline");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2015-08-17 15:06:44 +02:00
|
|
|
if (data[1] & SIXLOWPAN_IPHC2_DAC) {
|
2015-04-14 00:18:36 +02:00
|
|
|
printf("Stateful destinaton address compression: ");
|
|
|
|
|
2015-08-17 15:06:44 +02:00
|
|
|
switch (data[1] & SIXLOWPAN_IPHC2_DAM) {
|
2015-04-14 00:18:36 +02:00
|
|
|
case 0x00:
|
|
|
|
puts("reserved");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x10:
|
|
|
|
puts("64 bits inline");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x20:
|
|
|
|
puts("16 bits inline");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x40:
|
|
|
|
puts("elided (use L2 address)");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
printf("Stateless destinaton address compression: ");
|
|
|
|
|
2015-08-17 15:06:44 +02:00
|
|
|
switch (data[1] & SIXLOWPAN_IPHC2_DAM) {
|
2015-04-14 00:18:36 +02:00
|
|
|
case 0x00:
|
|
|
|
puts("128 bits inline");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x10:
|
|
|
|
puts("64 bits inline");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x20:
|
|
|
|
puts("16 bits inline");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x40:
|
|
|
|
puts("elided (use L2 address)");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:06:44 +02:00
|
|
|
if (data[1] & SIXLOWPAN_IPHC2_CID_EXT) {
|
|
|
|
offset += SIXLOWPAN_IPHC_CID_EXT_LEN;
|
2015-04-14 00:18:36 +02:00
|
|
|
printf("SCI: 0x%" PRIx8 "; DCI: 0x%" PRIx8 "\n",
|
|
|
|
data[2] >> 4, data[2] & 0xf);
|
|
|
|
}
|
|
|
|
|
|
|
|
od_hex_dump(data + offset, size - offset, OD_WIDTH_DEFAULT);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @} */
|