Make phase module use neighbor-table.
This commit is contained in:
parent
cfb3bbdad9
commit
5a1d8d8006
3 changed files with 35 additions and 94 deletions
core/net/mac
|
@ -255,16 +255,6 @@ static struct compower_activity current_packet;
|
||||||
|
|
||||||
#include "net/mac/phase.h"
|
#include "net/mac/phase.h"
|
||||||
|
|
||||||
#ifdef CONTIKIMAC_CONF_MAX_PHASE_NEIGHBORS
|
|
||||||
#define MAX_PHASE_NEIGHBORS CONTIKIMAC_CONF_MAX_PHASE_NEIGHBORS
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifndef MAX_PHASE_NEIGHBORS
|
|
||||||
#define MAX_PHASE_NEIGHBORS 30
|
|
||||||
#endif
|
|
||||||
|
|
||||||
PHASE_LIST(phase_list, MAX_PHASE_NEIGHBORS);
|
|
||||||
|
|
||||||
#endif /* WITH_PHASE_OPTIMIZATION */
|
#endif /* WITH_PHASE_OPTIMIZATION */
|
||||||
|
|
||||||
#define DEFAULT_STREAM_TIME (4 * CYCLE_TIME)
|
#define DEFAULT_STREAM_TIME (4 * CYCLE_TIME)
|
||||||
|
@ -665,7 +655,7 @@ send_packet(mac_callback_t mac_callback, void *mac_callback_ptr,
|
||||||
|
|
||||||
if(!is_broadcast && !is_receiver_awake) {
|
if(!is_broadcast && !is_receiver_awake) {
|
||||||
#if WITH_PHASE_OPTIMIZATION
|
#if WITH_PHASE_OPTIMIZATION
|
||||||
ret = phase_wait(&phase_list, packetbuf_addr(PACKETBUF_ADDR_RECEIVER),
|
ret = phase_wait(packetbuf_addr(PACKETBUF_ADDR_RECEIVER),
|
||||||
CYCLE_TIME, GUARD_TIME,
|
CYCLE_TIME, GUARD_TIME,
|
||||||
mac_callback, mac_callback_ptr, buf_list);
|
mac_callback, mac_callback_ptr, buf_list);
|
||||||
if(ret == PHASE_DEFERRED) {
|
if(ret == PHASE_DEFERRED) {
|
||||||
|
@ -865,7 +855,7 @@ send_packet(mac_callback_t mac_callback, void *mac_callback_ptr,
|
||||||
|
|
||||||
if(!is_broadcast) {
|
if(!is_broadcast) {
|
||||||
if(collisions == 0 && is_receiver_awake == 0) {
|
if(collisions == 0 && is_receiver_awake == 0) {
|
||||||
phase_update(&phase_list, packetbuf_addr(PACKETBUF_ADDR_RECEIVER),
|
phase_update(packetbuf_addr(PACKETBUF_ADDR_RECEIVER),
|
||||||
encounter_time, ret);
|
encounter_time, ret);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1049,7 +1039,7 @@ init(void)
|
||||||
contikimac_is_on = 1;
|
contikimac_is_on = 1;
|
||||||
|
|
||||||
#if WITH_PHASE_OPTIMIZATION
|
#if WITH_PHASE_OPTIMIZATION
|
||||||
phase_init(&phase_list);
|
phase_init();
|
||||||
#endif /* WITH_PHASE_OPTIMIZATION */
|
#endif /* WITH_PHASE_OPTIMIZATION */
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -40,11 +40,24 @@
|
||||||
#include "net/mac/phase.h"
|
#include "net/mac/phase.h"
|
||||||
#include "net/packetbuf.h"
|
#include "net/packetbuf.h"
|
||||||
#include "sys/clock.h"
|
#include "sys/clock.h"
|
||||||
#include "lib/memb.h"
|
|
||||||
#include "sys/ctimer.h"
|
#include "sys/ctimer.h"
|
||||||
#include "net/queuebuf.h"
|
#include "net/queuebuf.h"
|
||||||
#include "dev/watchdog.h"
|
#include "net/neighbor-table.h"
|
||||||
#include "dev/leds.h"
|
|
||||||
|
#if PHASE_CONF_DRIFT_CORRECT
|
||||||
|
#define PHASE_DRIFT_CORRECT PHASE_CONF_DRIFT_CORRECT
|
||||||
|
#else
|
||||||
|
#define PHASE_DRIFT_CORRECT 0
|
||||||
|
#endif
|
||||||
|
|
||||||
|
struct phase {
|
||||||
|
rtimer_clock_t time;
|
||||||
|
#if PHASE_DRIFT_CORRECT
|
||||||
|
rtimer_clock_t drift;
|
||||||
|
#endif
|
||||||
|
uint8_t noacks;
|
||||||
|
struct timer noacks_timer;
|
||||||
|
};
|
||||||
|
|
||||||
struct phase_queueitem {
|
struct phase_queueitem {
|
||||||
struct ctimer timer;
|
struct ctimer timer;
|
||||||
|
@ -62,6 +75,7 @@ struct phase_queueitem {
|
||||||
#define MAX_NOACKS_TIME CLOCK_SECOND * 30
|
#define MAX_NOACKS_TIME CLOCK_SECOND * 30
|
||||||
|
|
||||||
MEMB(queued_packets_memb, struct phase_queueitem, PHASE_QUEUESIZE);
|
MEMB(queued_packets_memb, struct phase_queueitem, PHASE_QUEUESIZE);
|
||||||
|
NEIGHBOR_TABLE(struct phase, nbr_phase);
|
||||||
|
|
||||||
#define DEBUG 0
|
#define DEBUG 0
|
||||||
#if DEBUG
|
#if DEBUG
|
||||||
|
@ -73,38 +87,14 @@ MEMB(queued_packets_memb, struct phase_queueitem, PHASE_QUEUESIZE);
|
||||||
#define PRINTDEBUG(...)
|
#define PRINTDEBUG(...)
|
||||||
#endif
|
#endif
|
||||||
/*---------------------------------------------------------------------------*/
|
/*---------------------------------------------------------------------------*/
|
||||||
struct phase *
|
|
||||||
find_neighbor(const struct phase_list *list, const rimeaddr_t *addr)
|
|
||||||
{
|
|
||||||
struct phase *e;
|
|
||||||
for(e = list_head(*list->list); e != NULL; e = list_item_next(e)) {
|
|
||||||
if(rimeaddr_cmp(addr, &e->neighbor)) {
|
|
||||||
return e;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
/*---------------------------------------------------------------------------*/
|
|
||||||
void
|
void
|
||||||
phase_remove(const struct phase_list *list, const rimeaddr_t *neighbor)
|
phase_update(const rimeaddr_t *neighbor, rtimer_clock_t time,
|
||||||
{
|
|
||||||
struct phase *e;
|
|
||||||
e = find_neighbor(list, neighbor);
|
|
||||||
if(e != NULL) {
|
|
||||||
list_remove(*list->list, e);
|
|
||||||
memb_free(list->memb, e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
/*---------------------------------------------------------------------------*/
|
|
||||||
void
|
|
||||||
phase_update(const struct phase_list *list,
|
|
||||||
const rimeaddr_t *neighbor, rtimer_clock_t time,
|
|
||||||
int mac_status)
|
int mac_status)
|
||||||
{
|
{
|
||||||
struct phase *e;
|
struct phase *e;
|
||||||
|
|
||||||
/* If we have an entry for this neighbor already, we renew it. */
|
/* If we have an entry for this neighbor already, we renew it. */
|
||||||
e = find_neighbor(list, neighbor);
|
e = nbr_table_get_from_lladdr(nbr_phase, neighbor);
|
||||||
if(e != NULL) {
|
if(e != NULL) {
|
||||||
if(mac_status == MAC_TX_OK) {
|
if(mac_status == MAC_TX_OK) {
|
||||||
#if PHASE_DRIFT_CORRECT
|
#if PHASE_DRIFT_CORRECT
|
||||||
|
@ -123,8 +113,7 @@ phase_update(const struct phase_list *list,
|
||||||
}
|
}
|
||||||
if(e->noacks >= MAX_NOACKS || timer_expired(&e->noacks_timer)) {
|
if(e->noacks >= MAX_NOACKS || timer_expired(&e->noacks_timer)) {
|
||||||
PRINTF("drop %d\n", neighbor->u8[0]);
|
PRINTF("drop %d\n", neighbor->u8[0]);
|
||||||
list_remove(*list->list, e);
|
nbr_table_remove(nbr_phase, e);
|
||||||
memb_free(list->memb, e);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
} else if(mac_status == MAC_TX_OK) {
|
} else if(mac_status == MAC_TX_OK) {
|
||||||
|
@ -133,20 +122,14 @@ phase_update(const struct phase_list *list,
|
||||||
} else {
|
} else {
|
||||||
/* No matching phase was found, so we allocate a new one. */
|
/* No matching phase was found, so we allocate a new one. */
|
||||||
if(mac_status == MAC_TX_OK && e == NULL) {
|
if(mac_status == MAC_TX_OK && e == NULL) {
|
||||||
e = memb_alloc(list->memb);
|
e = nbr_table_add_lladdr(nbr_phase, neighbor);
|
||||||
if(e == NULL) {
|
if(e) {
|
||||||
PRINTF("phase alloc NULL\n");
|
|
||||||
/* We could not allocate memory for this phase, so we drop
|
|
||||||
the last item on the list and reuse it for our phase. */
|
|
||||||
e = list_chop(*list->list);
|
|
||||||
}
|
|
||||||
rimeaddr_copy(&e->neighbor, neighbor);
|
|
||||||
e->time = time;
|
e->time = time;
|
||||||
#if PHASE_DRIFT_CORRECT
|
#if PHASE_DRIFT_CORRECT
|
||||||
e->drift = 0;
|
e->drift = 0;
|
||||||
#endif
|
#endif
|
||||||
e->noacks = 0;
|
e->noacks = 0;
|
||||||
list_push(*list->list, e);
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -168,8 +151,7 @@ send_packet(void *ptr)
|
||||||
}
|
}
|
||||||
/*---------------------------------------------------------------------------*/
|
/*---------------------------------------------------------------------------*/
|
||||||
phase_status_t
|
phase_status_t
|
||||||
phase_wait(struct phase_list *list,
|
phase_wait(const rimeaddr_t *neighbor, rtimer_clock_t cycle_time,
|
||||||
const rimeaddr_t *neighbor, rtimer_clock_t cycle_time,
|
|
||||||
rtimer_clock_t guard_time,
|
rtimer_clock_t guard_time,
|
||||||
mac_callback_t mac_callback, void *mac_callback_ptr,
|
mac_callback_t mac_callback, void *mac_callback_ptr,
|
||||||
struct rdc_buf_list *buf_list)
|
struct rdc_buf_list *buf_list)
|
||||||
|
@ -180,7 +162,7 @@ phase_wait(struct phase_list *list,
|
||||||
phase for this particular neighbor. If so, we can compute the
|
phase for this particular neighbor. If so, we can compute the
|
||||||
time for the next expected phase and setup a ctimer to switch on
|
time for the next expected phase and setup a ctimer to switch on
|
||||||
the radio just before the phase. */
|
the radio just before the phase. */
|
||||||
e = find_neighbor(list, neighbor);
|
e = nbr_table_get_from_lladdr(nbr_phase, neighbor);
|
||||||
if(e != NULL) {
|
if(e != NULL) {
|
||||||
rtimer_clock_t wait, now, expected, sync;
|
rtimer_clock_t wait, now, expected, sync;
|
||||||
clock_time_t ctimewait;
|
clock_time_t ctimewait;
|
||||||
|
@ -242,15 +224,12 @@ phase_wait(struct phase_list *list,
|
||||||
p->buf_list = buf_list;
|
p->buf_list = buf_list;
|
||||||
ctimer_set(&p->timer, ctimewait, send_packet, p);
|
ctimer_set(&p->timer, ctimewait, send_packet, p);
|
||||||
return PHASE_DEFERRED;
|
return PHASE_DEFERRED;
|
||||||
} else {
|
|
||||||
memb_free(&queued_packets_memb, p);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
expected = now + wait - guard_time;
|
expected = now + wait - guard_time;
|
||||||
if(!RTIMER_CLOCK_LT(expected, now)) {
|
if(!RTIMER_CLOCK_LT(expected, now)) {
|
||||||
/* Wait until the receiver is expected to be awake */
|
/* Wait until the receiver is expected to be awake */
|
||||||
// printf("%d ",expected%cycle_time); //for spreadsheet export
|
|
||||||
while(RTIMER_CLOCK_LT(RTIMER_NOW(), expected));
|
while(RTIMER_CLOCK_LT(RTIMER_NOW(), expected));
|
||||||
}
|
}
|
||||||
return PHASE_SEND_NOW;
|
return PHASE_SEND_NOW;
|
||||||
|
@ -259,10 +238,9 @@ phase_wait(struct phase_list *list,
|
||||||
}
|
}
|
||||||
/*---------------------------------------------------------------------------*/
|
/*---------------------------------------------------------------------------*/
|
||||||
void
|
void
|
||||||
phase_init(struct phase_list *list)
|
phase_init(void)
|
||||||
{
|
{
|
||||||
list_init(*list->list);
|
|
||||||
memb_init(list->memb);
|
|
||||||
memb_init(&queued_packets_memb);
|
memb_init(&queued_packets_memb);
|
||||||
|
nbr_table_register(nbr_phase, NULL);
|
||||||
}
|
}
|
||||||
/*---------------------------------------------------------------------------*/
|
/*---------------------------------------------------------------------------*/
|
||||||
|
|
|
@ -47,28 +47,6 @@
|
||||||
#include "lib/memb.h"
|
#include "lib/memb.h"
|
||||||
#include "net/netstack.h"
|
#include "net/netstack.h"
|
||||||
|
|
||||||
#if PHASE_CONF_DRIFT_CORRECT
|
|
||||||
#define PHASE_DRIFT_CORRECT PHASE_CONF_DRIFT_CORRECT
|
|
||||||
#else
|
|
||||||
#define PHASE_DRIFT_CORRECT 0
|
|
||||||
#endif
|
|
||||||
|
|
||||||
struct phase {
|
|
||||||
struct phase *next;
|
|
||||||
rimeaddr_t neighbor;
|
|
||||||
rtimer_clock_t time;
|
|
||||||
#if PHASE_DRIFT_CORRECT
|
|
||||||
rtimer_clock_t drift;
|
|
||||||
#endif
|
|
||||||
uint8_t noacks;
|
|
||||||
struct timer noacks_timer;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct phase_list {
|
|
||||||
list_t *list;
|
|
||||||
struct memb *memb;
|
|
||||||
};
|
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
PHASE_UNKNOWN,
|
PHASE_UNKNOWN,
|
||||||
PHASE_SEND_NOW,
|
PHASE_SEND_NOW,
|
||||||
|
@ -76,18 +54,13 @@ typedef enum {
|
||||||
} phase_status_t;
|
} phase_status_t;
|
||||||
|
|
||||||
|
|
||||||
#define PHASE_LIST(name, num) LIST(phase_list_list); \
|
void phase_init(void);
|
||||||
MEMB(phase_list_memb, struct phase, num); \
|
phase_status_t phase_wait(const rimeaddr_t *neighbor,
|
||||||
struct phase_list name = { &phase_list_list, &phase_list_memb }
|
|
||||||
|
|
||||||
void phase_init(struct phase_list *list);
|
|
||||||
phase_status_t phase_wait(struct phase_list *list, const rimeaddr_t *neighbor,
|
|
||||||
rtimer_clock_t cycle_time, rtimer_clock_t wait_before,
|
rtimer_clock_t cycle_time, rtimer_clock_t wait_before,
|
||||||
mac_callback_t mac_callback, void *mac_callback_ptr,
|
mac_callback_t mac_callback, void *mac_callback_ptr,
|
||||||
struct rdc_buf_list *buf_list);
|
struct rdc_buf_list *buf_list);
|
||||||
void phase_update(const struct phase_list *list, const rimeaddr_t *neighbor,
|
void phase_update(const rimeaddr_t *neighbor,
|
||||||
rtimer_clock_t time, int mac_status);
|
rtimer_clock_t time, int mac_status);
|
||||||
|
void phase_remove(const rimeaddr_t *neighbor);
|
||||||
void phase_remove(const struct phase_list *list, const rimeaddr_t *neighbor);
|
|
||||||
|
|
||||||
#endif /* PHASE_H */
|
#endif /* PHASE_H */
|
||||||
|
|
Loading…
Reference in a new issue