2013-06-22 05:11:53 +02:00
|
|
|
/**
|
2013-08-08 13:39:00 +02:00
|
|
|
* IPv6 implementation
|
2013-06-22 05:11:53 +02:00
|
|
|
*
|
|
|
|
* Copyright (C) 2013 INRIA.
|
|
|
|
*
|
|
|
|
* This file 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 sixlowip.c
|
2013-08-08 13:39:00 +02:00
|
|
|
* @brief 6lowpan IP layer functions
|
2013-06-22 05:11:53 +02:00
|
|
|
* @author Stephan Zeisberg <zeisberg@mi.fu-berlin.de>
|
|
|
|
* @author Martin Lenders <mlenders@inf.fu-berlin.de>
|
|
|
|
* @author Eric Engel <eric.engel@fu-berlin.de>
|
|
|
|
* @author Oliver Gesch <oliver.gesch@googlemail.com>
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2012-11-05 22:29:11 +01:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
2013-07-27 15:04:51 +02:00
|
|
|
#include <errno.h>
|
2013-08-08 13:39:00 +02:00
|
|
|
|
|
|
|
#include "vtimer.h"
|
|
|
|
#include "mutex.h"
|
2012-11-05 22:29:11 +01:00
|
|
|
#include "msg.h"
|
2013-08-08 16:22:37 +02:00
|
|
|
#include "sixlowpan/mac.h"
|
2013-08-08 13:39:00 +02:00
|
|
|
|
2013-08-05 16:10:54 +02:00
|
|
|
#include "ip.h"
|
|
|
|
#include "icmp.h"
|
|
|
|
#include "lowpan.h"
|
2013-08-08 13:39:00 +02:00
|
|
|
|
2012-11-05 22:29:11 +01:00
|
|
|
#include "sys/net/destiny/socket.h"
|
|
|
|
#include "sys/net/net_help/net_help.h"
|
|
|
|
#include "sys/net/net_help/msg_help.h"
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
#define IP_PKT_RECV_BUF_SIZE (64)
|
|
|
|
#define LLHDR_IPV6HDR_LEN (LL_HDR_LEN + IPV6_HDR_LEN)
|
|
|
|
|
2012-11-05 22:29:11 +01:00
|
|
|
uint8_t ip_send_buffer[BUFFER_SIZE];
|
|
|
|
uint8_t buffer[BUFFER_SIZE];
|
|
|
|
msg_t msg_queue[IP_PKT_RECV_BUF_SIZE];
|
2013-06-24 14:11:30 +02:00
|
|
|
ipv6_hdr_t *ipv6_buf;
|
2013-08-13 06:41:05 +02:00
|
|
|
icmpv6_hdr_t *icmp_buf;
|
2012-11-05 22:29:11 +01:00
|
|
|
uint8_t *nextheader;
|
2013-08-13 06:41:05 +02:00
|
|
|
|
2012-11-05 22:29:11 +01:00
|
|
|
uint8_t iface_addr_list_count = 0;
|
|
|
|
int udp_packet_handler_pid = 0;
|
|
|
|
int tcp_packet_handler_pid = 0;
|
|
|
|
int rpl_process_pid = 0;
|
|
|
|
|
2013-07-27 15:04:51 +02:00
|
|
|
/* registered upper layer threads */
|
|
|
|
int sixlowip_reg[SIXLOWIP_MAX_REGISTERED];
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
void ipv6_send_bytes(ipv6_hdr_t *bytes)
|
2013-08-12 11:08:19 +02:00
|
|
|
{
|
2013-08-13 06:41:05 +02:00
|
|
|
uint16_t offset = IPV6_HDR_LEN + HTONS(bytes->length);
|
2013-08-12 11:08:19 +02:00
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
bytes->flowlabel = HTONS(bytes->flowlabel);
|
|
|
|
bytes->length = HTONS(bytes->length);
|
2013-08-12 11:08:19 +02:00
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
memset(bytes, 0, BUFFER_SIZE);
|
|
|
|
memcpy(bytes + LL_HDR_LEN, bytes, offset);
|
2013-08-12 11:08:19 +02:00
|
|
|
|
2013-08-15 11:27:30 +02:00
|
|
|
sixlowpan_lowpan_sendto((ieee_802154_long_t *) &bytes->destaddr.uint16[4],
|
2013-08-13 06:41:05 +02:00
|
|
|
(uint8_t *)bytes,
|
2013-08-12 11:08:19 +02:00
|
|
|
offset);
|
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
ipv6_hdr_t *ipv6_get_buf_send(void)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-08-15 11:27:30 +02:00
|
|
|
return ((ipv6_hdr_t *) &ip_send_buffer[LL_HDR_LEN]);
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
uint8_t *get_payload_buf_send(uint8_t ext_len)
|
|
|
|
{
|
2012-11-05 22:29:11 +01:00
|
|
|
return &(ip_send_buffer[LLHDR_IPV6HDR_LEN + ext_len]);
|
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
ipv6_hdr_t *ipv6_get_buf(void)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-08-15 11:27:30 +02:00
|
|
|
return ((ipv6_hdr_t *) &buffer[LL_HDR_LEN]);
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
icmpv6_hdr_t *get_icmpv6_buf(uint8_t ext_len)
|
|
|
|
{
|
2013-08-15 11:27:30 +02:00
|
|
|
return ((icmpv6_hdr_t *) &buffer[LLHDR_IPV6HDR_LEN + ext_len]);
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
uint8_t *get_payload_buf(uint8_t ext_len)
|
|
|
|
{
|
2012-11-05 22:29:11 +01:00
|
|
|
return &(buffer[LLHDR_IPV6HDR_LEN + ext_len]);
|
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
void ipv6_sendto(const ipv6_addr_t *dest, uint8_t next_header,
|
|
|
|
const uint8_t *payload, uint16_t payload_length)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
|
|
|
uint8_t *p_ptr;
|
2013-08-09 03:13:37 +02:00
|
|
|
uint16_t packet_length;
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
if (next_header == IPV6_PROTO_NUM_TCP) {
|
2013-06-22 05:11:53 +02:00
|
|
|
p_ptr = get_payload_buf_send(ipv6_ext_hdr_len);
|
2013-08-13 06:41:05 +02:00
|
|
|
ipv6_buf = ipv6_get_buf_send();
|
2013-06-22 05:11:53 +02:00
|
|
|
}
|
|
|
|
else {
|
2013-08-13 06:41:05 +02:00
|
|
|
ipv6_buf = ipv6_get_buf();
|
2013-06-22 05:11:53 +02:00
|
|
|
p_ptr = get_payload_buf(ipv6_ext_hdr_len);
|
|
|
|
}
|
|
|
|
|
2012-11-05 22:29:11 +01:00
|
|
|
ipv6_buf->version_trafficclass = IPV6_VER;
|
|
|
|
ipv6_buf->trafficclass_flowlabel = 0;
|
|
|
|
ipv6_buf->flowlabel = 0;
|
|
|
|
ipv6_buf->nextheader = next_header;
|
|
|
|
ipv6_buf->hoplimit = MULTIHOP_HOPLIMIT;
|
2013-08-13 06:41:05 +02:00
|
|
|
ipv6_buf->length = payload_length;
|
2012-11-05 22:29:11 +01:00
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
memcpy(&(ipv6_buf->destaddr), dest, 16);
|
|
|
|
ipv6_iface_get_best_src_addr(&(ipv6_buf->srcaddr), &(ipv6_buf->destaddr));
|
2012-11-05 22:29:11 +01:00
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
memcpy(p_ptr, payload, payload_length);
|
2012-11-05 22:29:11 +01:00
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
packet_length = IPV6_HDR_LEN + payload_length;
|
2012-11-05 22:29:11 +01:00
|
|
|
|
2013-08-15 11:27:30 +02:00
|
|
|
sixlowpan_lowpan_sendto((ieee_802154_long_t *) &ipv6_buf->destaddr.uint16[4],
|
2013-08-09 03:13:37 +02:00
|
|
|
(uint8_t *)ipv6_buf, packet_length);
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-07-27 15:04:51 +02:00
|
|
|
/* Register an upper layer thread */
|
2013-08-13 06:41:05 +02:00
|
|
|
uint8_t ipv6_register_packet_handler(int pid)
|
2013-07-27 15:04:51 +02:00
|
|
|
{
|
|
|
|
uint8_t i;
|
|
|
|
|
2013-09-23 12:41:39 +02:00
|
|
|
for (i = 0; ((i < SIXLOWIP_MAX_REGISTERED) && (sixlowip_reg[i] != pid) &&
|
2013-08-09 03:13:37 +02:00
|
|
|
(sixlowip_reg[i] != 0)); i++) {
|
|
|
|
;
|
|
|
|
}
|
2013-07-27 15:04:51 +02:00
|
|
|
|
|
|
|
if (i >= SIXLOWIP_MAX_REGISTERED) {
|
|
|
|
return ENOMEM;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
sixlowip_reg[i] = pid;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
int icmpv6_demultiplex(const icmpv6_hdr_t *hdr)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-08-08 13:39:00 +02:00
|
|
|
switch (hdr->type) {
|
2013-08-14 04:35:58 +02:00
|
|
|
case (ICMPV6_TYPE_ECHO_REQUEST): {
|
2013-07-24 17:06:25 +02:00
|
|
|
puts("INFO: packet type: icmp echo request");
|
|
|
|
/* processing echo request */
|
|
|
|
recv_echo_req();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-08-14 04:35:58 +02:00
|
|
|
case (ICMPV6_TYPE_ECHO_REPLY): {
|
2013-07-24 17:06:25 +02:00
|
|
|
puts("INFO: packet type: icmp echo reply");
|
|
|
|
/* processing echo reply */
|
|
|
|
recv_echo_repl();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-08-14 04:35:58 +02:00
|
|
|
case (ICMPV6_TYPE_ROUTER_SOL): {
|
2012-11-05 22:29:11 +01:00
|
|
|
puts("INFO: packet type: icmp router solicitation");
|
|
|
|
/* processing router solicitation */
|
|
|
|
recv_rtr_sol();
|
|
|
|
/* init solicited router advertisment*/
|
|
|
|
break;
|
|
|
|
}
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-08-14 04:35:58 +02:00
|
|
|
case (ICMPV6_TYPE_ROUTER_ADV): {
|
2012-11-05 22:29:11 +01:00
|
|
|
puts("INFO: packet type: icmp router advertisment");
|
|
|
|
/* processing router advertisment */
|
|
|
|
recv_rtr_adv();
|
|
|
|
/* init neighbor solicitation */
|
|
|
|
break;
|
|
|
|
}
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-08-14 04:35:58 +02:00
|
|
|
case (ICMPV6_TYPE_NEIGHBOR_SOL): {
|
2012-11-05 22:29:11 +01:00
|
|
|
puts("INFO: packet type: icmp neighbor solicitation");
|
|
|
|
recv_nbr_sol();
|
|
|
|
break;
|
|
|
|
}
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-08-14 04:35:58 +02:00
|
|
|
case (ICMPV6_TYPE_NEIGHBOR_ADV): {
|
2012-11-05 22:29:11 +01:00
|
|
|
puts("INFO: packet type: icmp neighbor advertisment");
|
|
|
|
recv_nbr_adv();
|
|
|
|
break;
|
|
|
|
}
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-08-14 04:35:58 +02:00
|
|
|
case (ICMPV6_TYPE_RPL_CONTROL): {
|
2013-06-22 05:11:53 +02:00
|
|
|
puts("INFO: packet type: RPL message");
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (rpl_process_pid != 0) {
|
2013-06-22 05:11:53 +02:00
|
|
|
msg_t m_send;
|
|
|
|
m_send.content.ptr = (char *) &hdr->code;
|
|
|
|
msg_send(&m_send, rpl_process_pid, 1);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("INFO: no RPL handler registered");
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2012-11-05 22:29:11 +01:00
|
|
|
default:
|
|
|
|
return -1;
|
|
|
|
}
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2012-11-05 22:29:11 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
uint8_t ipv6_get_addr_match(const ipv6_addr_t *src,
|
|
|
|
const ipv6_addr_t *dst)
|
|
|
|
{
|
|
|
|
uint8_t val = 0, xor;
|
|
|
|
|
|
|
|
for (int i = 0; i < 16; 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;
|
|
|
|
}
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
void ipv6_process(void)
|
|
|
|
{
|
|
|
|
msg_t m_recv_lowpan, m_send_lowpan;
|
|
|
|
msg_t m_recv, m_send;
|
|
|
|
ipv6_addr_t myaddr;
|
2013-07-27 15:04:51 +02:00
|
|
|
uint8_t i;
|
2013-08-09 03:13:37 +02:00
|
|
|
uint16_t packet_length;
|
2013-07-27 15:04:51 +02:00
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
ipv6_addr_init(&myaddr, 0xabcd, 0x0, 0x0, 0x0, 0x3612, 0x00ff, 0xfe00,
|
|
|
|
sixlowpan_mac_get_radio_address());
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
while (1) {
|
2012-11-05 22:29:11 +01:00
|
|
|
msg_receive(&m_recv_lowpan);
|
|
|
|
|
2013-06-24 14:11:30 +02:00
|
|
|
ipv6_buf = (ipv6_hdr_t *)m_recv_lowpan.content.ptr;
|
2012-11-05 22:29:11 +01:00
|
|
|
|
|
|
|
/* identifiy packet */
|
|
|
|
nextheader = &ipv6_buf->nextheader;
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if ((ipv6_get_addr_match(&myaddr, &ipv6_buf->destaddr) >= 112) &&
|
2013-08-08 13:39:00 +02:00
|
|
|
(ipv6_buf->destaddr.uint8[15] != myaddr.uint8[15])) {
|
2013-08-09 03:13:37 +02:00
|
|
|
packet_length = IPV6_HDR_LEN + ipv6_buf->length;
|
2013-08-13 06:41:05 +02:00
|
|
|
memcpy(ipv6_get_buf_send(), ipv6_get_buf(), packet_length);
|
2013-08-15 11:27:30 +02:00
|
|
|
sixlowpan_lowpan_sendto((ieee_802154_long_t *) &ipv6_buf->destaddr.uint16[4],
|
2013-08-13 06:41:05 +02:00
|
|
|
(uint8_t *)ipv6_get_buf_send(),
|
2013-08-09 03:13:37 +02:00
|
|
|
packet_length);
|
2013-06-22 05:11:53 +02:00
|
|
|
}
|
|
|
|
else {
|
2013-07-27 15:04:51 +02:00
|
|
|
for (i = 0; i < SIXLOWIP_MAX_REGISTERED; i++) {
|
|
|
|
if (sixlowip_reg[i]) {
|
|
|
|
msg_t m_send;
|
|
|
|
m_send.content.ptr = (char *) &ipv6_buf;
|
|
|
|
msg_send(&m_send, sixlowip_reg[i], 1);
|
|
|
|
}
|
|
|
|
}
|
2013-08-09 03:13:37 +02:00
|
|
|
|
2013-08-08 13:39:00 +02:00
|
|
|
switch (*nextheader) {
|
2013-08-13 06:41:05 +02:00
|
|
|
case (IPV6_PROTO_NUM_ICMPV6): {
|
2013-06-22 05:11:53 +02:00
|
|
|
/* checksum test*/
|
2013-08-13 06:41:05 +02:00
|
|
|
if (icmpv6_csum(IPV6_PROTO_NUM_ICMPV6) != 0xffff) {
|
2013-06-22 05:11:53 +02:00
|
|
|
printf("ERROR: wrong checksum\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
icmp_buf = get_icmpv6_buf(ipv6_ext_hdr_len);
|
|
|
|
icmpv6_demultiplex(icmp_buf);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
case (IPV6_PROTO_NUM_TCP): {
|
2013-06-24 22:37:35 +02:00
|
|
|
if (tcp_packet_handler_pid != 0) {
|
2013-06-22 05:11:53 +02:00
|
|
|
m_send.content.ptr = (char *) ipv6_buf;
|
|
|
|
msg_send_receive(&m_send, &m_recv, tcp_packet_handler_pid);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
printf("INFO: No TCP handler registered.\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
case (IPV6_PROTO_NUM_UDP): {
|
2013-06-24 22:37:35 +02:00
|
|
|
if (udp_packet_handler_pid != 0) {
|
2013-06-22 05:11:53 +02:00
|
|
|
m_send.content.ptr = (char *) ipv6_buf;
|
|
|
|
msg_send_receive(&m_send, &m_recv, udp_packet_handler_pid);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
printf("INFO: No UDP handler registered.\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
case (IPV6_PROTO_NUM_NONE): {
|
2013-06-22 05:11:53 +02:00
|
|
|
printf("INFO: Packet with no Header following the IPv6 Header received.\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-05 22:29:11 +01:00
|
|
|
msg_reply(&m_recv_lowpan, &m_send_lowpan);
|
2013-06-22 05:11:53 +02:00
|
|
|
}
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
void ipv6_iface_add_addr(const ipv6_addr_t *addr, ipv6_addr_type_t type,
|
|
|
|
ndp_addr_state_t state, uint32_t val_ltime,
|
|
|
|
uint32_t pref_ltime)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-08-13 06:41:05 +02:00
|
|
|
if (ipv6_addr_is_unspecified(addr) == 128) {
|
2012-11-05 22:29:11 +01:00
|
|
|
printf("ERROR: unspecified address (::) can't be assigned to interface.\n");
|
|
|
|
return;
|
|
|
|
}
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (ipv6_iface_addr_match(addr) != 0) {
|
2012-11-05 22:29:11 +01:00
|
|
|
return;
|
|
|
|
}
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (iface_addr_list_count < IFACE_ADDR_LIST_LEN) {
|
2013-06-22 05:11:53 +02:00
|
|
|
memcpy(&(iface.addr_list[iface_addr_list_count].addr.uint8[0]),
|
|
|
|
&(addr->uint8[0]), 16);
|
2012-11-05 22:29:11 +01:00
|
|
|
iface.addr_list[iface_addr_list_count].state = state;
|
|
|
|
timex_t valtime = {val_ltime, 0};
|
|
|
|
timex_t preftime = {pref_ltime, 0};
|
2013-06-13 10:22:55 +02:00
|
|
|
timex_t now;
|
|
|
|
vtimer_now(&now);
|
2012-11-05 22:29:11 +01:00
|
|
|
iface.addr_list[iface_addr_list_count].val_ltime = timex_add(now, valtime);
|
|
|
|
iface.addr_list[iface_addr_list_count].pref_ltime = timex_add(now, preftime);
|
|
|
|
iface.addr_list[iface_addr_list_count].type = type;
|
|
|
|
iface_addr_list_count++;
|
2013-06-22 05:11:53 +02:00
|
|
|
|
|
|
|
/* Register to Solicited-Node multicast address according to RFC 4291 */
|
2013-08-13 06:41:05 +02:00
|
|
|
if (type == IPV6_ADDR_TYPE_ANYCAST || type == IPV6_ADDR_TYPE_LINK_LOCAL ||
|
|
|
|
type == IPV6_ADDR_TYPE_GLOBAL || type == IPV6_ADDR_TYPE_UNICAST) {
|
2012-11-05 22:29:11 +01:00
|
|
|
ipv6_addr_t sol_node_mcast_addr;
|
2013-08-13 06:41:05 +02:00
|
|
|
ipv6_addr_set_solicited_node_addr(&sol_node_mcast_addr, addr);
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (ipv6_iface_addr_match(&sol_node_mcast_addr) == NULL) {
|
2013-08-13 06:41:05 +02:00
|
|
|
ipv6_iface_add_addr(&sol_node_mcast_addr,
|
|
|
|
IPV6_ADDR_TYPE_SOLICITED_NODE,
|
|
|
|
state, val_ltime, pref_ltime);
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
addr_list_t *ipv6_iface_addr_match(const ipv6_addr_t *addr)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2012-11-05 22:29:11 +01:00
|
|
|
int i;
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
for (i = 0; i < iface_addr_list_count; i++) {
|
|
|
|
if (memcmp(&(iface.addr_list[i].addr.uint8[0]),
|
2013-08-08 13:39:00 +02:00
|
|
|
&(addr->uint8[0]), 16) == 0) {
|
2012-11-05 22:29:11 +01:00
|
|
|
return &(iface.addr_list[i]);
|
|
|
|
}
|
|
|
|
}
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2012-11-05 22:29:11 +01:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
addr_list_t *ipv6_iface_addr_prefix_eq(ipv6_addr_t *addr)
|
|
|
|
{
|
2012-11-05 22:29:11 +01:00
|
|
|
int i;
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
for (i = 0; i < iface_addr_list_count; i++) {
|
|
|
|
if (memcmp(&(iface.addr_list[i].addr.uint8[0]),
|
2013-08-08 13:39:00 +02:00
|
|
|
&(addr->uint8[0]), 8) == 0) {
|
2012-11-05 22:29:11 +01:00
|
|
|
return &(iface.addr_list[i]);
|
|
|
|
}
|
|
|
|
}
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2012-11-05 22:29:11 +01:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
void ipv6_iface_print_addrs(void)
|
|
|
|
{
|
2013-06-24 22:37:35 +02:00
|
|
|
for (int i = 0; i < iface_addr_list_count; i++) {
|
2013-08-13 06:41:05 +02:00
|
|
|
char addr_str[IPV6_MAX_ADDR_STR_LEN];
|
|
|
|
printf("%s\n", ipv6_addr_to_str(addr_str,
|
|
|
|
&(iface.addr_list[i].addr)));
|
2013-06-22 05:11:53 +02:00
|
|
|
}
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
void ipv6_addr_set_by_eui64(ipv6_addr_t *out, const ipv6_addr_t *prefix)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-08-13 06:41:05 +02:00
|
|
|
out->uint16[0] = prefix->uint16[0];
|
|
|
|
out->uint16[1] = prefix->uint16[1];
|
|
|
|
out->uint16[2] = prefix->uint16[2];
|
|
|
|
out->uint16[3] = prefix->uint16[3];
|
2012-11-05 22:29:11 +01:00
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
memcpy(&(out->uint8[8]), &(iface.laddr.uint8[0]), 8);
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
void ipv6_addr_init_prefix(ipv6_addr_t *out, const ipv6_addr_t *prefix,
|
|
|
|
uint8_t bits)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-08-13 06:41:05 +02:00
|
|
|
if (bits > 128) {
|
|
|
|
bits = 128;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t bytes = bits / 8, mask;
|
|
|
|
|
|
|
|
if (bits % 8) {
|
|
|
|
mask = 0xff << (bits - (bytes * 8));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
mask = 0x00;
|
|
|
|
}
|
|
|
|
|
|
|
|
bytes++;
|
|
|
|
memset(out, 0, 16);
|
|
|
|
memcpy(out, prefix, bytes);
|
|
|
|
out->uint8[bytes] = prefix->uint8[bytes] & mask;
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
void ipv6_addr_set_all_routers_addr(ipv6_addr_t *ipv6_addr)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-08-13 06:41:05 +02:00
|
|
|
ipv6_addr->uint16[0] = HTONS(0xff02);
|
|
|
|
ipv6_addr->uint16[1] = 0;
|
|
|
|
ipv6_addr->uint16[2] = 0;
|
|
|
|
ipv6_addr->uint16[3] = 0;
|
|
|
|
ipv6_addr->uint16[4] = 0;
|
|
|
|
ipv6_addr->uint16[5] = 0;
|
|
|
|
ipv6_addr->uint16[6] = 0;
|
|
|
|
ipv6_addr->uint16[7] = HTONS(0x0002);
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
void ipv6_addr_set_all_nodes_addr(ipv6_addr_t *ipv6_addr)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-08-13 06:41:05 +02:00
|
|
|
ipv6_addr->uint16[0] = HTONS(0xff02);
|
|
|
|
ipv6_addr->uint16[1] = 0;
|
|
|
|
ipv6_addr->uint16[2] = 0;
|
|
|
|
ipv6_addr->uint16[3] = 0;
|
|
|
|
ipv6_addr->uint16[4] = 0;
|
|
|
|
ipv6_addr->uint16[5] = 0;
|
|
|
|
ipv6_addr->uint16[6] = 0;
|
|
|
|
ipv6_addr->uint16[7] = HTONS(0x0001);
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
void ipv6_addr_set_loopback_addr(ipv6_addr_t *ipv6_addr)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-08-13 06:41:05 +02:00
|
|
|
ipv6_addr->uint16[0] = 0;
|
|
|
|
ipv6_addr->uint16[1] = 0;
|
|
|
|
ipv6_addr->uint16[2] = 0;
|
|
|
|
ipv6_addr->uint16[3] = 0;
|
|
|
|
ipv6_addr->uint16[4] = 0;
|
|
|
|
ipv6_addr->uint16[5] = 0;
|
|
|
|
ipv6_addr->uint16[6] = 0;
|
|
|
|
ipv6_addr->uint16[7] = HTONS(0x0001);
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
void ipv6_iface_get_best_src_addr(ipv6_addr_t *src, const ipv6_addr_t *dest)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2012-11-05 22:29:11 +01:00
|
|
|
/* try to find best match if dest is not mcast or link local */
|
|
|
|
int8_t itmp = -1;
|
2013-06-22 05:11:53 +02:00
|
|
|
uint8_t tmp = 0;
|
|
|
|
uint8_t bmatch = 0;
|
2012-11-05 22:29:11 +01:00
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
if (!(ipv6_addr_is_link_local(dest)) && !(ipv6_addr_is_multicast(dest))) {
|
2013-06-24 22:37:35 +02:00
|
|
|
for (int i = 0; i < IFACE_ADDR_LIST_LEN; i++) {
|
2013-08-13 06:41:05 +02:00
|
|
|
if (iface.addr_list[i].state == NDP_ADDR_STATE_PREFERRED) {
|
2013-08-14 17:04:17 +02:00
|
|
|
if (!ipv6_addr_is_link_local(&(iface.addr_list[i].addr)) &&
|
2013-08-14 16:21:01 +02:00
|
|
|
!ipv6_addr_is_multicast(&(iface.addr_list[i].addr)) &&
|
|
|
|
!ipv6_addr_is_unique_local_unicast(&(iface.addr_list[i].addr))) {
|
2013-08-13 06:41:05 +02:00
|
|
|
tmp = ipv6_get_addr_match(dest, &(iface.addr_list[i].addr));
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (tmp >= bmatch) {
|
2012-11-05 22:29:11 +01:00
|
|
|
bmatch = tmp;
|
|
|
|
itmp = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-06-22 05:11:53 +02:00
|
|
|
}
|
|
|
|
else {
|
2013-06-24 22:37:35 +02:00
|
|
|
for (int j = 0; j < IFACE_ADDR_LIST_LEN; j++) {
|
2013-08-13 06:41:05 +02:00
|
|
|
if ((iface.addr_list[j].state == NDP_ADDR_STATE_PREFERRED) &&
|
2013-08-14 16:21:01 +02:00
|
|
|
ipv6_addr_is_link_local(&(iface.addr_list[j].addr)) &&
|
|
|
|
!ipv6_addr_is_multicast(&(iface.addr_list[j].addr))) {
|
2013-06-22 05:11:53 +02:00
|
|
|
itmp = j;
|
|
|
|
}
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (itmp == -1) {
|
2012-11-05 22:29:11 +01:00
|
|
|
memset(src, 0, 16);
|
2013-06-22 05:11:53 +02:00
|
|
|
}
|
|
|
|
else {
|
2012-11-05 22:29:11 +01:00
|
|
|
memcpy(src, &(iface.addr_list[itmp].addr), 16);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
int ipv6_addr_is_equal(const ipv6_addr_t *a, const ipv6_addr_t *b)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-08-14 17:45:14 +02:00
|
|
|
return (ipv6_get_addr_match(a, b) == 128);
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
void ipv6_addr_set_link_local_prefix(ipv6_addr_t *ipv6_addr)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-08-13 06:41:05 +02:00
|
|
|
ipv6_addr->uint16[0] = HTONS(0xfe80);
|
|
|
|
ipv6_addr->uint16[1] = 0;
|
|
|
|
ipv6_addr->uint16[2] = 0;
|
|
|
|
ipv6_addr->uint16[3] = 0;
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
void ipv6_addr_init(ipv6_addr_t *out, uint16_t addr0, uint16_t addr1,
|
|
|
|
uint16_t addr2, uint16_t addr3, uint16_t addr4,
|
|
|
|
uint16_t addr5, uint16_t addr6, uint16_t addr7)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-08-13 06:41:05 +02:00
|
|
|
out->uint16[0] = HTONS(addr0);
|
|
|
|
out->uint16[1] = HTONS(addr1);
|
|
|
|
out->uint16[2] = HTONS(addr2);
|
|
|
|
out->uint16[3] = HTONS(addr3);
|
|
|
|
out->uint16[4] = HTONS(addr4);
|
|
|
|
out->uint16[5] = HTONS(addr5);
|
|
|
|
out->uint16[6] = HTONS(addr6);
|
|
|
|
out->uint16[7] = HTONS(addr7);
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
int ipv6_addr_is_link_local(const ipv6_addr_t *addr)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-09-23 14:56:15 +02:00
|
|
|
return (addr->uint16[0] == HTONS(0xfe80));
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-08-14 16:21:01 +02:00
|
|
|
int ipv6_addr_is_unique_local_unicast(const ipv6_addr_t *addr)
|
|
|
|
{
|
2013-08-14 17:45:14 +02:00
|
|
|
return (addr->uint8[0] == 0xfc || addr->uint8[0] == 0xfd);
|
2013-08-14 16:21:01 +02:00
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
int ipv6_addr_is_multicast(const ipv6_addr_t *addr)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-08-14 17:45:14 +02:00
|
|
|
return (addr->uint8[0] == 0xff);
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
int ipv6_addr_is_unspecified(const ipv6_addr_t *ipv6_addr)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-08-13 06:41:05 +02:00
|
|
|
return (ipv6_addr->uint32[0] == 0) && (ipv6_addr->uint32[1] == 0) &&
|
|
|
|
(ipv6_addr->uint32[2] == 0) && (ipv6_addr->uint32[3] == 0);
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
int ipv6_addr_is_solicited_node(const ipv6_addr_t *ipv6_addr)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-08-13 06:41:05 +02:00
|
|
|
return (ipv6_addr->uint8[0] == 0xFF) &&
|
|
|
|
(ipv6_addr->uint8[1] == 0x02) &&
|
|
|
|
(ipv6_addr->uint16[1] == 0x00) &&
|
|
|
|
(ipv6_addr->uint16[2] == 0x00) &&
|
|
|
|
(ipv6_addr->uint16[3] == 0x00) &&
|
|
|
|
(ipv6_addr->uint16[4] == 0x00) &&
|
|
|
|
(ipv6_addr->uint8[10] == 0x00) &&
|
|
|
|
(ipv6_addr->uint8[11] == 0x01) &&
|
|
|
|
(ipv6_addr->uint8[12] == 0xFF);
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
void ipv6_addr_set_solicited_node_addr(ipv6_addr_t *ipv6_addr_out,
|
|
|
|
const ipv6_addr_t *ipv6_addr_in)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2012-11-05 22:29:11 +01:00
|
|
|
/* copy only the last 24-bit of the ip-address that is beeing resolved */
|
2013-08-13 06:41:05 +02:00
|
|
|
ipv6_addr_out->uint16[0] = HTONS(0xff02);
|
|
|
|
ipv6_addr_out->uint16[1] = 0;
|
|
|
|
ipv6_addr_out->uint16[2] = 0;
|
|
|
|
ipv6_addr_out->uint16[3] = 0;
|
|
|
|
ipv6_addr_out->uint16[4] = 0;
|
|
|
|
ipv6_addr_out->uint16[5] = HTONS(0x0001);
|
|
|
|
ipv6_addr_out->uint8[12] = 0xff;
|
|
|
|
ipv6_addr_out->uint8[13] = ipv6_addr_in->uint8[13];
|
|
|
|
ipv6_addr_out->uint16[7] = ipv6_addr_in->uint16[7];
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
char *ipv6_addr_to_str(char *addr_str, const ipv6_addr_t *ipv6_addr)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-08-13 06:41:05 +02:00
|
|
|
sprintf(addr_str,
|
|
|
|
"%04x:%04x:%04x:%04x:%04x:%04x:%04x:%04x",
|
|
|
|
ipv6_addr->uint16[0], ipv6_addr->uint16[1],
|
|
|
|
ipv6_addr->uint16[2], ipv6_addr->uint16[3],
|
|
|
|
ipv6_addr->uint16[4], ipv6_addr->uint16[5],
|
|
|
|
ipv6_addr->uint16[6], ipv6_addr->uint16[7]);
|
|
|
|
return addr_str;
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
uint32_t get_remaining_time(timex_t *t)
|
|
|
|
{
|
2013-06-13 10:22:55 +02:00
|
|
|
timex_t now;
|
|
|
|
vtimer_now(&now);
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2012-11-05 22:29:11 +01:00
|
|
|
return (timex_sub(*t, now).seconds);
|
|
|
|
}
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
void set_remaining_time(timex_t *t, uint32_t time)
|
|
|
|
{
|
2012-11-05 22:29:11 +01:00
|
|
|
timex_t tmp = {time, 0};
|
|
|
|
|
2013-06-13 10:22:55 +02:00
|
|
|
timex_t now;
|
2013-06-14 20:37:39 +02:00
|
|
|
vtimer_now(&now);
|
2013-06-13 10:22:55 +02:00
|
|
|
*t = timex_add(now, tmp);
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
void ipv6_init_iface_as_router(void)
|
|
|
|
{
|
2012-11-05 22:29:11 +01:00
|
|
|
ipv6_addr_t addr;
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
ipv6_addr_set_all_routers_addr(&addr);
|
|
|
|
ipv6_iface_add_addr(&addr, NDP_ADDR_STATE_PREFERRED, 0, 0, IPV6_ADDR_TYPE_MULTICAST);
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
uint8_t ipv6_is_router(void)
|
|
|
|
{
|
2012-11-05 22:29:11 +01:00
|
|
|
ipv6_addr_t addr;
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
ipv6_addr_set_all_routers_addr(&addr);
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (ipv6_iface_addr_match(&addr) != NULL) {
|
2012-11-05 22:29:11 +01:00
|
|
|
return 1;
|
|
|
|
}
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2012-11-05 22:29:11 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
void set_tcp_packet_handler_pid(int pid)
|
|
|
|
{
|
|
|
|
tcp_packet_handler_pid = pid;
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
void set_udp_packet_handler_pid(int pid)
|
|
|
|
{
|
|
|
|
udp_packet_handler_pid = pid;
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|
|
|
|
|
2013-08-13 06:41:05 +02:00
|
|
|
void ipv6_register_next_header_handler(uint8_t next_header, int pid)
|
|
|
|
{
|
|
|
|
switch (next_header) {
|
|
|
|
case (IPV6_PROTO_NUM_TCP):
|
|
|
|
set_tcp_packet_handler_pid(pid);
|
|
|
|
break;
|
|
|
|
case (IPV6_PROTO_NUM_UDP):
|
|
|
|
set_udp_packet_handler_pid(pid);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* TODO */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ipv6_register_rpl_handler(int pid)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
|
|
|
rpl_process_pid = pid;
|
2012-11-05 22:29:11 +01:00
|
|
|
}
|