mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
1828 lines
57 KiB
C
1828 lines
57 KiB
C
/*
|
|
* 6lowpan implementation
|
|
*
|
|
* Copyright (C) 2013 INRIA.
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU Lesser General
|
|
* Public License. See the file LICENSE in the top level directory for more
|
|
* details.
|
|
*
|
|
* @ingroup sixlowpan
|
|
* @{
|
|
* @file sixlowpan.c
|
|
* @brief 6lowpan functions
|
|
* @author Stephan Zeisberg <zeisberg@mi.fu-berlin.de>
|
|
* @author Martin Lenders <mlenders@inf.fu-berlin.de>
|
|
* @author Oliver Gesch <oliver.gesch@googlemail.com>
|
|
* @author Eric Engel <eric.engel@fu-berlin.de>
|
|
* @}
|
|
*/
|
|
|
|
#include <inttypes.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <limits.h>
|
|
#include <errno.h>
|
|
|
|
#include "vtimer.h"
|
|
#include "timex.h"
|
|
#include "thread.h"
|
|
#include "mutex.h"
|
|
#include "hwtimer.h"
|
|
#include "msg.h"
|
|
#include "transceiver.h"
|
|
#include "sixlowpan/mac.h"
|
|
#include "sixlowpan/ndp.h"
|
|
|
|
#include "lowpan.h"
|
|
#ifdef MODULE_SIXLOWBORDER
|
|
#include "border/border.h"
|
|
#endif
|
|
#include "ip.h"
|
|
#include "icmp.h"
|
|
|
|
#include "ieee802154_frame.h"
|
|
#include "destiny/in.h"
|
|
#include "net_help.h"
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
#if ENABLE_DEBUG
|
|
#define DEBUG_ENABLED
|
|
char addr_str[IPV6_MAX_ADDR_STR_LEN];
|
|
#endif
|
|
#include "debug.h"
|
|
|
|
#define CON_STACKSIZE (KERNEL_CONF_STACKSIZE_DEFAULT)
|
|
#define LOWPAN_TRANSFER_BUF_STACKSIZE (KERNEL_CONF_STACKSIZE_DEFAULT)
|
|
|
|
#define SIXLOWPAN_MAX_REGISTERED (4)
|
|
|
|
#define LOWPAN_REAS_BUF_TIMEOUT (15 * 1000 * 1000)
|
|
/* TODO: Set back to 3 * 1000 * (1000) */
|
|
|
|
#define IPV6_LL_ADDR_LEN (8)
|
|
|
|
#define SIXLOWPAN_FRAG_HDR_MASK (0xf8)
|
|
|
|
typedef struct lowpan_interval_list_t {
|
|
uint8_t start;
|
|
uint8_t end;
|
|
struct lowpan_interval_list_t *next;
|
|
} lowpan_interval_list_t;
|
|
|
|
/**
|
|
* @brief 6LoWPAN reassembly buffer.
|
|
*
|
|
* @see <a href="http://tools.ietf.org/html/rfc4944#section-5.3">
|
|
* RFC 4944, section 5.3
|
|
* </a>
|
|
*/
|
|
typedef struct lowpan_reas_buf_t {
|
|
net_if_eui64_t s_addr; ///< Source address
|
|
net_if_eui64_t d_addr; ///< Destination address
|
|
uint16_t tag; ///< Fragment tag
|
|
timex_t timestamp; ///< Timestamp of last packet fragment
|
|
/**
|
|
* @brief Size of reassembled packet with possible IPHC header
|
|
*/
|
|
uint16_t packet_size;
|
|
/**
|
|
* @brief Additive size of currently already received fragments
|
|
*/
|
|
uint16_t current_packet_size;
|
|
/**
|
|
* @brief Pointer to allocated memory for reassembled packet + 6LoWPAN
|
|
* Dispatch Byte
|
|
*/
|
|
uint8_t *packet;
|
|
/**
|
|
* @brief Pointer to list of intervals of received packet fragments
|
|
* (if any)
|
|
*/
|
|
lowpan_interval_list_t *interval_list_head;
|
|
struct lowpan_reas_buf_t *next;
|
|
} lowpan_reas_buf_t;
|
|
|
|
extern mutex_t lowpan_context_mutex;
|
|
uint16_t tag = 0;
|
|
uint8_t max_frag_initial = 0;
|
|
uint8_t max_frag;
|
|
|
|
static uint16_t packet_length;
|
|
static sixlowpan_lowpan_iphc_status_t iphc_status = LOWPAN_IPHC_ENABLE;
|
|
static ipv6_hdr_t *ipv6_buf;
|
|
static lowpan_reas_buf_t *head = NULL;
|
|
static lowpan_reas_buf_t *packet_fifo = NULL;
|
|
|
|
/* length of compressed packet */
|
|
uint16_t comp_len;
|
|
uint8_t frag_size;
|
|
uint8_t reas_buf[512];
|
|
uint8_t comp_buf[512];
|
|
uint8_t first_frag = 0;
|
|
mutex_t fifo_mutex;
|
|
|
|
int ip_process_pid = 0;
|
|
int nd_nbr_cache_rem_pid = 0;
|
|
int contexts_rem_pid = 0;
|
|
int transfer_pid = 0;
|
|
|
|
mutex_t lowpan_context_mutex;
|
|
|
|
/* registered upper layer threads */
|
|
int sixlowpan_reg[SIXLOWPAN_MAX_REGISTERED];
|
|
static sixlowpan_lowpan_frame_t current_frame;
|
|
|
|
char ip_process_buf[IP_PROCESS_STACKSIZE];
|
|
char con_buf[CON_STACKSIZE];
|
|
char lowpan_transfer_buf[LOWPAN_TRANSFER_BUF_STACKSIZE];
|
|
lowpan_context_t contexts[NDP_6LOWPAN_CONTEXT_MAX];
|
|
uint8_t context_len = 0;
|
|
uint16_t local_address = 0;
|
|
|
|
int lowpan_init(int as_border);
|
|
void lowpan_context_auto_remove(void);
|
|
uint8_t lowpan_iphc_encoding(int if_id, const uint8_t *dest, int dest_len,
|
|
ipv6_hdr_t *ipv6_buf_extra, uint8_t *ptr);
|
|
void lowpan_iphc_decoding(uint8_t *data, uint8_t length, net_if_eui64_t *s_addr,
|
|
net_if_eui64_t *d_addr);
|
|
void add_fifo_packet(lowpan_reas_buf_t *current_packet);
|
|
lowpan_reas_buf_t *collect_garbage_fifo(lowpan_reas_buf_t *current_buf);
|
|
lowpan_reas_buf_t *collect_garbage(lowpan_reas_buf_t *current_buf);
|
|
void init_reas_bufs(lowpan_reas_buf_t *buf);
|
|
void check_timeout(void);
|
|
void print_long_local_addr(net_if_eui64_t *saddr);
|
|
|
|
lowpan_context_t *lowpan_context_lookup(ipv6_addr_t *addr);
|
|
|
|
/* deliver packet to mac*/
|
|
int sixlowpan_lowpan_sendto(int if_id, const void *dest, int dest_len,
|
|
uint8_t *data, uint16_t data_len)
|
|
{
|
|
uint8_t mcast = 0;
|
|
|
|
ipv6_buf = (ipv6_hdr_t *) data;
|
|
uint16_t send_packet_length = data_len;
|
|
|
|
if (ipv6_addr_is_multicast(&ipv6_buf->destaddr)) {
|
|
/* send broadcast */
|
|
mcast = 1;
|
|
}
|
|
|
|
/* check if packet needs to be fragmented */
|
|
DEBUG("sixlowpan_lowpan_sendto(%d, dest, %d, data, %"PRIu16")\n",
|
|
if_id, dest_len, data_len);
|
|
#ifdef DEBUG_ENABLED
|
|
DEBUG("dest: ");
|
|
|
|
if (dest_len == 8) {
|
|
print_long_local_addr((net_if_eui64_t *)dest);
|
|
}
|
|
else {
|
|
printf("0x%04"PRIx16"\n", NTOHS(*((uint16_t *)dest)));
|
|
}
|
|
|
|
DEBUG("data: \n");
|
|
|
|
for (int i = 0; i < data_len; i++) {
|
|
printf("%02x ", data[i]);
|
|
|
|
if (!((i + 1) % 16) || i == data_len - 1) {
|
|
printf("\n");
|
|
}
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
if (iphc_status == LOWPAN_IPHC_ENABLE) {
|
|
if (!lowpan_iphc_encoding(if_id, dest, dest_len, ipv6_buf, data)) {
|
|
return -1;
|
|
}
|
|
|
|
data = &comp_buf[0];
|
|
send_packet_length = comp_len;
|
|
}
|
|
else {
|
|
memmove(data + 1, data, data_len);
|
|
data[0] = SIXLOWPAN_IPV6_DISPATCH;
|
|
send_packet_length++;
|
|
}
|
|
|
|
if (send_packet_length > PAYLOAD_SIZE - IEEE_802154_MAX_HDR_LEN) {
|
|
uint16_t remaining;
|
|
uint16_t position, datagram_size = send_packet_length;
|
|
uint8_t max_frame = PAYLOAD_SIZE - IEEE_802154_MAX_HDR_LEN;
|
|
uint8_t fragbuf[max_frame + 5];
|
|
|
|
if (!(net_if_get_interface(if_id)->transceivers & (IEEE802154_TRANSCEIVER))) {
|
|
max_frame = PAYLOAD_SIZE - IEEE_802154_MAX_HDR_LEN;
|
|
}
|
|
else {
|
|
max_frame = PAYLOAD_SIZE;
|
|
}
|
|
|
|
/* first fragment */
|
|
max_frag_initial = ((max_frame - 4) / 8) * 8;
|
|
|
|
if (data[0] == (char)SIXLOWPAN_IPV6_DISPATCH) {
|
|
/* XXX: weird, but only this way we get correct packet output */
|
|
max_frag_initial++;
|
|
datagram_size--;
|
|
}
|
|
|
|
memcpy(&fragbuf[4], data, max_frag_initial);
|
|
|
|
fragbuf[0] = ((SIXLOWPAN_FRAG1_DISPATCH << 8) | datagram_size) >> 8;
|
|
fragbuf[1] = (SIXLOWPAN_FRAG1_DISPATCH << 8) | datagram_size;
|
|
fragbuf[2] = tag >> 8;
|
|
fragbuf[3] = tag;
|
|
|
|
sixlowpan_mac_send_ieee802154_frame(if_id, dest, dest_len,
|
|
&fragbuf,
|
|
max_frag_initial + 4,
|
|
mcast);
|
|
|
|
/* subsequent fragments */
|
|
position = max_frag_initial;
|
|
max_frag = ((max_frame - 5) / 8) * 8;
|
|
|
|
data += position;
|
|
|
|
while (send_packet_length - position > max_frame - 5) {
|
|
memset(&fragbuf[0], 0, sizeof(fragbuf));
|
|
memcpy(&fragbuf[5], data, max_frag);
|
|
|
|
fragbuf[0] = ((SIXLOWPAN_FRAGN_DISPATCH << 8) | datagram_size) >> 8;
|
|
fragbuf[1] = (SIXLOWPAN_FRAGN_DISPATCH << 8) | datagram_size;
|
|
fragbuf[2] = tag >> 8;
|
|
fragbuf[3] = tag;
|
|
fragbuf[4] = position / 8;
|
|
|
|
sixlowpan_mac_send_ieee802154_frame(if_id, dest, dest_len,
|
|
&fragbuf,
|
|
max_frame + 5, mcast);
|
|
data += max_frag;
|
|
position += max_frag;
|
|
}
|
|
|
|
remaining = send_packet_length - position;
|
|
|
|
memset(&fragbuf[0], 0, sizeof(fragbuf));
|
|
memcpy(&fragbuf[5], data, remaining);
|
|
|
|
fragbuf[0] = ((SIXLOWPAN_FRAGN_DISPATCH << 8) | datagram_size) >> 8;
|
|
fragbuf[1] = (SIXLOWPAN_FRAGN_DISPATCH << 8) | datagram_size;
|
|
fragbuf[2] = tag >> 8;
|
|
fragbuf[3] = tag;
|
|
fragbuf[4] = position / 8;
|
|
|
|
tag++;
|
|
|
|
if (sixlowpan_mac_send_ieee802154_frame(if_id, dest, dest_len,
|
|
&fragbuf, remaining + 5, mcast) < 0) {
|
|
return -1;
|
|
}
|
|
}
|
|
else {
|
|
return sixlowpan_mac_send_ieee802154_frame(if_id, dest, dest_len, data,
|
|
send_packet_length, mcast);
|
|
}
|
|
|
|
return data_len;
|
|
}
|
|
|
|
void sixlowpan_lowpan_set_iphc_status(
|
|
sixlowpan_lowpan_iphc_status_t status)
|
|
{
|
|
iphc_status = status;
|
|
}
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
void print_long_local_addr(net_if_eui64_t *saddr)
|
|
{
|
|
printf("%02x%02x:%02x%02x:%02x%02x:%02x%02x\n",
|
|
((uint8_t *)saddr)[0], ((uint8_t *)saddr)[1], ((uint8_t *)saddr)[2],
|
|
((uint8_t *)saddr)[3], ((uint8_t *)saddr)[4], ((uint8_t *)saddr)[5],
|
|
((uint8_t *)saddr)[6], ((uint8_t *)saddr)[7]);
|
|
}
|
|
|
|
void sixlowpan_lowpan_print_reassembly_buffers(void)
|
|
{
|
|
lowpan_reas_buf_t *temp_buffer;
|
|
lowpan_interval_list_t *temp_interval;
|
|
temp_buffer = head;
|
|
|
|
printf("\n\n--- Reassembly Buffers ---\n");
|
|
|
|
while (temp_buffer != NULL) {
|
|
print_long_local_addr(&temp_buffer->s_addr);
|
|
printf("Ident.: %i, Packet Size: %i/%i, Timestamp: %"PRIu64"\n",
|
|
temp_buffer->tag, temp_buffer->current_packet_size,
|
|
temp_buffer->packet_size, timex_uint64(temp_buffer->timestamp));
|
|
temp_interval = temp_buffer->interval_list_head;
|
|
|
|
while (temp_interval != NULL) {
|
|
printf("\t%i - %i\n", temp_interval->start, temp_interval->end);
|
|
temp_interval = temp_interval->next;
|
|
}
|
|
|
|
temp_buffer = temp_buffer->next;
|
|
}
|
|
}
|
|
|
|
void sixlowpan_lowpan_print_fifo_buffers(void)
|
|
{
|
|
lowpan_reas_buf_t *temp_buffer;
|
|
lowpan_interval_list_t *temp_interval;
|
|
temp_buffer = packet_fifo;
|
|
|
|
printf("\n\n--- Reassembly Buffers ---\n");
|
|
|
|
while (temp_buffer != NULL) {
|
|
print_long_local_addr(&temp_buffer->s_addr);
|
|
printf("Ident.: %i, Packet Size: %i/%i, Timestamp: %"PRIu64"\n",
|
|
temp_buffer->tag, temp_buffer->current_packet_size,
|
|
temp_buffer->packet_size, timex_uint64(temp_buffer->timestamp));
|
|
temp_interval = temp_buffer->interval_list_head;
|
|
|
|
while (temp_interval != NULL) {
|
|
printf("\t%i - %i\n", temp_interval->start, temp_interval->end);
|
|
temp_interval = temp_interval->next;
|
|
}
|
|
|
|
temp_buffer = temp_buffer->next;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
void lowpan_transfer(void)
|
|
{
|
|
msg_t m_recv, m_send;
|
|
ipv6_hdr_t *ipv6_buf;
|
|
lowpan_reas_buf_t *current_buf;
|
|
uint8_t gotosleep;
|
|
|
|
while (1) {
|
|
|
|
gotosleep = 1;
|
|
mutex_lock(&fifo_mutex);
|
|
current_buf = packet_fifo;
|
|
|
|
if (current_buf != NULL) {
|
|
mutex_unlock(&fifo_mutex);
|
|
|
|
if (current_buf->packet[0] == SIXLOWPAN_IPV6_DISPATCH) {
|
|
DEBUG("INFO: Uncompressed IPv6 dispatch (0x%02x) received\n",
|
|
current_buf->packet[0]);
|
|
ipv6_buf = ipv6_get_buf();
|
|
memcpy(ipv6_buf, (current_buf->packet) + 1, current_buf->packet_size - 1);
|
|
m_send.content.ptr = (char *)ipv6_buf;
|
|
packet_length = current_buf->packet_size - 1;
|
|
msg_send_receive(&m_send, &m_recv, ip_process_pid);
|
|
}
|
|
else if (((current_buf->packet[0] & 0xf0) == IPV6_VER) &&
|
|
(iphc_status == LOWPAN_IPHC_DISABLE)) {
|
|
ipv6_buf = ipv6_get_buf();
|
|
memcpy(ipv6_buf, (current_buf->packet), current_buf->packet_size);
|
|
m_send.content.ptr = (char *)ipv6_buf;
|
|
packet_length = current_buf->packet_size;
|
|
msg_send_receive(&m_send, &m_recv, ip_process_pid);
|
|
}
|
|
else if (((current_buf->packet[0] & 0xe0) == SIXLOWPAN_IPHC1_DISPATCH) &&
|
|
(iphc_status == LOWPAN_IPHC_ENABLE)) {
|
|
DEBUG("INFO: IPHC1 dispatch 0x%02x received, decompress\n",
|
|
current_buf->packet[0]);
|
|
lowpan_iphc_decoding(current_buf->packet,
|
|
current_buf->packet_size,
|
|
&(current_buf->s_addr),
|
|
&(current_buf->d_addr));
|
|
|
|
ipv6_buf = ipv6_get_buf();
|
|
m_send.content.ptr = (char *) ipv6_buf;
|
|
msg_send_receive(&m_send, &m_recv, ip_process_pid);
|
|
}
|
|
else {
|
|
DEBUG("ERROR: packet with unknown dispatch 0x%02x received\n",
|
|
current_buf->packet[0]);
|
|
}
|
|
|
|
collect_garbage_fifo(current_buf);
|
|
gotosleep = 0;
|
|
}
|
|
|
|
|
|
if (gotosleep == 1) {
|
|
mutex_unlock(&fifo_mutex);
|
|
thread_sleep();
|
|
}
|
|
}
|
|
}
|
|
|
|
uint8_t ll_get_addr_match(net_if_eui64_t *src, net_if_eui64_t *dst)
|
|
{
|
|
uint8_t val = 0, xor;
|
|
|
|
for (uint8_t i = 0; i < sizeof(net_if_eui64_t); i++) {
|
|
/* if bytes are equal add 8 */
|
|
if (src->uint8[i] == dst->uint8[i]) {
|
|
val += 8;
|
|
}
|
|
else {
|
|
xor = src->uint8[i] ^ dst->uint8[i];
|
|
|
|
/* while bits from byte equal add 1 */
|
|
for (int j = 0; j < 8; j++) {
|
|
if ((xor & 0x80) == 0) {
|
|
val++;
|
|
xor = xor << 1;
|
|
}
|
|
else {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return val;
|
|
}
|
|
|
|
lowpan_reas_buf_t *new_packet_buffer(uint16_t datagram_size,
|
|
uint16_t datagram_tag,
|
|
net_if_eui64_t *s_addr,
|
|
net_if_eui64_t *d_addr,
|
|
lowpan_reas_buf_t *current_buf,
|
|
lowpan_reas_buf_t *temp_buf)
|
|
{
|
|
lowpan_reas_buf_t *new_buf = NULL;
|
|
|
|
/* Allocate new memory for a new packet to be reassembled */
|
|
new_buf = malloc(sizeof(lowpan_reas_buf_t));
|
|
|
|
if (new_buf != NULL) {
|
|
init_reas_bufs(new_buf);
|
|
|
|
new_buf->packet = malloc(datagram_size);
|
|
|
|
if (new_buf->packet != NULL) {
|
|
memcpy(&new_buf->s_addr, s_addr, 8);
|
|
memcpy(&new_buf->d_addr, d_addr, 8);
|
|
|
|
new_buf->tag = datagram_tag;
|
|
new_buf->packet_size = datagram_size;
|
|
|
|
vtimer_now(&new_buf->timestamp);
|
|
|
|
if ((current_buf == NULL) && (temp_buf == NULL)) {
|
|
head = new_buf;
|
|
}
|
|
else {
|
|
temp_buf->next = new_buf;
|
|
}
|
|
|
|
return new_buf;
|
|
}
|
|
else {
|
|
return NULL;
|
|
}
|
|
}
|
|
else {
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
lowpan_reas_buf_t *get_packet_frag_buf(uint16_t datagram_size,
|
|
uint16_t datagram_tag,
|
|
net_if_eui64_t *s_addr,
|
|
net_if_eui64_t *d_addr)
|
|
{
|
|
lowpan_reas_buf_t *current_buf = NULL, *temp_buf = NULL;
|
|
current_buf = head;
|
|
|
|
while (current_buf != NULL) {
|
|
if (((ll_get_addr_match(¤t_buf->s_addr, s_addr)) == 64) &&
|
|
((ll_get_addr_match(¤t_buf->d_addr, d_addr)) == 64) &&
|
|
(current_buf->packet_size == datagram_size) &&
|
|
(current_buf->tag == datagram_tag) &&
|
|
current_buf->interval_list_head != NULL) {
|
|
/* Found buffer for current packet fragment */
|
|
vtimer_now(¤t_buf->timestamp);
|
|
return current_buf;
|
|
}
|
|
|
|
temp_buf = current_buf;
|
|
current_buf = current_buf->next;
|
|
}
|
|
|
|
return new_packet_buffer(datagram_size, datagram_tag, s_addr, d_addr,
|
|
current_buf, temp_buf);
|
|
}
|
|
|
|
uint8_t is_in_interval(uint8_t start1, uint8_t end1, uint8_t start2, uint8_t end2)
|
|
{
|
|
/* 1: Interval 1 and 2 are the same or overlapping */
|
|
/* 0: Interval 1 and 2 are not overlapping or the same */
|
|
|
|
if (((start1 < start2) && (start2 <= end1)) ||
|
|
((start2 < start1) && (start1 <= end2)) ||
|
|
((start1 == start2) && (end1 == end2))) {
|
|
return 1;
|
|
}
|
|
else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
uint8_t handle_packet_frag_interval(lowpan_reas_buf_t *current_buf,
|
|
uint8_t datagram_offset, uint8_t frag_size)
|
|
{
|
|
/* 0: Error, discard fragment */
|
|
/* 1: Finished correctly */
|
|
lowpan_interval_list_t *temp_interval = NULL, *current_interval = NULL, *new_interval = NULL;
|
|
current_interval = current_buf->interval_list_head;
|
|
|
|
while (current_interval != NULL) {
|
|
if (is_in_interval(current_interval->start, current_interval->end, datagram_offset, datagram_offset + frag_size) == 1) {
|
|
/* Interval is overlapping or the same as one of a previous fragment, discard fragment */
|
|
return 0;
|
|
}
|
|
|
|
temp_interval = current_interval;
|
|
current_interval = current_interval->next;
|
|
}
|
|
|
|
new_interval = malloc(sizeof(lowpan_interval_list_t));
|
|
|
|
if (new_interval != NULL) {
|
|
new_interval->start = datagram_offset;
|
|
new_interval->end = datagram_offset + frag_size - 1;
|
|
new_interval->next = NULL;
|
|
|
|
if ((current_interval == NULL) && (temp_interval == NULL)) {
|
|
current_buf->interval_list_head = new_interval;
|
|
}
|
|
else {
|
|
temp_interval->next = new_interval;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
lowpan_reas_buf_t *collect_garbage_fifo(lowpan_reas_buf_t *current_buf)
|
|
{
|
|
lowpan_interval_list_t *temp_list, *current_list;
|
|
lowpan_reas_buf_t *temp_buf, *my_buf, *return_buf;
|
|
|
|
mutex_lock(&fifo_mutex);
|
|
|
|
temp_buf = packet_fifo;
|
|
my_buf = temp_buf;
|
|
|
|
if (packet_fifo == current_buf) {
|
|
packet_fifo = current_buf->next;
|
|
return_buf = packet_fifo;
|
|
}
|
|
else {
|
|
while (temp_buf != current_buf) {
|
|
my_buf = temp_buf;
|
|
temp_buf = temp_buf->next;
|
|
}
|
|
|
|
my_buf->next = current_buf->next;
|
|
|
|
return_buf = my_buf->next;
|
|
}
|
|
|
|
mutex_unlock(&fifo_mutex);
|
|
|
|
current_list = current_buf->interval_list_head;
|
|
temp_list = current_list;
|
|
|
|
while (current_list != NULL) {
|
|
temp_list = current_list->next;
|
|
free(current_list);
|
|
current_list = temp_list;
|
|
}
|
|
|
|
free(current_buf->packet);
|
|
free(current_buf);
|
|
|
|
return return_buf;
|
|
}
|
|
|
|
lowpan_reas_buf_t *collect_garbage(lowpan_reas_buf_t *current_buf)
|
|
{
|
|
lowpan_interval_list_t *temp_list, *current_list;
|
|
lowpan_reas_buf_t *temp_buf, *my_buf, *return_buf;
|
|
|
|
temp_buf = head;
|
|
my_buf = temp_buf;
|
|
|
|
if (head == current_buf) {
|
|
head = current_buf->next;
|
|
return_buf = head;
|
|
}
|
|
else {
|
|
while (temp_buf != current_buf) {
|
|
my_buf = temp_buf;
|
|
temp_buf = temp_buf->next;
|
|
}
|
|
|
|
my_buf->next = current_buf->next;
|
|
|
|
return_buf = my_buf->next;
|
|
}
|
|
|
|
current_list = current_buf->interval_list_head;
|
|
temp_list = current_list;
|
|
|
|
while (current_list != NULL) {
|
|
temp_list = current_list->next;
|
|
free(current_list);
|
|
current_list = temp_list;
|
|
}
|
|
|
|
free(current_buf->packet);
|
|
free(current_buf);
|
|
|
|
return return_buf;
|
|
}
|
|
|
|
void handle_packet_fragment(uint8_t *data, uint8_t datagram_offset,
|
|
uint16_t datagram_size, uint16_t datagram_tag,
|
|
net_if_eui64_t *s_addr, net_if_eui64_t *d_addr,
|
|
uint8_t hdr_length, uint8_t frag_size)
|
|
{
|
|
lowpan_reas_buf_t *current_buf;
|
|
/* Is there already a reassembly buffer for this packet fragment? */
|
|
current_buf = get_packet_frag_buf(datagram_size, datagram_tag, s_addr,
|
|
d_addr);
|
|
|
|
if ((current_buf != NULL) && (handle_packet_frag_interval(current_buf,
|
|
datagram_offset,
|
|
frag_size) == 1)) {
|
|
/* Copy fragment bytes into corresponding packet space area */
|
|
memcpy(current_buf->packet + datagram_offset, data + hdr_length, frag_size);
|
|
current_buf->current_packet_size += frag_size;
|
|
|
|
if (current_buf->current_packet_size == current_buf->packet_size) {
|
|
add_fifo_packet(current_buf);
|
|
|
|
if (thread_getstatus(transfer_pid) == STATUS_SLEEPING) {
|
|
thread_wakeup(transfer_pid);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
/* No memory left or duplicate */
|
|
if (current_buf == NULL) {
|
|
printf("ERROR: no memory left!\n");
|
|
}
|
|
else {
|
|
printf("ERROR: duplicate fragment!\n");
|
|
}
|
|
}
|
|
}
|
|
|
|
void check_timeout(void)
|
|
{
|
|
lowpan_reas_buf_t *temp_buf, *smallest_time = NULL;
|
|
int count = 0;
|
|
|
|
timex_t now;
|
|
vtimer_now(&now);
|
|
temp_buf = head;
|
|
|
|
while (temp_buf != NULL) {
|
|
if ((timex_uint64(now) - timex_uint64(temp_buf->timestamp)) >= LOWPAN_REAS_BUF_TIMEOUT) {
|
|
printf("TIMEOUT!cur_time: %" PRIu64 ", temp_buf: %" PRIu64 "\n", timex_uint64(now),
|
|
timex_uint64(temp_buf->timestamp));
|
|
temp_buf = collect_garbage(temp_buf);
|
|
}
|
|
else {
|
|
if (smallest_time == NULL) {
|
|
smallest_time = temp_buf;
|
|
}
|
|
else if (timex_cmp(temp_buf->timestamp, smallest_time->timestamp) < 0) {
|
|
smallest_time = temp_buf;
|
|
}
|
|
|
|
temp_buf = temp_buf->next;
|
|
count++;
|
|
}
|
|
}
|
|
|
|
if ((count > 10) && (smallest_time != NULL)) {
|
|
collect_garbage(smallest_time);
|
|
}
|
|
}
|
|
|
|
void add_fifo_packet(lowpan_reas_buf_t *current_packet)
|
|
{
|
|
lowpan_reas_buf_t *temp_buf, *my_buf;
|
|
|
|
if (head == current_packet) {
|
|
head = current_packet->next;
|
|
}
|
|
else {
|
|
temp_buf = head;
|
|
|
|
while (temp_buf != current_packet) {
|
|
my_buf = temp_buf;
|
|
temp_buf = temp_buf->next;
|
|
}
|
|
|
|
my_buf->next = current_packet->next;
|
|
}
|
|
|
|
mutex_lock(&fifo_mutex);
|
|
|
|
if (packet_fifo == NULL) {
|
|
packet_fifo = current_packet;
|
|
}
|
|
else {
|
|
temp_buf = packet_fifo;
|
|
|
|
while (temp_buf != NULL) {
|
|
my_buf = temp_buf;
|
|
temp_buf = temp_buf->next;
|
|
}
|
|
|
|
my_buf->next = current_packet;
|
|
}
|
|
|
|
mutex_unlock(&fifo_mutex);
|
|
current_packet->next = NULL;
|
|
}
|
|
|
|
/* Register an upper layer thread */
|
|
uint8_t sixlowpan_lowpan_register(int pid)
|
|
{
|
|
uint8_t i;
|
|
|
|
for (i = 0; ((sixlowpan_reg[i] != pid) && (i < SIXLOWPAN_MAX_REGISTERED) &&
|
|
(sixlowpan_reg[i] != 0)); i++) {
|
|
;
|
|
}
|
|
|
|
if (i >= SIXLOWPAN_MAX_REGISTERED) {
|
|
return ENOMEM;
|
|
}
|
|
else {
|
|
sixlowpan_reg[i] = pid;
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
void lowpan_read(uint8_t *data, uint8_t length, net_if_eui64_t *s_addr,
|
|
net_if_eui64_t *d_addr)
|
|
{
|
|
/* check if packet is fragmented */
|
|
uint8_t hdr_length = 0;
|
|
uint8_t datagram_offset = 0;
|
|
uint16_t datagram_size = 0;
|
|
uint16_t datagram_tag = 0;
|
|
short i;
|
|
|
|
check_timeout();
|
|
|
|
for (i = 0; i < SIXLOWPAN_MAX_REGISTERED; i++) {
|
|
if (sixlowpan_reg[i]) {
|
|
msg_t m_send;
|
|
m_send.type = LOWPAN_FRAME_RECEIVED;;
|
|
current_frame.length = length;
|
|
current_frame.data = data;
|
|
m_send.content.ptr = (char *) ¤t_frame;
|
|
msg_send(&m_send, sixlowpan_reg[i], 1);
|
|
}
|
|
}
|
|
|
|
/* Fragmented Packet */
|
|
if (((data[0] & SIXLOWPAN_FRAG_HDR_MASK) == SIXLOWPAN_FRAG1_DISPATCH) ||
|
|
((data[0] & SIXLOWPAN_FRAG_HDR_MASK) == SIXLOWPAN_FRAGN_DISPATCH)) {
|
|
uint16_t byte_offset;
|
|
DEBUG("INFO: fragmentation dispatch 0x%02x received\n",
|
|
data[0] & SIXLOWPAN_FRAG_HDR_MASK);
|
|
/* get 11-bit from first 2 byte*/
|
|
datagram_size = (((uint16_t)(data[0] << 8)) | data[1]) & 0x07ff;
|
|
|
|
/* get 16-bit datagram tag */
|
|
datagram_tag = (((uint16_t)(data[2] << 8)) | data[3]);
|
|
|
|
switch (data[0] & SIXLOWPAN_FRAG_HDR_MASK) {
|
|
/* First Fragment */
|
|
case (SIXLOWPAN_FRAG1_DISPATCH): {
|
|
datagram_offset = 0;
|
|
hdr_length += 4;
|
|
|
|
if (data[4] == (char)SIXLOWPAN_IPV6_DISPATCH) {
|
|
hdr_length++;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
/* Subsequent Fragment */
|
|
case (SIXLOWPAN_FRAGN_DISPATCH): {
|
|
datagram_offset = data[4];
|
|
hdr_length += 5;
|
|
break;
|
|
}
|
|
}
|
|
|
|
frag_size = length - hdr_length;
|
|
byte_offset = datagram_offset * 8;
|
|
|
|
if ((frag_size % 8) != 0) {
|
|
if ((byte_offset + frag_size) != datagram_size) {
|
|
printf("ERROR: received invalid fragment\n");
|
|
return;
|
|
}
|
|
}
|
|
|
|
handle_packet_fragment(data, byte_offset, datagram_size, datagram_tag,
|
|
s_addr, d_addr, hdr_length, frag_size);
|
|
}
|
|
/* Regular Packet */
|
|
else {
|
|
DEBUG("INFO: unfragmentated packet with first byte 0x%02x received\n",
|
|
data[0]);
|
|
lowpan_reas_buf_t *current_buf = get_packet_frag_buf(length, 0, s_addr, d_addr);
|
|
|
|
if (current_buf && current_buf->packet) {
|
|
/* Copy packet bytes into corresponding packet space area */
|
|
memcpy(current_buf->packet, data, length);
|
|
current_buf->current_packet_size += length;
|
|
add_fifo_packet(current_buf);
|
|
}
|
|
else {
|
|
DEBUG("ERROR: no memory left in packet buffer!\n");
|
|
}
|
|
|
|
if (thread_getstatus(transfer_pid) == STATUS_SLEEPING) {
|
|
thread_wakeup(transfer_pid);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
/* draft-ietf-6lowpan-hc-13#section-3.1 */
|
|
uint8_t lowpan_iphc_encoding(int if_id, const uint8_t *dest, int dest_len,
|
|
ipv6_hdr_t *ipv6_buf_extra, uint8_t *ptr)
|
|
{
|
|
uint16_t payload_length = NTOHS(ipv6_buf->length);
|
|
uint8_t lowpan_iphc[2];
|
|
uint8_t *ipv6_hdr_fields = &comp_buf[2];
|
|
lowpan_context_t *con = NULL;
|
|
uint16_t hdr_pos = 0;
|
|
uint8_t tc;
|
|
net_if_eui64_t own_iid;
|
|
|
|
if (net_if_get_src_address_mode(if_id) == NET_IF_TRANS_ADDR_M_SHORT) {
|
|
if (!net_if_get_eui64(&own_iid, if_id, 1)) {
|
|
return 1;
|
|
}
|
|
}
|
|
else {
|
|
if (!net_if_get_eui64(&own_iid, if_id, 0)) {
|
|
return 1;
|
|
}
|
|
|
|
own_iid.uint8[0] ^= 0x02;
|
|
}
|
|
|
|
ipv6_buf = ipv6_buf_extra;
|
|
|
|
memset(&lowpan_iphc, 0, 2);
|
|
|
|
/* set iphc dispatch */
|
|
lowpan_iphc[0] = SIXLOWPAN_IPHC1_DISPATCH;
|
|
|
|
/* TF: Traffic Class, Flow Label:
|
|
* first we need to change DSCP and ECN because in 6lowpan-nd-13 these
|
|
* fields are reverse, the original order is DSCP/ECN (rfc 3168) */
|
|
tc = (ipv6_buf->version_trafficclass << 4) |
|
|
(ipv6_buf->trafficclass_flowlabel >> 4);
|
|
tc = (tc >> 2) | (tc << 6);
|
|
|
|
if ((ipv6_buf->flowlabel == 0) &&
|
|
(ipv6_buf->trafficclass_flowlabel & 0x0f) == 0) {
|
|
/* flowlabel is elided */
|
|
lowpan_iphc[0] |= SIXLOWPAN_IPHC1_FL_C;
|
|
|
|
if (((ipv6_buf->version_trafficclass & 0x0f) == 0) &&
|
|
((ipv6_buf->trafficclass_flowlabel & 0xf0) == 0)) {
|
|
/* traffic class is elided */
|
|
lowpan_iphc[0] |= SIXLOWPAN_IPHC1_TC_C;
|
|
}
|
|
else {
|
|
/* ECN + DSCP (1 byte), Flow Label is elided */
|
|
ipv6_hdr_fields[hdr_pos] = tc;
|
|
hdr_pos++;
|
|
}
|
|
}
|
|
else {
|
|
/* flowlabel not compressible */
|
|
if (((ipv6_buf->version_trafficclass & 0x0f) == 0) &&
|
|
((ipv6_buf->trafficclass_flowlabel & 0xf0) == 0)) {
|
|
/* traffic class is elided */
|
|
lowpan_iphc[0] |= SIXLOWPAN_IPHC1_TC_C;
|
|
/* ECN + 2-bit Pad + Flow Label (3 bytes), DSCP is elided */
|
|
ipv6_hdr_fields[hdr_pos] = ((tc & 0xc0) |
|
|
(ipv6_buf->trafficclass_flowlabel & 0x0f));
|
|
memcpy(&(ipv6_hdr_fields[hdr_pos]), &ipv6_buf->flowlabel , 2);
|
|
hdr_pos += 3;
|
|
}
|
|
else {
|
|
/* ECN + DSCP + 4-bit Pad + Flow Label (4 bytes) */
|
|
memcpy(&ipv6_hdr_fields[hdr_pos], &ipv6_buf->version_trafficclass, 4);
|
|
ipv6_hdr_fields[hdr_pos] = tc;
|
|
hdr_pos += 4;
|
|
}
|
|
}
|
|
|
|
/* NH: Next Header:
|
|
* TODO: NHC */
|
|
ipv6_hdr_fields[hdr_pos] = ipv6_buf->nextheader;
|
|
hdr_pos++;
|
|
|
|
/* HLIM: Hop Limit: */
|
|
switch (ipv6_buf->hoplimit) {
|
|
case (1): {
|
|
/* 01: The Hop Limit field is compressed and the hop limit is 1. */
|
|
lowpan_iphc[0] |= 0x01;
|
|
break;
|
|
}
|
|
|
|
case (64): {
|
|
/* 10: The Hop Limit field is compressed and the hop limit is 64. */
|
|
lowpan_iphc[0] |= 0x02;
|
|
break;
|
|
}
|
|
|
|
case (255): {
|
|
/* 11: The Hop Limit field is compressed and the hop limit is 255. */
|
|
lowpan_iphc[0] |= 0x03;
|
|
break;
|
|
}
|
|
|
|
default: {
|
|
ipv6_hdr_fields[hdr_pos] = ipv6_buf->hoplimit;
|
|
hdr_pos++;
|
|
break;
|
|
}
|
|
}
|
|
|
|
mutex_lock(&lowpan_context_mutex);
|
|
|
|
/* CID: Context Identifier Extension: */
|
|
if ((lowpan_context_lookup(&ipv6_buf->srcaddr) != NULL) ||
|
|
(lowpan_context_lookup(&ipv6_buf->destaddr) != NULL)) {
|
|
lowpan_iphc[1] |= SIXLOWPAN_IPHC2_CID;
|
|
memmove(&ipv6_hdr_fields[1], &ipv6_hdr_fields[0], hdr_pos);
|
|
hdr_pos++;
|
|
}
|
|
|
|
/* SAC: Source Address Compression */
|
|
if (ipv6_addr_is_unspecified(&(ipv6_buf->srcaddr))) {
|
|
/* SAC = 1 and SAM = 00 */
|
|
lowpan_iphc[1] |= SIXLOWPAN_IPHC2_SAC;
|
|
}
|
|
else {
|
|
if ((con = lowpan_context_lookup(&ipv6_buf->srcaddr)) != NULL) {
|
|
/* 1: Source address compression uses stateful, context-based
|
|
* compression. */
|
|
lowpan_iphc[1] |= SIXLOWPAN_IPHC2_SAC;
|
|
ipv6_hdr_fields[0] |= (con->num << 4);
|
|
}
|
|
|
|
if (con || ipv6_addr_is_link_local(&ipv6_buf->srcaddr)) {
|
|
/* 0: Source address compression uses stateless compression.*/
|
|
if (memcmp(&(ipv6_buf->srcaddr.uint8[8]), &own_iid, 8) == 0) {
|
|
/* 0 bits. The address is derived using context information
|
|
* and possibly the link-layer addresses.*/
|
|
lowpan_iphc[1] |= 0x30;
|
|
}
|
|
else if ((ipv6_buf->srcaddr.uint16[4] == 0) &&
|
|
(ipv6_buf->srcaddr.uint16[5] == 0) &&
|
|
(ipv6_buf->srcaddr.uint16[6] == 0) &&
|
|
((ipv6_buf->srcaddr.uint8[14]) & 0x80) == 0) {
|
|
/* 49-bit of interface identifier are 0, so we can compress
|
|
* source address-iid to 16-bit */
|
|
memcpy(&ipv6_hdr_fields[hdr_pos], &ipv6_buf->srcaddr.uint16[7], 2);
|
|
hdr_pos += 2;
|
|
/* 16 bits. The address is derived using context information
|
|
* and the 16 bits carried inline. */
|
|
lowpan_iphc[1] |= 0x20;
|
|
}
|
|
else {
|
|
memcpy(&ipv6_hdr_fields[hdr_pos], &(ipv6_buf->srcaddr.uint16[4]), 8);
|
|
hdr_pos += 8;
|
|
/* 64 bits. The address is derived using context information
|
|
* and the 64 bits carried inline. */
|
|
lowpan_iphc[1] |= 0x10;
|
|
}
|
|
}
|
|
else {
|
|
/* full address carried inline */
|
|
memcpy(&ipv6_hdr_fields[hdr_pos], &(ipv6_buf->srcaddr.uint8[0]), 16);
|
|
hdr_pos += 16;
|
|
}
|
|
}
|
|
|
|
/* M: Multicast Compression */
|
|
if (ipv6_addr_is_multicast(&ipv6_buf->destaddr)) {
|
|
/* 1: Destination address is a multicast address. */
|
|
lowpan_iphc[1] |= SIXLOWPAN_IPHC2_M;
|
|
|
|
/* just another cool if condition */
|
|
if ((ipv6_buf->destaddr.uint8[1] == 2) &&
|
|
(ipv6_buf->destaddr.uint16[1] == 0) &&
|
|
(ipv6_buf->destaddr.uint16[2] == 0) &&
|
|
(ipv6_buf->destaddr.uint16[3] == 0) &&
|
|
(ipv6_buf->destaddr.uint16[4] == 0) &&
|
|
(ipv6_buf->destaddr.uint16[5] == 0) &&
|
|
(ipv6_buf->destaddr.uint16[6] == 0) &&
|
|
(ipv6_buf->destaddr.uint8[14] == 0)) {
|
|
/* 11: 8 bits. The address takes the form FF02::00XX. */
|
|
lowpan_iphc[1] |= 0x03;
|
|
ipv6_hdr_fields[hdr_pos] = ipv6_buf->destaddr.uint8[15];
|
|
hdr_pos++;
|
|
}
|
|
else if ((ipv6_buf->destaddr.uint16[1] == 0) &&
|
|
(ipv6_buf->destaddr.uint16[2] == 0) &&
|
|
(ipv6_buf->destaddr.uint16[3] == 0) &&
|
|
(ipv6_buf->destaddr.uint16[4] == 0) &&
|
|
(ipv6_buf->destaddr.uint16[5] == 0) &&
|
|
(ipv6_buf->destaddr.uint8[12] == 0)) {
|
|
/* 10: 32 bits. The address takes the form FFXX::00XX:XXXX. */
|
|
lowpan_iphc[1] |= 0x02;
|
|
/* copy second and last 3 byte */
|
|
ipv6_hdr_fields[hdr_pos] = ipv6_buf->destaddr.uint8[1];
|
|
hdr_pos++;
|
|
memcpy(&ipv6_hdr_fields[hdr_pos], &ipv6_buf->destaddr.uint8[13], 3);
|
|
hdr_pos += 3;
|
|
}
|
|
else if ((ipv6_buf->destaddr.uint16[1] == 0) &&
|
|
(ipv6_buf->destaddr.uint16[2] == 0) &&
|
|
(ipv6_buf->destaddr.uint16[3] == 0) &&
|
|
(ipv6_buf->destaddr.uint16[4] == 0) &&
|
|
(ipv6_buf->destaddr.uint8[10] == 0)) {
|
|
/* 01: 48 bits. The address takes the form FFXX::00XX:XXXX:XXXX */
|
|
lowpan_iphc[1] |= 0x01;
|
|
/* copy second and last 5 byte */
|
|
ipv6_hdr_fields[hdr_pos] = ipv6_buf->destaddr.uint8[1];
|
|
hdr_pos++;
|
|
memcpy(&ipv6_hdr_fields[hdr_pos], &ipv6_buf->destaddr.uint8[11], 5);
|
|
hdr_pos += 5;
|
|
}
|
|
else {
|
|
memcpy(&ipv6_hdr_fields[hdr_pos], &ipv6_buf->destaddr.uint8[0], 16);
|
|
hdr_pos += 16;
|
|
}
|
|
}
|
|
else {
|
|
/* 0: Destination address is not a multicast address. */
|
|
if ((con = lowpan_context_lookup(&ipv6_buf->destaddr)) != NULL) {
|
|
/* 1: Destination address compression uses stateful, context-based
|
|
* compression. */
|
|
lowpan_iphc[1] |= SIXLOWPAN_IPHC2_DAC;
|
|
ipv6_hdr_fields[0] = con->num;
|
|
|
|
}
|
|
|
|
if (con || ipv6_addr_is_link_local(&ipv6_buf->destaddr)) {
|
|
if (dest_len == 8 &&
|
|
ipv6_buf->destaddr.uint8[8] == (dest[0] ^ 0x02) &&
|
|
memcmp(&ipv6_buf->destaddr.uint8[9], &dest[1], 7) == 0) {
|
|
/* 0 bits. The address is derived using context information
|
|
* and possibly the link-layer addresses.*/
|
|
lowpan_iphc[1] |= 0x03;
|
|
}
|
|
else if (dest_len == 2 &&
|
|
ipv6_buf->destaddr.uint32[2] == HTONL(0x000000ff) &&
|
|
ipv6_buf->destaddr.uint16[6] == HTONS(0xfe00) &&
|
|
ipv6_buf->destaddr.uint16[7] == *((uint16_t *) dest)) {
|
|
/* 0 bits. The address is derived using context information
|
|
* and possibly the link-layer addresses.*/
|
|
lowpan_iphc[1] |= 0x03;
|
|
}
|
|
else if ((ipv6_buf->destaddr.uint16[4] == 0) &&
|
|
(ipv6_buf->destaddr.uint16[5] == 0) &&
|
|
(ipv6_buf->destaddr.uint16[6] == 0) &&
|
|
((ipv6_buf->destaddr.uint8[14]) & 0x80) == 0) {
|
|
/* 49-bit of interface identifier are 0, so we can compress
|
|
* source address-iid to 16-bit */
|
|
memcpy(&ipv6_hdr_fields[hdr_pos], &ipv6_buf->destaddr.uint16[7], 2);
|
|
hdr_pos += 2;
|
|
/* 16 bits. The address is derived using context information
|
|
* and the 16 bits carried inline. */
|
|
lowpan_iphc[1] |= 0x02;
|
|
}
|
|
else {
|
|
memcpy(&ipv6_hdr_fields[hdr_pos], &(ipv6_buf->destaddr.uint16[4]), 8);
|
|
hdr_pos += 8;
|
|
/* 64 bits. The address is derived using context information
|
|
* and the 64 bits carried inline. */
|
|
lowpan_iphc[1] |= 0x01;
|
|
}
|
|
}
|
|
else {
|
|
memcpy(&ipv6_hdr_fields[hdr_pos], &(ipv6_buf->destaddr.uint8[0]), 16);
|
|
hdr_pos += 16;
|
|
}
|
|
}
|
|
|
|
mutex_unlock(&lowpan_context_mutex);
|
|
|
|
comp_buf[0] = lowpan_iphc[0];
|
|
comp_buf[1] = lowpan_iphc[1];
|
|
|
|
/*uint8_t *ptr;
|
|
if (ipv6_buf->nextheader == IPV6_PROTO_NUM_TCP)
|
|
{
|
|
ptr = get_payload_buf_send(ipv6_ext_hdr_len);
|
|
}
|
|
else
|
|
{
|
|
ptr = get_payload_buf(ipv6_ext_hdr_len);
|
|
}
|
|
*/
|
|
memcpy(&ipv6_hdr_fields[hdr_pos], &ptr[IPV6_HDR_LEN], NTOHS(ipv6_buf->length));
|
|
|
|
comp_len = 2 + hdr_pos + payload_length;
|
|
|
|
return 1;
|
|
}
|
|
|
|
void lowpan_iphc_decoding(uint8_t *data, uint8_t length, net_if_eui64_t *s_addr,
|
|
net_if_eui64_t *d_addr)
|
|
{
|
|
uint8_t hdr_pos = 0;
|
|
uint8_t *ipv6_hdr_fields = data;
|
|
uint8_t lowpan_iphc[2];
|
|
uint8_t cid = 0;
|
|
uint8_t sci = 0;
|
|
uint8_t dci = 0;
|
|
|
|
uint8_t ll_prefix[2] = {0xfe, 0x80};
|
|
uint8_t m_prefix[2] = {0xff, 0x02};
|
|
lowpan_context_t *con = NULL;
|
|
|
|
ipv6_buf = ipv6_get_buf();
|
|
|
|
lowpan_iphc[0] = ipv6_hdr_fields[0];
|
|
lowpan_iphc[1] = ipv6_hdr_fields[1];
|
|
hdr_pos += 2;
|
|
|
|
/* first check if CID flag is set */
|
|
if (lowpan_iphc[1] & SIXLOWPAN_IPHC2_CID) {
|
|
hdr_pos++;
|
|
cid = 1;
|
|
}
|
|
|
|
/* TF: Traffic Class, Flow Label: */
|
|
if (lowpan_iphc[0] & SIXLOWPAN_IPHC1_FL_C) {
|
|
/* flowlabel is elided */
|
|
if (lowpan_iphc[0] & SIXLOWPAN_IPHC1_TC_C) {
|
|
/* traffic class is elided */
|
|
ipv6_buf->version_trafficclass = 0x60;
|
|
ipv6_buf->trafficclass_flowlabel = 0;
|
|
ipv6_buf->flowlabel = 0;
|
|
}
|
|
else {
|
|
/* toogle ecn/dscp order */
|
|
ipv6_buf->version_trafficclass = 0x60 | (0x0f &
|
|
(ipv6_hdr_fields[hdr_pos] >> 2));
|
|
ipv6_buf->trafficclass_flowlabel = ((ipv6_hdr_fields[hdr_pos] >> 2) & 0x30) |
|
|
((ipv6_hdr_fields[hdr_pos] << 6) & 0xc0);
|
|
ipv6_buf->flowlabel = 0;
|
|
hdr_pos += 3;
|
|
}
|
|
}
|
|
else {
|
|
/* flowlabel carried inline */
|
|
if (lowpan_iphc[0] & SIXLOWPAN_IPHC1_TC_C) {
|
|
/* traffic class is elided */
|
|
ipv6_buf->version_trafficclass = 0x60;
|
|
/* ecn + 4 bit flowlabel*/
|
|
ipv6_buf->trafficclass_flowlabel = ((ipv6_hdr_fields[hdr_pos] >> 2) & 0x30) |
|
|
(ipv6_hdr_fields[hdr_pos] & 0x0f);
|
|
hdr_pos++;
|
|
/* copy 2byte flowlabel */
|
|
memcpy(&ipv6_buf->flowlabel, &ipv6_hdr_fields[hdr_pos], 2);
|
|
hdr_pos += 2;
|
|
}
|
|
else {
|
|
ipv6_buf->version_trafficclass = 0x60 | (0x0f &
|
|
(ipv6_hdr_fields[hdr_pos] >> 2));
|
|
ipv6_buf->trafficclass_flowlabel = ((ipv6_hdr_fields[hdr_pos] >> 2) & 0x30) |
|
|
(ipv6_hdr_fields[hdr_pos] & 0x0f) |
|
|
(ipv6_hdr_fields[hdr_pos + 1] & 0x0f);
|
|
hdr_pos += 2;
|
|
memcpy(&ipv6_buf->trafficclass_flowlabel,
|
|
&ipv6_hdr_fields[hdr_pos], 2);
|
|
hdr_pos += 2;
|
|
}
|
|
}
|
|
|
|
/* NH: Next Header: */
|
|
if (lowpan_iphc[0] & SIXLOWPAN_IPHC1_NH) {
|
|
// TODO: next header decompression
|
|
}
|
|
else {
|
|
ipv6_buf->nextheader = ipv6_hdr_fields[hdr_pos];
|
|
hdr_pos++;
|
|
}
|
|
|
|
/* HLIM: Hop Limit: */
|
|
if (lowpan_iphc[0] & 0x03) {
|
|
switch (lowpan_iphc[0] & 0x03) {
|
|
case (0x01): {
|
|
ipv6_buf->hoplimit = 1;
|
|
break;
|
|
}
|
|
|
|
case (0x02): {
|
|
ipv6_buf->hoplimit = 64;
|
|
break;
|
|
}
|
|
|
|
case (0x03): {
|
|
ipv6_buf->hoplimit = 255;
|
|
break;
|
|
}
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
else {
|
|
ipv6_buf->hoplimit = ipv6_hdr_fields[hdr_pos];
|
|
hdr_pos++;
|
|
}
|
|
|
|
/* CID: Context Identifier Extension: + SAC: Source Address Compression */
|
|
if (lowpan_iphc[1] & SIXLOWPAN_IPHC2_SAC) {
|
|
/* 1: Source address compression uses stateful, context-based
|
|
* compression.*/
|
|
if (cid) {
|
|
sci = ipv6_hdr_fields[3] >> 4;
|
|
}
|
|
|
|
mutex_lock(&lowpan_context_mutex);
|
|
|
|
/* check context number */
|
|
if (((lowpan_iphc[1] & SIXLOWPAN_IPHC2_SAM) >> 4) & 0x03) {
|
|
con = lowpan_context_num_lookup(sci);
|
|
}
|
|
|
|
if (con == NULL) {
|
|
printf("ERROR: context not found\n");
|
|
return;
|
|
}
|
|
|
|
switch (((lowpan_iphc[1] & SIXLOWPAN_IPHC2_SAM) >> 4) & 0x03) {
|
|
case (0x01): {
|
|
/* 64-bits */
|
|
memcpy(&(ipv6_buf->srcaddr.uint8[8]), &ipv6_hdr_fields[hdr_pos], 8);
|
|
/* By RFC 6282 3.1.1. Bits covered by context
|
|
* information are always used. */
|
|
memcpy(&(ipv6_buf->srcaddr.uint8[0]), &con->prefix, con->length);
|
|
hdr_pos += 8;
|
|
break;
|
|
}
|
|
|
|
case (0x02): {
|
|
/* 16-bits */
|
|
memset(&(ipv6_buf->srcaddr.uint8[8]), 0, 6);
|
|
memcpy(&(ipv6_buf->srcaddr.uint8[14]), &ipv6_hdr_fields[hdr_pos], 2);
|
|
/* By RFC 6282 3.1.1. Bits covered by context
|
|
* information are always used. */
|
|
memcpy(&(ipv6_buf->srcaddr.uint8[0]), &con->prefix, con->length);
|
|
hdr_pos += 2;
|
|
break;
|
|
}
|
|
|
|
case (0x03): {
|
|
/* 0-bits */
|
|
memcpy(&(ipv6_buf->srcaddr.uint8[8]), &s_addr->uint8[0], 8);
|
|
/* By RFC 6282 3.1.1. Bits covered by context
|
|
* information are always used. */
|
|
memcpy(&(ipv6_buf->srcaddr.uint8[0]), &con->prefix, con->length);
|
|
break;
|
|
}
|
|
|
|
default: {
|
|
/* unspecified address */
|
|
memset(&(ipv6_buf->srcaddr.uint8[0]), 0, 16);
|
|
break;
|
|
}
|
|
}
|
|
|
|
mutex_unlock(&lowpan_context_mutex);
|
|
}
|
|
else {
|
|
switch (((lowpan_iphc[1] & SIXLOWPAN_IPHC2_SAM) >> 4) & 0x03) {
|
|
case (0x01): {
|
|
/* 64-bits */
|
|
memcpy(&(ipv6_buf->srcaddr.uint8[0]), &ll_prefix[0], 2);
|
|
memset(&(ipv6_buf->srcaddr.uint8[2]), 0, 6);
|
|
memcpy(&(ipv6_buf->srcaddr.uint8[8]), &ipv6_hdr_fields[hdr_pos], 8);
|
|
hdr_pos += 8;
|
|
break;
|
|
}
|
|
|
|
case (0x02): {
|
|
/* 16-bits */
|
|
memcpy(&(ipv6_buf->srcaddr.uint8[0]), &ll_prefix[0], 2);
|
|
memset(&(ipv6_buf->srcaddr.uint8[2]), 0, 12);
|
|
memcpy(&(ipv6_buf->srcaddr.uint8[14]), &ipv6_hdr_fields[hdr_pos], 2);
|
|
hdr_pos += 2;
|
|
break;
|
|
}
|
|
|
|
case (0x03): {
|
|
/* 0-bits */
|
|
memcpy(&(ipv6_buf->srcaddr.uint8[0]), &ll_prefix[0], 2);
|
|
memset(&(ipv6_buf->srcaddr.uint8[2]), 0, 20);
|
|
memcpy(&(ipv6_buf->srcaddr.uint8[8]), &s_addr->uint8[0], 8);
|
|
break;
|
|
}
|
|
|
|
default: {
|
|
/* full address carried inline */
|
|
memcpy(&(ipv6_buf->srcaddr.uint8[0]),
|
|
&ipv6_hdr_fields[hdr_pos], 16);
|
|
hdr_pos += 16;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* M: Multicast Compression + DAC: Destination Address Compression */
|
|
if (lowpan_iphc[1] & SIXLOWPAN_IPHC2_M) {
|
|
/* 1: Destination address is a multicast address. */
|
|
if (lowpan_iphc[1] & SIXLOWPAN_IPHC2_DAC) {
|
|
/* 1: Destination address compression uses stateful, context-based
|
|
* compression.
|
|
* If M=1 and DAC=1: */
|
|
if (cid) {
|
|
dci = ipv6_hdr_fields[3] & 0x0f;
|
|
}
|
|
|
|
mutex_lock(&lowpan_context_mutex);
|
|
|
|
if ((lowpan_iphc[1] & SIXLOWPAN_IPHC2_DAM) & 0x03) {
|
|
con = lowpan_context_num_lookup(dci);
|
|
}
|
|
|
|
if (con == NULL) {
|
|
printf("ERROR: context not found\n");
|
|
return;
|
|
}
|
|
|
|
// TODO:
|
|
mutex_unlock(&lowpan_context_mutex);
|
|
}
|
|
else {
|
|
/* If M=1 and DAC=0: */
|
|
switch (lowpan_iphc[1] & 0x03) {
|
|
case (0x01): {
|
|
m_prefix[1] = ipv6_hdr_fields[hdr_pos];
|
|
hdr_pos++;
|
|
break;
|
|
}
|
|
|
|
case (0x02): {
|
|
m_prefix[1] = ipv6_hdr_fields[hdr_pos];
|
|
hdr_pos++;
|
|
break;
|
|
}
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
switch (lowpan_iphc[1] & 0x03) {
|
|
case (0x01): {
|
|
memcpy(&(ipv6_buf->destaddr.uint8[0]), &m_prefix[0], 2);
|
|
memset(&(ipv6_buf->destaddr.uint8[2]), 0, 9);
|
|
memcpy(&(ipv6_buf->destaddr.uint8[11]), &ipv6_hdr_fields[hdr_pos], 5);
|
|
hdr_pos += 5;
|
|
break;
|
|
}
|
|
|
|
case (0x02): {
|
|
memcpy(&(ipv6_buf->destaddr.uint8[0]), &m_prefix[0], 2);
|
|
memset(&(ipv6_buf->destaddr.uint8[2]), 0, 11);
|
|
memcpy(&(ipv6_buf->destaddr.uint8[13]), &ipv6_hdr_fields[hdr_pos], 3);
|
|
hdr_pos += 3;
|
|
break;
|
|
}
|
|
|
|
case (0x03): {
|
|
memcpy(&(ipv6_buf->destaddr.uint8[0]), &m_prefix[0], 2);
|
|
memset(&(ipv6_buf->destaddr.uint8[2]), 0, 13);
|
|
memcpy(&(ipv6_buf->destaddr.uint8[15]), &ipv6_hdr_fields[hdr_pos], 1);
|
|
hdr_pos++;
|
|
break;
|
|
}
|
|
|
|
default: {
|
|
memcpy(&(ipv6_buf->destaddr.uint8[0]), &ipv6_hdr_fields[hdr_pos], 16);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
if (lowpan_iphc[1] & SIXLOWPAN_IPHC2_DAC) {
|
|
/* 1: Destination address compression uses stateful, context-based
|
|
* compression.
|
|
* If M=1 and DAC=1: */
|
|
if (cid) {
|
|
dci = ipv6_hdr_fields[3] & 0x0f;
|
|
}
|
|
|
|
mutex_lock(&lowpan_context_mutex);
|
|
|
|
if ((lowpan_iphc[1] & SIXLOWPAN_IPHC2_DAM) & 0x03) {
|
|
con = lowpan_context_num_lookup(dci);
|
|
}
|
|
|
|
if (con == NULL) {
|
|
printf("ERROR: context not found\n");
|
|
return;
|
|
}
|
|
|
|
switch ((lowpan_iphc[1] & SIXLOWPAN_IPHC2_DAM) & 0x03) {
|
|
case (0x01): {
|
|
memcpy(&(ipv6_buf->destaddr.uint8[8]), &ipv6_hdr_fields[hdr_pos], 8);
|
|
/* By draft-ietf-6lowpan-hc-15 3.1.1. Bits covered by context information are always used. */
|
|
memcpy(&(ipv6_buf->srcaddr.uint8[0]), &con->prefix, con->length);
|
|
hdr_pos += 8;
|
|
break;
|
|
}
|
|
|
|
case (0x02): {
|
|
memset(&(ipv6_buf->destaddr.uint8[8]), 0, 6);
|
|
memcpy(&(ipv6_buf->destaddr.uint8[14]), &ipv6_hdr_fields[hdr_pos], 2);
|
|
/* By draft-ietf-6lowpan-hc-15 3.1.1. Bits covered by context information are always used. */
|
|
memcpy(&(ipv6_buf->srcaddr.uint8[0]), &con->prefix, con->length);
|
|
hdr_pos += 2;
|
|
break;
|
|
}
|
|
|
|
case (0x03): {
|
|
memset(&(ipv6_buf->destaddr.uint8[0]), 0, 8);
|
|
memcpy(&(ipv6_buf->destaddr.uint8[8]), &d_addr->uint8[0], 8);
|
|
/* By draft-ietf-6lowpan-hc-15 3.1.1. Bits covered by context information are always used. */
|
|
memcpy(&(ipv6_buf->srcaddr.uint8[0]), &con->prefix, con->length);
|
|
break;
|
|
}
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
mutex_unlock(&lowpan_context_mutex);
|
|
}
|
|
else {
|
|
switch ((lowpan_iphc[1] & SIXLOWPAN_IPHC2_DAM) & 0x03) {
|
|
case (0x01): {
|
|
memcpy(&(ipv6_buf->destaddr.uint8[0]), &ll_prefix[0], 2);
|
|
memset(&(ipv6_buf->destaddr.uint8[2]), 0, 6);
|
|
memcpy(&(ipv6_buf->destaddr.uint8[8]),
|
|
&ipv6_hdr_fields[hdr_pos], 8);
|
|
hdr_pos += 8;
|
|
break;
|
|
}
|
|
|
|
case (0x02): {
|
|
memcpy(&(ipv6_buf->destaddr.uint8[0]), &ll_prefix[0], 2);
|
|
memset(&(ipv6_buf->destaddr.uint8[2]), 0, 12);
|
|
memcpy(&(ipv6_buf->destaddr.uint8[14]),
|
|
&ipv6_hdr_fields[hdr_pos], 2);
|
|
hdr_pos += 2;
|
|
break;
|
|
}
|
|
|
|
case (0x03): {
|
|
memcpy(&(ipv6_buf->destaddr.uint8[0]), &ll_prefix, 2);
|
|
memset(&(ipv6_buf->destaddr.uint8[2]), 0, 14);
|
|
memcpy(&(ipv6_buf->destaddr.uint8[8]), &d_addr->uint8[0], 8);
|
|
break;
|
|
}
|
|
|
|
default: {
|
|
memcpy(&(ipv6_buf->destaddr.uint8[0]),
|
|
&ipv6_hdr_fields[hdr_pos], 16);
|
|
hdr_pos += 16;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
uint8_t *ptr = get_payload_buf(ipv6_ext_hdr_len);
|
|
|
|
memcpy(ptr, &ipv6_hdr_fields[hdr_pos], length - hdr_pos);
|
|
|
|
/* ipv6 length */
|
|
ipv6_buf->length = HTONS(length - hdr_pos);
|
|
packet_length = IPV6_HDR_LEN + ipv6_buf->length;
|
|
}
|
|
|
|
uint8_t lowpan_context_len()
|
|
{
|
|
return context_len;
|
|
}
|
|
|
|
void lowpan_context_remove(uint8_t num)
|
|
{
|
|
int i, j;
|
|
|
|
for (i = 0; i < NDP_6LOWPAN_CONTEXT_MAX; i++) {
|
|
if (contexts[i].num == num) {
|
|
context_len--;
|
|
break;
|
|
}
|
|
}
|
|
|
|
abr_remove_context(num);
|
|
|
|
for (j = i; j < NDP_6LOWPAN_CONTEXT_MAX; j++) {
|
|
contexts[j] = contexts[j + 1];
|
|
}
|
|
}
|
|
|
|
lowpan_context_t *lowpan_context_update(uint8_t num, const ipv6_addr_t *prefix,
|
|
uint8_t length, uint8_t comp,
|
|
uint16_t lifetime)
|
|
{
|
|
lowpan_context_t *context;
|
|
|
|
if (lifetime == 0) {
|
|
lowpan_context_remove(num);
|
|
return NULL;
|
|
}
|
|
|
|
if (context_len == NDP_6LOWPAN_CONTEXT_MAX) {
|
|
return NULL;
|
|
}
|
|
|
|
context = lowpan_context_num_lookup(num);
|
|
|
|
if (context == NULL) {
|
|
context = &(contexts[context_len++]);
|
|
}
|
|
|
|
context->num = num;
|
|
memset((void *)(&context->prefix), 0, 16);
|
|
// length in bits
|
|
memcpy((void *)(&context->prefix), (void *)prefix, length / 8);
|
|
context->length = length;
|
|
context->comp = comp;
|
|
context->lifetime = lifetime;
|
|
return context;
|
|
}
|
|
|
|
lowpan_context_t *lowpan_context_get()
|
|
{
|
|
return contexts;
|
|
}
|
|
|
|
lowpan_context_t *lowpan_context_lookup(ipv6_addr_t *addr)
|
|
{
|
|
int i;
|
|
|
|
lowpan_context_t *context = NULL;
|
|
|
|
for (i = 0; i < lowpan_context_len(); i++) {
|
|
if (contexts[i].length > 0 && memcmp((void *)addr, &(contexts[i].prefix),
|
|
contexts[i].length) == 0) {
|
|
/* longer prefixes are always prefered */
|
|
if (context == NULL || context->length < contexts[i].length) {
|
|
context = &contexts[i];
|
|
}
|
|
}
|
|
}
|
|
|
|
return context;
|
|
}
|
|
|
|
lowpan_context_t *lowpan_context_num_lookup(uint8_t num)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < lowpan_context_len(); i++) {
|
|
if (contexts[i].num == num) {
|
|
return &contexts[i];
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void lowpan_context_auto_remove(void)
|
|
{
|
|
timex_t minute = timex_set(60, 0);
|
|
int i;
|
|
int8_t to_remove[NDP_6LOWPAN_CONTEXT_MAX];
|
|
int8_t to_remove_size;
|
|
|
|
while (1) {
|
|
vtimer_sleep(minute);
|
|
to_remove_size = 0;
|
|
mutex_lock(&lowpan_context_mutex);
|
|
|
|
for (i = 0; i < lowpan_context_len(); i++) {
|
|
if (--(contexts[i].lifetime) == 0) {
|
|
to_remove[to_remove_size++] = contexts[i].num;
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < to_remove_size; i++) {
|
|
lowpan_context_remove(to_remove[i]);
|
|
}
|
|
|
|
mutex_unlock(&lowpan_context_mutex);
|
|
}
|
|
}
|
|
|
|
void init_reas_bufs(lowpan_reas_buf_t *buf)
|
|
{
|
|
memset(&buf->s_addr, 0, sizeof(net_if_eui64_t));
|
|
memset(&buf->d_addr, 0, sizeof(net_if_eui64_t));
|
|
buf->tag = 0;
|
|
memset(&buf->timestamp, 0, sizeof(timex_t));
|
|
buf->packet_size = 0;
|
|
buf->current_packet_size = 0;
|
|
buf->packet = NULL;
|
|
buf->interval_list_head = NULL;
|
|
buf->next = NULL;
|
|
}
|
|
|
|
int sixlowpan_lowpan_init_adhoc_interface(int if_id, const ipv6_addr_t *prefix)
|
|
{
|
|
ipv6_addr_t tmp;
|
|
|
|
/* if prefix is set */
|
|
if (prefix != NULL) {
|
|
/* init network prefix */
|
|
ipv6_addr_t save_prefix;
|
|
|
|
ipv6_addr_init_prefix(&save_prefix, prefix, 64);
|
|
ndp_add_prefix_info(if_id, &save_prefix, 64,
|
|
NDP_OPT_PI_VLIFETIME_INFINITE,
|
|
NDP_OPT_PI_PLIFETIME_INFINITE, 1,
|
|
ICMPV6_NDP_OPT_PI_FLAG_AUTONOM);
|
|
ipv6_init_as_router();
|
|
|
|
/* add global address */
|
|
if (!ipv6_addr_set_by_eui64(&tmp, if_id, prefix)) {
|
|
return 0;
|
|
}
|
|
|
|
DEBUG("%s, %d: set unique address for interface %d to %s, according "
|
|
"to prefix %s\n", __FILE__, __LINE__, if_id,
|
|
ipv6_addr_to_str(addr_str, IPV6_MAX_ADDR_STR_LEN, &tmp),
|
|
ipv6_addr_to_str(addr_str, IPV6_MAX_ADDR_STR_LEN, &tmp));
|
|
|
|
if (!ipv6_net_if_add_addr(if_id, &tmp, NDP_ADDR_STATE_PREFERRED,
|
|
NDP_OPT_PI_VLIFETIME_INFINITE,
|
|
NDP_OPT_PI_PLIFETIME_INFINITE, 0)) {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return sixlowpan_lowpan_init_interface(if_id);
|
|
}
|
|
|
|
int sixlowpan_lowpan_init_interface(int if_id)
|
|
{
|
|
ipv6_addr_t tmp;
|
|
|
|
/* init link-local prefix */
|
|
ipv6_addr_set_link_local_prefix(&tmp);
|
|
|
|
if (!ipv6_addr_set_by_eui64(&tmp, if_id, &tmp)) {
|
|
DEBUG("Can not set link-local by EUI-64 on interface %d\n", if_id);
|
|
return 0;
|
|
}
|
|
|
|
DEBUG("%s, %d: sixlowpan_lowpan_init(): add link local address to "
|
|
"interface %d: %s\n", __FILE__, __LINE__, if_id,
|
|
ipv6_addr_to_str(addr_str, IPV6_MAX_ADDR_STR_LEN, &tmp));
|
|
|
|
if (!ipv6_net_if_add_addr(if_id, &tmp, NDP_ADDR_STATE_PREFERRED,
|
|
NDP_OPT_PI_VLIFETIME_INFINITE,
|
|
NDP_OPT_PI_PLIFETIME_INFINITE, 0)) {
|
|
DEBUG("Can not add link-local address to interface %d\n", if_id);
|
|
return 0;
|
|
}
|
|
|
|
/* add solicited nodes multicast address of link local address */
|
|
ipv6_addr_set_solicited_node_addr(&tmp, &tmp);
|
|
DEBUG("%s, %d: sixlowpan_lowpan_init(): add solicited nodes multicast address "
|
|
"to of link layer address interface %d: %s\n", __FILE__, __LINE__,
|
|
if_id, ipv6_addr_to_str(addr_str, IPV6_MAX_ADDR_STR_LEN, &tmp));
|
|
|
|
if (!ipv6_net_if_add_addr(if_id, &tmp, NDP_ADDR_STATE_PREFERRED,
|
|
NDP_OPT_PI_VLIFETIME_INFINITE,
|
|
NDP_OPT_PI_PLIFETIME_INFINITE, 0)) {
|
|
DEBUG("Can not add all nodes address to interface %d\n", if_id);
|
|
return 0;
|
|
}
|
|
|
|
/* add all nodes multicast address */
|
|
ipv6_addr_set_all_nodes_addr(&tmp);
|
|
DEBUG("%s, %d: sixlowpan_lowpan_init(): add all nodes multicast address "
|
|
"to interface %d: %s\n", __FILE__, __LINE__, if_id,
|
|
ipv6_addr_to_str(addr_str, IPV6_MAX_ADDR_STR_LEN, &tmp));
|
|
|
|
if (!ipv6_net_if_add_addr(if_id, &tmp, NDP_ADDR_STATE_PREFERRED,
|
|
NDP_OPT_PI_VLIFETIME_INFINITE,
|
|
NDP_OPT_PI_PLIFETIME_INFINITE, 0)) {
|
|
DEBUG("Can not add all nodes address to interface %d\n", if_id);
|
|
return 0;
|
|
}
|
|
|
|
/* add loopback address */
|
|
ipv6_addr_set_loopback_addr(&tmp);
|
|
DEBUG("%s, %d: sixlowpan_lowpan_init(): add loopback address to interface "
|
|
"0: %s\n", __FILE__, __LINE__,
|
|
ipv6_addr_to_str(addr_str, IPV6_MAX_ADDR_STR_LEN, &tmp));
|
|
|
|
if (!ipv6_net_if_add_addr(0, &tmp, NDP_ADDR_STATE_PREFERRED,
|
|
NDP_OPT_PI_VLIFETIME_INFINITE,
|
|
NDP_OPT_PI_PLIFETIME_INFINITE, 0)) {
|
|
DEBUG("Can not add all nodes address to interface %d\n", if_id);
|
|
return 0;
|
|
}
|
|
|
|
icmpv6_send_router_sol(OPT_SLLAO);
|
|
|
|
return 1;
|
|
}
|
|
|
|
int sixlowpan_lowpan_init(void)
|
|
{
|
|
short i;
|
|
|
|
/* init mac-layer and radio transceiver */
|
|
sixlowpan_mac_init();
|
|
|
|
/* init lowpan context mutex */
|
|
mutex_init(&lowpan_context_mutex);
|
|
|
|
/* init packet_fifo mutex */
|
|
mutex_init(&fifo_mutex);
|
|
|
|
if (!ip_process_pid) {
|
|
ip_process_pid = thread_create(ip_process_buf, IP_PROCESS_STACKSIZE,
|
|
PRIORITY_MAIN - 1, CREATE_STACKTEST,
|
|
ipv6_process, "ip_process");
|
|
}
|
|
|
|
if (ip_process_pid < 0) {
|
|
return 0;
|
|
}
|
|
|
|
nbr_cache_auto_rem();
|
|
|
|
contexts_rem_pid = thread_create(con_buf, CON_STACKSIZE,
|
|
PRIORITY_MAIN + 1, CREATE_STACKTEST,
|
|
lowpan_context_auto_remove, "lowpan_context_rem");
|
|
|
|
if (contexts_rem_pid < 0) {
|
|
return 0;
|
|
}
|
|
|
|
transfer_pid = thread_create(lowpan_transfer_buf, LOWPAN_TRANSFER_BUF_STACKSIZE,
|
|
PRIORITY_MAIN - 1, CREATE_STACKTEST,
|
|
lowpan_transfer, "lowpan_transfer");
|
|
|
|
if (transfer_pid < 0) {
|
|
return 0;
|
|
}
|
|
|
|
for (i = 0; i < SIXLOWIP_MAX_REGISTERED; i++) {
|
|
sixlowip_reg[i] = 0;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
void sixlowpan_lowpan_bootstrapping(void)
|
|
{
|
|
|
|
icmpv6_send_router_sol(OPT_SLLAO);
|
|
}
|