2014-02-03 22:47:31 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2013 INRIA
|
|
|
|
*
|
2014-07-31 19:45:27 +02:00
|
|
|
* 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.
|
2014-02-03 22:47:31 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup examples
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief UDP RPL example application
|
|
|
|
*
|
|
|
|
* @author Oliver Hahm <oliver.hahm@inria.fr>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2014-01-06 10:10:47 +01:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include "vtimer.h"
|
|
|
|
#include "thread.h"
|
2014-02-13 16:06:25 +01:00
|
|
|
#include "net_if.h"
|
2014-01-06 10:10:47 +01:00
|
|
|
#include "sixlowpan.h"
|
|
|
|
#include "destiny.h"
|
|
|
|
#include "rpl.h"
|
|
|
|
#include "rpl_dodag.h"
|
|
|
|
#include "demo.h"
|
2014-04-02 14:30:17 +02:00
|
|
|
#include "transceiver.h"
|
2014-01-06 10:10:47 +01:00
|
|
|
|
2014-03-03 00:25:38 +01:00
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
2014-04-02 14:30:17 +02:00
|
|
|
#define TRANSCEIVER TRANSCEIVER_DEFAULT
|
2014-01-06 10:10:47 +01:00
|
|
|
|
|
|
|
char monitor_stack_buffer[MONITOR_STACK_SIZE];
|
|
|
|
radio_address_t id;
|
|
|
|
ipv6_addr_t std_addr;
|
|
|
|
|
|
|
|
uint8_t is_root = 0;
|
|
|
|
|
2014-02-24 23:39:23 +01:00
|
|
|
void rpl_udp_init(int argc, char **argv)
|
2014-01-06 10:10:47 +01:00
|
|
|
{
|
|
|
|
transceiver_command_t tcmd;
|
|
|
|
msg_t m;
|
|
|
|
uint8_t chan = RADIO_CHANNEL;
|
|
|
|
|
2014-02-24 23:39:23 +01:00
|
|
|
if (argc != 2) {
|
|
|
|
printf("Usage: %s (r|n)\n", argv[0]);
|
2014-01-06 10:10:47 +01:00
|
|
|
printf("\tr\tinitialize as root\n");
|
|
|
|
printf("\tn\tinitialize as node router\n");
|
2014-01-18 01:16:38 +01:00
|
|
|
return;
|
2014-01-06 10:10:47 +01:00
|
|
|
}
|
2014-01-25 11:22:09 +01:00
|
|
|
|
2014-01-06 10:10:47 +01:00
|
|
|
uint8_t state;
|
|
|
|
|
2014-02-24 23:39:23 +01:00
|
|
|
char command = argv[1][0];
|
2014-01-06 10:10:47 +01:00
|
|
|
if ((command == 'n') || (command == 'r')) {
|
|
|
|
printf("INFO: Initialize as %s on address %d\n", ((command == 'n') ? "node" : "root"), id);
|
2014-01-25 11:22:09 +01:00
|
|
|
|
2014-01-06 10:10:47 +01:00
|
|
|
if (!id || (id > 255)) {
|
|
|
|
printf("ERROR: address not a valid 8 bit integer\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-03-03 10:59:19 +01:00
|
|
|
DEBUGF("Setting HW address to %u\n", id);
|
2014-02-13 16:06:25 +01:00
|
|
|
net_if_set_hardware_address(0, id);
|
|
|
|
|
2014-03-03 10:59:19 +01:00
|
|
|
DEBUGF("Initializing RPL for interface 0\n");
|
2014-02-13 16:06:25 +01:00
|
|
|
state = rpl_init(0);
|
2014-01-06 10:10:47 +01:00
|
|
|
|
|
|
|
if (state != SIXLOWERROR_SUCCESS) {
|
|
|
|
printf("Error initializing RPL\n");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("6LoWPAN and RPL initialized.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (command == 'r') {
|
|
|
|
rpl_init_root();
|
|
|
|
is_root = 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
ipv6_iface_set_routing_provider(rpl_get_next_hop);
|
|
|
|
}
|
2014-01-25 11:22:09 +01:00
|
|
|
|
2014-03-03 10:59:19 +01:00
|
|
|
DEBUGF("Start monitor\n");
|
2014-07-06 22:57:56 +02:00
|
|
|
kernel_pid_t monitor_pid = thread_create(monitor_stack_buffer,
|
|
|
|
sizeof(monitor_stack_buffer),
|
|
|
|
PRIORITY_MAIN - 2,
|
|
|
|
CREATE_STACKTEST,
|
|
|
|
rpl_udp_monitor,
|
|
|
|
NULL,
|
|
|
|
"monitor");
|
2014-03-03 10:59:19 +01:00
|
|
|
DEBUGF("Register at transceiver %02X\n", TRANSCEIVER);
|
2014-01-06 10:10:47 +01:00
|
|
|
transceiver_register(TRANSCEIVER, monitor_pid);
|
|
|
|
ipv6_register_packet_handler(monitor_pid);
|
|
|
|
//sixlowpan_lowpan_register(monitor_pid);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
printf("ERROR: Unknown command '%c'\n", command);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO: check if this works as intended */
|
|
|
|
ipv6_addr_t prefix, tmp;
|
|
|
|
ipv6_addr_init(&std_addr, 0xABCD, 0xEF12, 0, 0, 0x1034, 0x00FF, 0xFE00, id);
|
|
|
|
ipv6_addr_init_prefix(&prefix, &std_addr, 64);
|
2014-02-13 16:06:25 +01:00
|
|
|
ndp_add_prefix_info(0, &prefix, 64, NDP_OPT_PI_VLIFETIME_INFINITE,
|
|
|
|
NDP_OPT_PI_PLIFETIME_INFINITE, 1,
|
|
|
|
ICMPV6_NDP_OPT_PI_FLAG_AUTONOM);
|
|
|
|
ipv6_init_as_router();
|
2014-01-06 10:10:47 +01:00
|
|
|
/* add global address */
|
2014-02-13 16:06:25 +01:00
|
|
|
ipv6_addr_set_by_eui64(&tmp, 0, &std_addr);
|
|
|
|
ipv6_net_if_add_addr(0, &tmp, NDP_ADDR_STATE_PREFERRED, 0, 0, 0);
|
2014-01-25 11:22:09 +01:00
|
|
|
|
2014-01-06 10:10:47 +01:00
|
|
|
/* set channel to 10 */
|
|
|
|
tcmd.transceivers = TRANSCEIVER;
|
|
|
|
tcmd.data = &chan;
|
|
|
|
m.type = SET_CHANNEL;
|
|
|
|
m.content.ptr = (void *) &tcmd;
|
|
|
|
|
|
|
|
msg_send_receive(&m, &m, transceiver_pid);
|
|
|
|
printf("Channel set to %u\n", RADIO_CHANNEL);
|
|
|
|
|
|
|
|
puts("Destiny initialized");
|
|
|
|
/* start transceiver watchdog */
|
|
|
|
}
|
|
|
|
|
2014-02-24 23:39:23 +01:00
|
|
|
void rpl_udp_loop(int argc, char **argv)
|
2014-01-06 10:10:47 +01:00
|
|
|
{
|
2014-02-24 23:39:23 +01:00
|
|
|
(void) argc;
|
|
|
|
(void) argv;
|
2014-01-06 10:10:47 +01:00
|
|
|
|
|
|
|
rpl_routing_entry_t *rtable;
|
|
|
|
|
|
|
|
rtable = rpl_get_routing_table();
|
|
|
|
rpl_dodag_t *mydodag = rpl_get_my_dodag();
|
|
|
|
|
|
|
|
if (mydodag == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("---------------------------\n");
|
|
|
|
printf("OUTPUT\n");
|
|
|
|
printf("my rank: %d\n", mydodag->my_rank);
|
2014-01-25 11:22:09 +01:00
|
|
|
|
2014-01-06 10:10:47 +01:00
|
|
|
if (!is_root) {
|
|
|
|
printf("my preferred parent:\n");
|
2014-02-13 16:06:25 +01:00
|
|
|
printf("%s\n", ipv6_addr_to_str(addr_str, IPV6_MAX_ADDR_STR_LEN,
|
|
|
|
(&mydodag->my_preferred_parent->addr)));
|
2014-01-06 10:10:47 +01:00
|
|
|
printf("parent lifetime: %d\n", mydodag->my_preferred_parent->lifetime);
|
|
|
|
}
|
2014-01-25 11:22:09 +01:00
|
|
|
|
2014-01-06 10:10:47 +01:00
|
|
|
printf("---------------------------$\n");
|
|
|
|
|
|
|
|
for (int i = 0; i < RPL_MAX_ROUTING_ENTRIES; i++) {
|
|
|
|
if (rtable[i].used) {
|
2014-02-13 16:06:25 +01:00
|
|
|
printf("%s\n", ipv6_addr_to_str(addr_str, IPV6_MAX_ADDR_STR_LEN,
|
|
|
|
(&rtable[i].address)));
|
2014-01-06 10:10:47 +01:00
|
|
|
puts("next hop");
|
2014-02-13 16:06:25 +01:00
|
|
|
printf("%s\n", ipv6_addr_to_str(addr_str, IPV6_MAX_ADDR_STR_LEN,
|
|
|
|
(&rtable[i].next_hop)));
|
2014-01-06 10:10:47 +01:00
|
|
|
printf("entry %d lifetime %d\n", i, rtable[i].lifetime);
|
|
|
|
|
|
|
|
if (!rpl_equal_id(&rtable[i].address, &rtable[i].next_hop)) {
|
|
|
|
puts("multi-hop");
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("---------------------------$\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("########################\n");
|
|
|
|
}
|
|
|
|
|
2014-02-24 23:39:23 +01:00
|
|
|
void rpl_udp_table(int argc, char **argv)
|
2014-01-06 10:10:47 +01:00
|
|
|
{
|
2014-02-24 23:39:23 +01:00
|
|
|
(void) argc;
|
|
|
|
(void) argv;
|
2014-01-06 10:10:47 +01:00
|
|
|
|
|
|
|
rpl_routing_entry_t *rtable;
|
|
|
|
rtable = rpl_get_routing_table();
|
|
|
|
printf("---------------------------\n");
|
|
|
|
printf("OUTPUT\n");
|
|
|
|
printf("---------------------------\n");
|
|
|
|
|
|
|
|
for (int i = 0; i < RPL_MAX_ROUTING_ENTRIES; i++) {
|
|
|
|
if (rtable[i].used) {
|
2014-02-13 16:06:25 +01:00
|
|
|
printf("%s\n", ipv6_addr_to_str(addr_str, IPV6_MAX_ADDR_STR_LEN,
|
|
|
|
(&rtable[i].address)));
|
2014-01-06 10:10:47 +01:00
|
|
|
printf("entry %d lifetime %d\n", i, rtable[i].lifetime);
|
|
|
|
|
|
|
|
if (!rpl_equal_id(&rtable[i].address, &rtable[i].next_hop)) {
|
|
|
|
puts("multi-hop");
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("--------------\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("$\n");
|
|
|
|
}
|
|
|
|
|
2014-02-24 23:39:23 +01:00
|
|
|
void rpl_udp_dodag(int argc, char **argv)
|
2014-01-06 10:10:47 +01:00
|
|
|
{
|
2014-02-24 23:39:23 +01:00
|
|
|
(void) argc;
|
|
|
|
(void) argv;
|
2014-01-06 10:10:47 +01:00
|
|
|
|
|
|
|
printf("---------------------------\n");
|
|
|
|
rpl_dodag_t *mydodag = rpl_get_my_dodag();
|
|
|
|
|
|
|
|
if (mydodag == NULL) {
|
|
|
|
printf("Not part of a dodag\n");
|
2014-01-18 01:15:29 +01:00
|
|
|
printf("---------------------------\n");
|
2014-01-06 10:10:47 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("Part of Dodag:\n");
|
2014-02-13 16:06:25 +01:00
|
|
|
printf("%s\n", ipv6_addr_to_str(addr_str, IPV6_MAX_ADDR_STR_LEN,
|
|
|
|
(&mydodag->dodag_id)));
|
2014-01-06 10:10:47 +01:00
|
|
|
printf("my rank: %d\n", mydodag->my_rank);
|
2014-01-25 11:22:09 +01:00
|
|
|
|
2014-01-06 10:10:47 +01:00
|
|
|
if (!is_root) {
|
|
|
|
printf("my preferred parent:\n");
|
2014-02-13 16:06:25 +01:00
|
|
|
printf("%s\n", ipv6_addr_to_str(addr_str, IPV6_MAX_ADDR_STR_LEN,
|
|
|
|
(&mydodag->my_preferred_parent->addr)));
|
2014-01-06 10:10:47 +01:00
|
|
|
}
|
2014-01-25 11:22:09 +01:00
|
|
|
|
2014-01-18 01:15:29 +01:00
|
|
|
printf("---------------------------\n");
|
2014-01-06 10:10:47 +01:00
|
|
|
}
|