1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/sys/net/rpl/etx_beaconing.c

540 lines
16 KiB
C
Raw Normal View History

2013-03-03 17:47:11 +01:00
/*
* etx_beaconing.c
*
* Created on: Feb 26, 2013
* Author: stephan
*/
#include "etx_beaconing.h"
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#include <hwtimer.h>
#include <vtimer.h>
2013-03-03 17:47:11 +01:00
#include <thread.h>
#include <transceiver.h>
2013-08-05 16:10:54 +02:00
#include "sixlowpan.h"
2013-09-18 14:13:19 +02:00
#include "ieee802154_frame.h"
2013-03-24 20:03:18 +01:00
2013-03-28 15:27:30 +01:00
//prototytpes
static uint8_t etx_count_packet_tx(etx_neighbor_t *candidate);
2013-03-28 15:27:30 +01:00
static void etx_set_packets_received(void);
static bool etx_equal_id(ipv6_addr_t *id1, ipv6_addr_t *id2);
2013-03-03 17:47:11 +01:00
//Buffer
2013-03-28 15:27:30 +01:00
char etx_beacon_buf[ETX_BEACON_STACKSIZE] = { 0 };
char etx_radio_buf[ETX_RADIO_STACKSIZE] = { 0 };
char etx_clock_buf[ETX_CLOCK_STACKSIZE] = { 0 };
uint8_t etx_send_buf[ETX_BUF_SIZE] = { 0 };
uint8_t etx_rec_buf[ETX_BUF_SIZE] = { 0 };
2013-03-24 20:03:18 +01:00
2013-03-28 15:27:30 +01:00
//PIDs
int etx_beacon_pid = 0;
int etx_radio_pid = 0;
int etx_clock_pid = 0;
/*
* xxx If you get a -Wmissing-braces warning here:
* A -Wmissing-braces warning at this point is a gcc-bug!
* Please delete this information once it's fixed
* See: http://gcc.gnu.org/bugzilla/show_bug.cgi?id=53119
*/
2013-03-24 20:03:18 +01:00
//Message queue for radio
msg_t msg_que[ETX_RCV_QUEUE_SIZE] = { 0 };
2013-03-03 17:47:11 +01:00
/*
* The counter for the current 'round'. An ETX beacon is sent every ETX_INTERVAL
* u-seconds and a node computes the ETX value by comparing the the received
* probes vs the expected probes from a neighbor every ETX_ROUND intervals.
*/
2013-03-28 15:27:30 +01:00
static uint8_t cur_round = 0;
/*
* If we have not yet reached WINDOW intervals, won't calculate the ETX just yet
*/
static char reached_window = 0;
2013-03-03 17:47:11 +01:00
/*
* This could (and should) be done differently, once the RPL implementation
* deals with candidate neighbors in another way than just defining that every
* possible neighbor we hear from is a parent.
* Right now, we need to keep track of the ETX values of other nodes without
* needing them to be in our parent array, so we have another array here in
* which we put all necessary info for up to ETX_MAX_CANDIDATE_NEIHGBORS
* candidates.
2013-03-28 15:27:30 +01:00
*
* xxx If you get a -Wmissing-braces warning here:
* A -Wmissing-braces warning at this point is a gcc-bug!
* Please delete this information once it's fixed
* See: http://gcc.gnu.org/bugzilla/show_bug.cgi?id=53119
*/
2013-03-28 15:27:30 +01:00
//Candidate array
static etx_neighbor_t candidates[ETX_MAX_CANDIDATE_NEIGHBORS] = { 0 };
2013-03-03 17:47:11 +01:00
2013-03-28 15:27:30 +01:00
/*
* Each time we send a beacon packet we need to reset some values for the
* current 'round' (a round being the time between each sent beacon packet).
*
* In this time, no packet may be handled, otherwise it could assume values
* from the last round to count for this round.
*/
mutex_t etx_mutex;
2013-03-03 17:47:11 +01:00
//Transceiver command for sending ETX probes
transceiver_command_t tcmd;
//Message to send probes with
msg_t mesg;
//RPL-address
static ipv6_addr_t *own_address;
static etx_probe_t *etx_get_send_buf(void)
{
return ((etx_probe_t *) &(etx_send_buf[0]));
2013-03-03 17:47:11 +01:00
}
static etx_probe_t *etx_get_rec_buf(void)
{
return ((etx_probe_t *) &(etx_rec_buf[0]));
2013-03-03 17:47:11 +01:00
}
void show_candidates(void)
{
etx_neighbor_t *candidate;
2013-03-28 15:27:30 +01:00
etx_neighbor_t *end;
for (candidate = &candidates[0], end = candidates
+ ETX_MAX_CANDIDATE_NEIGHBORS; candidate < end;
candidate++) {
2013-03-28 15:27:30 +01:00
if (candidate->used == 0) {
break;
}
2013-03-28 15:27:30 +01:00
printf("Candidates Addr:%d\n"
"\t cur_etx:%f\n"
"\t packets_rx:%d\n"
"\t packets_tx:%d\n"
"\t used:%d\n", candidate->addr.uint8[ETX_IPV6_LAST_BYTE],
candidate->cur_etx, candidate->packets_rx,
etx_count_packet_tx(candidate),
candidate->used);
}
}
2013-03-03 17:47:11 +01:00
void etx_init_beaconing(ipv6_addr_t *address)
{
mutex_init(&etx_mutex);
own_address = address;
//set code
puts("ETX BEACON INIT");
2013-03-28 15:27:30 +01:00
etx_send_buf[0] = ETX_PKT_OPTVAL;
2013-03-03 17:47:11 +01:00
etx_beacon_pid = thread_create(etx_beacon_buf, ETX_BEACON_STACKSIZE,
PRIORITY_MAIN - 1, CREATE_STACKTEST,
etx_beacon, "etx_beacon");
2013-03-03 17:47:11 +01:00
etx_radio_pid = thread_create(etx_radio_buf, ETX_RADIO_STACKSIZE,
PRIORITY_MAIN - 1, CREATE_STACKTEST,
etx_radio, "etx_radio");
2013-03-03 17:47:11 +01:00
2013-03-28 15:27:30 +01:00
etx_clock_pid = thread_create(etx_clock_buf, ETX_CLOCK_STACKSIZE,
PRIORITY_MAIN - 1, CREATE_STACKTEST,
etx_clock, "etx_clock");
//register at transceiver
2013-03-03 17:47:11 +01:00
transceiver_register(TRANSCEIVER_CC1100, etx_radio_pid);
puts("...[DONE]");
}
void etx_beacon(void)
{
2013-03-03 17:47:11 +01:00
/*
* Sends a message every ETX_INTERVAL +/- a jitter-value (default is 10%) .
* A correcting variable is needed to stay at a base interval of
* ETX_INTERVAL between the wakeups. It takes the old jittervalue in account
* and modifies the time to wait accordingly.
2013-03-03 17:47:11 +01:00
*/
etx_probe_t *packet = etx_get_send_buf();
2013-03-28 15:27:30 +01:00
uint8_t p_length = 0;
/*
2013-03-28 15:27:30 +01:00
* xxx If you get a -Wmissing-braces warning here:
* A -Wmissing-braces warning at this point is a gcc-bug!
* Please delete this information once it's fixed
* See: http://gcc.gnu.org/bugzilla/show_bug.cgi?id=53119
*/
ieee_802154_long_t empty_addr = { 0 };
2013-03-03 17:47:11 +01:00
2013-03-28 15:27:30 +01:00
while (true) {
thread_sleep();
mutex_lock(&etx_mutex);
//Build etx packet
p_length = 0;
for (uint8_t i = 0; i < ETX_BEST_CANDIDATES; i++) {
if (candidates[i].used != 0) {
packet->data[i * ETX_TUPLE_SIZE] =
candidates[i].addr.uint8[ETX_IPV6_LAST_BYTE];
packet->data[i * ETX_TUPLE_SIZE + ETX_PKT_REC_OFFSET] =
etx_count_packet_tx(&candidates[i]);
p_length = p_length + ETX_PKT_HDR_LEN;
}
}
packet->length = p_length;
2013-03-28 15:27:30 +01:00
send_ieee802154_frame(&empty_addr, &etx_send_buf[0],
ETX_DATA_MAXLEN + ETX_PKT_HDR_LEN, 1);
2013-03-28 15:27:30 +01:00
DEBUG("sent beacon!\n");
etx_set_packets_received();
cur_round++;
2013-03-28 15:27:30 +01:00
if (cur_round == ETX_WINDOW) {
if (reached_window != 1) {
//first round is through
reached_window = 1;
}
2013-03-28 15:27:30 +01:00
cur_round = 0;
}
2013-08-14 16:19:35 +02:00
mutex_unlock(&etx_mutex);
}
}
etx_neighbor_t *etx_find_candidate(ipv6_addr_t *address)
{
/*
* find the candidate with address address and returns it, or returns NULL
* if no candidate having this address was found.
*/
for (uint8_t i = 0; i < ETX_MAX_CANDIDATE_NEIGHBORS; i++) {
if (candidates[i].used
&& (etx_equal_id(&candidates[i].addr, address))) {
return &candidates[i];
}
}
return NULL ;
}
2013-03-03 17:47:11 +01:00
void etx_clock(void)
{
2013-03-28 15:27:30 +01:00
/*
* Manages the etx_beacon thread to wake up every full second +- jitter
*/
/*
* The jittercorrection and jitter variables keep usecond values divided
* through 1000 to fit into uint8 variables.
*
* That is why they are multiplied by 1000 when used for hwtimer_wait.
*/
uint8_t jittercorrection = ETX_DEF_JIT_CORRECT;
uint8_t jitter = (uint8_t)(rand() % ETX_JITTER_MOD);
2013-03-28 15:27:30 +01:00
while (true) {
thread_wakeup(etx_beacon_pid);
/*
* Vtimer is buggy, but I seem to have no hwtimers left, so using this
* for now.
*/
vtimer_usleep(
((ETX_INTERVAL - ETX_MAX_JITTER)*MS) + jittercorrection * MS + jitter * MS - ETX_CLOCK_ADJUST);
2013-03-28 15:27:30 +01:00
//hwtimer_wait(
// HWTIMER_TICKS(((ETX_INTERVAL - ETX_MAX_JITTER)*MS) + jittercorrection*MS + jitter*MS - ETX_CLOCK_ADJUST));
jittercorrection = (ETX_MAX_JITTER) - jitter;
jitter = (uint8_t)(rand() % ETX_JITTER_MOD);
2013-03-28 15:27:30 +01:00
}
}
double etx_get_metric(ipv6_addr_t *address)
{
etx_neighbor_t *candidate = etx_find_candidate(address);
if (candidate != NULL) {
2013-03-28 15:27:30 +01:00
if (etx_count_packet_tx(candidate) > 0) {
//this means the current etx_value is not outdated
return candidate->cur_etx;
}
else {
2013-03-28 15:27:30 +01:00
//The last time I received a packet is too long ago to give a
//good estimate of the etx value
return 0;
}
2013-03-03 17:47:11 +01:00
}
return 0;
2013-03-03 17:47:11 +01:00
}
etx_neighbor_t *etx_add_candidate(ipv6_addr_t *address)
{
2013-03-28 15:27:30 +01:00
DEBUG("add candidate\n");
/*
2013-03-24 20:03:18 +01:00
* Pre-Condition: etx_add_candidate should only be called when the
* candidate is not yet in the list.
* Otherwise the candidate will be added a second time,
* leading to unknown behavior.
*
* Check if there is still enough space to add this candidate
*
* a)
* Space is available:
* Add candidate
*
* b)
* Space is not available:
* ignore new candidate
* This shouldn't really happen though, since we have enough
* place in the array.
*
2013-03-24 20:03:18 +01:00
* Returns the pointer to the candidate if it was added, or a NULL-pointer
* otherwise.
*/
etx_neighbor_t *candidate;
etx_neighbor_t *end;
for (candidate = &candidates[0], end = candidates
+ ETX_MAX_CANDIDATE_NEIGHBORS; candidate < end;
candidate++) {
if (candidate->used) {
//skip
continue;
}
else {
//We still have a free place add the new candidate
2013-03-28 15:27:30 +01:00
memset(candidate, 0, sizeof(*candidate));
2013-03-24 20:03:18 +01:00
candidate->addr = *address;
candidate->cur_etx = 0;
2013-03-24 20:03:18 +01:00
candidate->packets_rx = 0;
candidate->used = 1;
2013-03-24 20:03:18 +01:00
return candidate;
}
}
2013-03-24 20:03:18 +01:00
return NULL ;
}
void etx_handle_beacon(ipv6_addr_t *candidate_address)
{
2013-03-03 17:47:11 +01:00
/*
* Handle the ETX probe that has been received and update all infos.
* If the candidate address is unknown, try to add it to my struct.
2013-03-03 17:47:11 +01:00
*/
2013-03-28 15:27:30 +01:00
DEBUG(
"ETX beacon package received with following values:\n"
"\tPackage Option:%x\n"
"\t Data Length:%u\n"
"\tSource Address:%d\n\n", etx_rec_buf[ETX_PKT_OPT], etx_rec_buf[ETX_PKT_LEN],
candidate_address->uint8[ETX_IPV6_LAST_BYTE]);
etx_neighbor_t *candidate = etx_find_candidate(candidate_address);
if (candidate == NULL) {
2013-03-24 20:03:18 +01:00
//Candidate was not found in my list, I should add it
candidate = etx_add_candidate(candidate_address);
if (candidate == NULL) {
2013-03-24 20:03:18 +01:00
puts("[ERROR] Candidate could not get added");
puts("Increase the constant ETX_MAX_CANDIDATE_NEIHGBORS");
2013-03-24 20:03:18 +01:00
return;
}
}
2013-03-28 15:27:30 +01:00
//I have received 1 packet from this candidate in this round
//This value will be reset by etx_update to 0
candidate->tx_cur_round = 1;
2013-03-24 20:03:18 +01:00
// If i find my address in this probe, update the packet_rx value for
// this candidate.
etx_probe_t *rec_pkt = etx_get_rec_buf();
for (uint8_t i = 0; i < rec_pkt->length / ETX_TUPLE_SIZE; i++) {
2013-03-28 15:27:30 +01:00
DEBUG("\tIPv6 short Addr:%u\n"
"\tPackets f. Addr:%u\n\n", rec_pkt->data[i * ETX_TUPLE_SIZE],
rec_pkt->data[i * ETX_TUPLE_SIZE + ETX_PKT_REC_OFFSET]);
2013-03-28 15:27:30 +01:00
if (rec_pkt->data[i * ETX_TUPLE_SIZE]
== own_address->uint8[ETX_IPV6_LAST_BYTE]) {
2013-03-24 20:03:18 +01:00
candidate->packets_rx = rec_pkt->data[i * ETX_TUPLE_SIZE
+ ETX_PKT_REC_OFFSET];
}
2013-03-03 17:47:11 +01:00
}
2013-03-28 15:27:30 +01:00
//Last, update the ETX value for this candidate
etx_update(candidate);
2013-03-03 17:47:11 +01:00
}
void etx_radio(void)
{
2013-03-03 17:47:11 +01:00
msg_t m;
radio_packet_t *p;
ieee802154_frame_t frame;
msg_init_queue(msg_que, ETX_RCV_QUEUE_SIZE);
ipv6_addr_t ll_address;
ipv6_addr_t candidate_addr;
2013-08-13 06:41:05 +02:00
ipv6_addr_set_link_local_prefix(&ll_address);
ipv6_iface_get_best_src_addr(&candidate_addr, &ll_address);
2013-03-03 17:47:11 +01:00
while (1) {
msg_receive(&m);
2013-03-03 17:47:11 +01:00
if (m.type == PKT_PENDING) {
p = (radio_packet_t *) m.content.ptr;
2013-03-03 17:47:11 +01:00
ieee802154_frame_read(p->data, &frame, p->length);
2013-03-28 15:27:30 +01:00
if (frame.payload[0] == ETX_PKT_OPTVAL) {
2013-03-03 17:47:11 +01:00
//copy to receive buffer
memcpy(etx_rec_buf, &frame.payload[0], frame.payload_len);
//create IPv6 address from radio packet
//we can do the cast here since rpl nodes can only have addr
//up to 8 bits
2013-03-19 14:04:23 +01:00
candidate_addr.uint8[ETX_IPV6_LAST_BYTE] = (uint8_t) p->src;
//handle the beacon
mutex_lock(&etx_mutex);
etx_handle_beacon(&candidate_addr);
2013-08-14 16:19:35 +02:00
mutex_unlock(&etx_mutex);
2013-03-03 17:47:11 +01:00
}
2013-03-03 17:47:11 +01:00
p->processing--;
}
else if (m.type == ENOBUFFER) {
puts("Transceiver buffer full");
}
else {
//packet is not for me, whatever
}
}
}
void etx_update(etx_neighbor_t *candidate)
{
2013-03-28 15:27:30 +01:00
DEBUG("update!\n");
/*
2013-03-28 15:27:30 +01:00
* Update the current ETX value of a candidate
*/
2013-03-28 15:27:30 +01:00
double d_f;
double d_r;
if (reached_window != 1 || candidate == NULL) {
2013-03-28 15:27:30 +01:00
//We will wait at least ETX_WINDOW beacons until we decide to
//calculate an ETX value, so that we have a good estimate
return;
}
/*
* Calculate d_f (the forward PDR) from ME to this candidate.
*/
d_f = candidate->packets_rx / (double) ETX_WINDOW;
/*
* Calculate d_r (the backwards PDR) from this candidate to ME
*/
d_r = etx_count_packet_tx(candidate) / (double) ETX_WINDOW;
/*
* Calculate the current ETX value for my link to this candidate.
*/
if (d_f *d_r != 0) {
2013-03-28 15:27:30 +01:00
candidate->cur_etx = 1 / (d_f * d_r);
}
else {
2013-03-28 15:27:30 +01:00
candidate->cur_etx = 0;
}
DEBUG(
"Estimated ETX Metric is %f for candidate w/ addr %d\n"
"Estimated PDR_forward is %f\n"
"Estimated PDR_backwrd is %f\n"
"\n"
"Received Packets: %d\n"
"Sent Packets : %d\n\n",
candidate->cur_etx, candidate->addr.uint8[ETX_IPV6_LAST_BYTE],
d_f, d_r, candidate->packets_rx, etx_count_packet_tx(candidate));
2013-03-28 15:27:30 +01:00
}
static uint8_t etx_count_packet_tx(etx_neighbor_t *candidate)
{
2013-03-28 15:27:30 +01:00
/*
* Counts the number of packets that were received for this candidate
* in the last ETX_WINDOW intervals.
*/
2013-03-28 17:40:01 +01:00
DEBUG("counting packets");
2013-03-28 15:27:30 +01:00
uint8_t pkt_count = 0;
DEBUG("[");
2013-03-28 15:27:30 +01:00
for (uint8_t i = 0; i < ETX_WINDOW; i++) {
if (i != cur_round) {
pkt_count = pkt_count + candidate->packets_tx[i];
DEBUG("%d", candidate->packets_tx[i]);
2013-03-28 15:27:30 +01:00
if (i < ETX_WINDOW - 1) {
DEBUG(",");
}
}
else {
2013-03-28 15:27:30 +01:00
//Check if I received something for the current round
if (candidate->tx_cur_round == 0) {
//Didn't receive a packet, zero the field and don't add
candidate->packets_tx[i] = 0;
DEBUG("%d!", candidate->packets_tx[i]);
2013-03-28 15:27:30 +01:00
if (i < ETX_WINDOW - 1) {
DEBUG(",");
}
}
else {
2013-03-28 15:27:30 +01:00
//Add 1 and set field
pkt_count = pkt_count + 1;
candidate->packets_tx[i] = 1;
DEBUG("%d!", candidate->packets_tx[i]);
2013-03-28 15:27:30 +01:00
if (i < ETX_WINDOW - 1) {
DEBUG(",");
}
}
}
}
2013-03-28 15:27:30 +01:00
DEBUG("]\n");
return pkt_count;
}
static void etx_set_packets_received(void)
{
2013-03-28 15:27:30 +01:00
/*
* Set for all candidates if they received a packet this round or not
*/
for (uint8_t i = 0; i < ETX_MAX_CANDIDATE_NEIGHBORS; i++) {
2013-03-28 15:27:30 +01:00
if (candidates[i].used) {
if (candidates[i].tx_cur_round != 0) {
candidates[i].packets_tx[cur_round] = 1;
candidates[i].tx_cur_round = 0;
}
2013-03-03 17:47:11 +01:00
}
}
}
bool etx_equal_id(ipv6_addr_t *id1, ipv6_addr_t *id2)
{
for (uint8_t i = 0; i < 4; i++) {
if (id1->uint32[i] != id2->uint32[i]) {
return false;
}
}
return true;
}