mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
2218 lines
83 KiB
C
2218 lines
83 KiB
C
/*
|
|
* Copyright (C) 2017 INRIA
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
/**
|
|
* @ingroup net_gnrc_gomach
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Implementation of GoMacH
|
|
*
|
|
* @author Shuguo Zhuo <shuguo.zhuo@inria.fr>
|
|
* @}
|
|
*/
|
|
|
|
#include <errno.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <stdint.h>
|
|
#include <stdbool.h>
|
|
|
|
#include "event.h"
|
|
#include "random.h"
|
|
#include "timex.h"
|
|
#include "periph/rtt.h"
|
|
#include "net/gnrc/netif.h"
|
|
#include "net/gnrc/netif/internal.h"
|
|
#include "net/gnrc/netif/ieee802154.h"
|
|
#include "net/netdev/ieee802154.h"
|
|
#include "net/gnrc.h"
|
|
#include "net/gnrc/nettype.h"
|
|
#include "net/netdev.h"
|
|
#include "net/gnrc/mac/internal.h"
|
|
#include "net/gnrc/gomach/gomach.h"
|
|
#include "net/gnrc/gomach/timeout.h"
|
|
#include "include/gomach_internal.h"
|
|
|
|
#ifndef LOG_LEVEL
|
|
/**
|
|
* @brief Default log level define
|
|
*/
|
|
#define LOG_LEVEL LOG_WARNING
|
|
#endif
|
|
|
|
#include "log.h"
|
|
|
|
#define ENABLE_DEBUG 0
|
|
#include "debug.h"
|
|
|
|
/**
|
|
* @brief GoMacH thread's PID
|
|
*/
|
|
static kernel_pid_t gomach_pid;
|
|
|
|
static int _gomach_init(gnrc_netif_t *netif);
|
|
static int _send(gnrc_netif_t *netif, gnrc_pktsnip_t *pkt);
|
|
static gnrc_pktsnip_t *_recv(gnrc_netif_t *netif);
|
|
static void _gomach_msg_handler(gnrc_netif_t *netif, msg_t *msg);
|
|
static void _gomach_event_cb(netdev_t *dev, netdev_event_t event);
|
|
|
|
static const gnrc_netif_ops_t gomach_ops = {
|
|
.init = _gomach_init,
|
|
.send = _send,
|
|
.recv = _recv,
|
|
.get = gnrc_netif_get_from_netdev,
|
|
.set = gnrc_netif_set_from_netdev,
|
|
.msg_handler = _gomach_msg_handler,
|
|
};
|
|
|
|
int gnrc_netif_gomach_create(gnrc_netif_t *netif, char *stack, int stacksize,
|
|
char priority, const char *name, netdev_t *dev)
|
|
{
|
|
return gnrc_netif_create(netif, stack, stacksize, priority, name, dev,
|
|
&gomach_ops);
|
|
}
|
|
|
|
static gnrc_pktsnip_t *_recv(gnrc_netif_t *netif)
|
|
{
|
|
netdev_t *dev = netif->dev;
|
|
netdev_ieee802154_rx_info_t rx_info;
|
|
netdev_ieee802154_t *state = container_of(dev, netdev_ieee802154_t, netdev);
|
|
gnrc_pktsnip_t *pkt = NULL;
|
|
int bytes_expected = dev->driver->recv(dev, NULL, 0, NULL);
|
|
|
|
if (bytes_expected > 0) {
|
|
int nread;
|
|
|
|
pkt = gnrc_pktbuf_add(NULL, NULL, bytes_expected, GNRC_NETTYPE_UNDEF);
|
|
if (pkt == NULL) {
|
|
DEBUG("_recv_ieee802154: cannot allocate pktsnip.\n");
|
|
return NULL;
|
|
}
|
|
nread = dev->driver->recv(dev, pkt->data, bytes_expected, &rx_info);
|
|
if (nread <= 0) {
|
|
gnrc_pktbuf_release(pkt);
|
|
return NULL;
|
|
}
|
|
if (!(state->flags & NETDEV_IEEE802154_RAW)) {
|
|
gnrc_pktsnip_t *ieee802154_hdr;
|
|
size_t mhr_len = ieee802154_get_frame_hdr_len(pkt->data);
|
|
|
|
if (mhr_len == 0) {
|
|
DEBUG("_recv_ieee802154: illegally formatted frame received\n");
|
|
gnrc_pktbuf_release(pkt);
|
|
return NULL;
|
|
}
|
|
nread -= mhr_len;
|
|
/* mark IEEE 802.15.4 header */
|
|
ieee802154_hdr = gnrc_pktbuf_mark(pkt, mhr_len, GNRC_NETTYPE_UNDEF);
|
|
if (ieee802154_hdr == NULL) {
|
|
DEBUG("_recv_ieee802154: no space left in packet buffer\n");
|
|
gnrc_pktbuf_release(pkt);
|
|
return NULL;
|
|
}
|
|
netif->mac.prot.gomach.rx_pkt_lqi = rx_info.lqi;
|
|
netif->mac.prot.gomach.rx_pkt_rssi = rx_info.rssi;
|
|
}
|
|
|
|
DEBUG("_recv_ieee802154: reallocating.\n");
|
|
gnrc_pktbuf_realloc_data(pkt, nread);
|
|
}
|
|
|
|
return pkt;
|
|
}
|
|
|
|
static void gomach_reinit_radio(gnrc_netif_t *netif)
|
|
{
|
|
/* Initialize low-level driver. */
|
|
netif->dev->driver->init(netif->dev);
|
|
|
|
/* Set MAC address length. */
|
|
uint16_t src_len = netif->l2addr_len;
|
|
netif->dev->driver->set(netif->dev, NETOPT_SRC_LEN, &src_len, sizeof(src_len));
|
|
|
|
/* Set the MAC address of the device. */
|
|
if (netif->l2addr_len == IEEE802154_LONG_ADDRESS_LEN) {
|
|
netif->dev->driver->set(netif->dev,
|
|
NETOPT_ADDRESS_LONG,
|
|
netif->l2addr,
|
|
sizeof(netif->l2addr));
|
|
}
|
|
else {
|
|
netif->dev->driver->set(netif->dev,
|
|
NETOPT_ADDR_LEN,
|
|
netif->l2addr,
|
|
sizeof(netif->l2addr));
|
|
}
|
|
|
|
/* Check if RX-start and TX-started and TX-END interrupts are supported */
|
|
if (IS_ACTIVE(DEVELHELP)) {
|
|
netopt_enable_t enable;
|
|
netif->dev->driver->get(netif->dev, NETOPT_RX_START_IRQ, &enable, sizeof(enable));
|
|
assert(enable == NETOPT_ENABLE);
|
|
netif->dev->driver->get(netif->dev, NETOPT_TX_END_IRQ, &enable, sizeof(enable));
|
|
assert(enable == NETOPT_ENABLE);
|
|
}
|
|
}
|
|
|
|
static void _gomach_rtt_cb(void *arg)
|
|
{
|
|
msg_t msg;
|
|
|
|
(void)arg;
|
|
msg.content.value = GNRC_GOMACH_EVENT_RTT_NEW_CYCLE;
|
|
msg.type = GNRC_GOMACH_EVENT_RTT_TYPE;
|
|
msg_send(&msg, gomach_pid);
|
|
|
|
if (sched_context_switch_request) {
|
|
thread_yield();
|
|
}
|
|
}
|
|
|
|
static void _gomach_rtt_handler(uint32_t event, gnrc_netif_t *netif)
|
|
{
|
|
switch (event & 0xffff) {
|
|
case GNRC_GOMACH_EVENT_RTT_NEW_CYCLE: {
|
|
/* Start duty-cycle scheme. */
|
|
if (!gnrc_gomach_get_duty_cycle_start(netif)) {
|
|
gnrc_gomach_set_duty_cycle_start(netif, true);
|
|
rtt_clear_alarm();
|
|
/* Record the new cycle's starting time. */
|
|
netif->mac.prot.gomach.last_wakeup = rtt_get_counter();
|
|
}
|
|
else {
|
|
/* The duty-cycle scheme has already started,
|
|
* record the new cycle's starting time. */
|
|
netif->mac.prot.gomach.last_wakeup = rtt_get_alarm();
|
|
gnrc_gomach_set_enter_new_cycle(netif, true);
|
|
}
|
|
|
|
netif->mac.prot.gomach.last_wakeup_phase_us = xtimer_now_usec64();
|
|
|
|
/* Set next cycle's starting time. */
|
|
uint32_t alarm = netif->mac.prot.gomach.last_wakeup +
|
|
RTT_US_TO_TICKS(CONFIG_GNRC_GOMACH_SUPERFRAME_DURATION_US);
|
|
rtt_set_alarm(alarm, _gomach_rtt_cb, NULL);
|
|
|
|
/* Update neighbors' public channel phases. */
|
|
gnrc_gomach_update_neighbor_pubchan(netif);
|
|
gnrc_gomach_set_update(netif, true);
|
|
} break;
|
|
default: {
|
|
LOG_ERROR("ERROR: [GOMACH] error RTT message type\n");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void gomach_bcast_init(gnrc_netif_t *netif)
|
|
{
|
|
/* Disable auto-ACK when sending broadcast packets, thus not to receive packet. */
|
|
gnrc_gomach_set_autoack(netif, NETOPT_DISABLE);
|
|
|
|
/* Firstly turn the radio to public channel 1. */
|
|
gnrc_gomach_turn_channel(netif, netif->mac.prot.gomach.pub_channel_1);
|
|
gnrc_gomach_set_on_pubchan_1(netif, true);
|
|
|
|
netif->mac.tx.broadcast_seq++;
|
|
|
|
/* Assemble the broadcast packet. */
|
|
gnrc_pktsnip_t *pkt = netif->mac.tx.packet;
|
|
gnrc_pktsnip_t *payload = netif->mac.tx.packet->next;
|
|
|
|
gnrc_gomach_frame_broadcast_t gomach_broadcast_hdr;
|
|
gomach_broadcast_hdr.header.type = GNRC_GOMACH_FRAME_BROADCAST;
|
|
gomach_broadcast_hdr.seq_nr = netif->mac.tx.broadcast_seq;
|
|
pkt->next = gnrc_pktbuf_add(pkt->next, &gomach_broadcast_hdr,
|
|
sizeof(gomach_broadcast_hdr),
|
|
GNRC_NETTYPE_GOMACH);
|
|
if (pkt->next == NULL) {
|
|
/* Make append payload after netif header again */
|
|
netif->mac.tx.packet->next = payload;
|
|
gnrc_pktbuf_release(netif->mac.tx.packet);
|
|
netif->mac.tx.packet = NULL;
|
|
LOG_ERROR("ERROR: [GOMACH] bcast: no memory to assemble bcast packet, drop packet.\n");
|
|
LOG_ERROR("ERROR: [GOMACH] bcast failed, go to listen mode.\n");
|
|
netif->mac.tx.bcast_state = GNRC_GOMACH_BCAST_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
return;
|
|
}
|
|
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_BCAST_FINISH,
|
|
CONFIG_GNRC_GOMACH_SUPERFRAME_DURATION_US);
|
|
|
|
gnrc_priority_pktqueue_flush(&netif->mac.rx.queue);
|
|
netif->mac.tx.bcast_state = GNRC_GOMACH_BCAST_SEND;
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
|
|
static bool _gomach_send_bcast_busy_handle(gnrc_netif_t *netif)
|
|
{
|
|
/* Quit sending broadcast packet if we found ongoing transmissions,
|
|
* for collision avoidance. */
|
|
if ((gnrc_gomach_get_netdev_state(netif) == NETOPT_STATE_RX) ||
|
|
(gnrc_netif_get_rx_started(netif) == true)) {
|
|
LOG_DEBUG("[GOMACH] bcast: found ongoing transmission, quit broadcast.\n");
|
|
/* Queue the broadcast packet back to the queue. */
|
|
gnrc_pktsnip_t *payload = netif->mac.tx.packet->next->next;
|
|
|
|
/* remove gomach header */
|
|
netif->mac.tx.packet->next->next = NULL;
|
|
gnrc_pktbuf_release(netif->mac.tx.packet->next);
|
|
|
|
/* make append payload after netif header again */
|
|
netif->mac.tx.packet->next = payload;
|
|
|
|
if (!gnrc_mac_queue_tx_packet(&netif->mac.tx, 0, netif->mac.tx.packet)) {
|
|
LOG_DEBUG("[GOMACH] bcast: TX queue full, release packet.\n");
|
|
gnrc_pktbuf_release(netif->mac.tx.packet);
|
|
}
|
|
netif->mac.tx.packet = NULL;
|
|
|
|
netif->mac.tx.bcast_state = GNRC_GOMACH_BCAST_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static void gomach_send_bcast_packet(gnrc_netif_t *netif)
|
|
{
|
|
/* Quit sending broadcast packet if we found ongoing transmissions,
|
|
* for collision avoidance. */
|
|
if (!_gomach_send_bcast_busy_handle(netif)) {
|
|
return;
|
|
}
|
|
|
|
gnrc_pktbuf_hold(netif->mac.tx.packet, 1);
|
|
|
|
/* Start sending the broadcast packet. */
|
|
gnrc_gomach_send(netif, netif->mac.tx.packet, NETOPT_DISABLE);
|
|
|
|
netif->mac.tx.bcast_state = GNRC_GOMACH_BCAST_WAIT_TX_FINISH;
|
|
gnrc_gomach_set_update(netif, false);
|
|
}
|
|
|
|
static void gomach_wait_bcast_tx_finish(gnrc_netif_t *netif)
|
|
{
|
|
if (gnrc_gomach_get_tx_finish(netif)) {
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_BCAST_INTERVAL,
|
|
CONFIG_GNRC_GOMACH_BCAST_INTERVAL_US);
|
|
netif->mac.tx.bcast_state = GNRC_GOMACH_BCAST_WAIT_NEXT_TX;
|
|
gnrc_gomach_set_update(netif, false);
|
|
}
|
|
|
|
/* This is to handle no-TX-complete issue. In case there is no no-TX-complete event,
|
|
* we will quit broadcasting, i.e., not getting stuck here. */
|
|
if (gnrc_gomach_timeout_is_expired(netif, GNRC_GOMACH_TIMEOUT_BCAST_FINISH)) {
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_BCAST_INTERVAL);
|
|
netif->mac.tx.bcast_state = GNRC_GOMACH_BCAST_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
}
|
|
|
|
static void gomach_wait_bcast_wait_next_tx(gnrc_netif_t *netif)
|
|
{
|
|
/* Quit sending broadcast packet if we found ongoing transmissions,
|
|
* for collision avoidance. */
|
|
if (!_gomach_send_bcast_busy_handle(netif)) {
|
|
return;
|
|
}
|
|
|
|
/* If the whole broadcast duration timeouts, release the packet and go to t2u end. */
|
|
if (gnrc_gomach_timeout_is_expired(netif, GNRC_GOMACH_TIMEOUT_BCAST_FINISH)) {
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_BCAST_INTERVAL);
|
|
gnrc_pktbuf_release(netif->mac.tx.packet);
|
|
netif->mac.tx.packet = NULL;
|
|
netif->mac.tx.bcast_state = GNRC_GOMACH_BCAST_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
return;
|
|
}
|
|
|
|
/* Toggle the radio channel and go to send the next broadcast packet. */
|
|
if (gnrc_gomach_timeout_is_expired(netif, GNRC_GOMACH_TIMEOUT_BCAST_INTERVAL)) {
|
|
if (gnrc_gomach_get_on_pubchan_1(netif)) {
|
|
gnrc_gomach_turn_channel(netif, netif->mac.prot.gomach.pub_channel_2);
|
|
gnrc_gomach_set_on_pubchan_1(netif, false);
|
|
}
|
|
else {
|
|
gnrc_gomach_turn_channel(netif, netif->mac.prot.gomach.pub_channel_1);
|
|
gnrc_gomach_set_on_pubchan_1(netif, true);
|
|
}
|
|
|
|
netif->mac.tx.bcast_state = GNRC_GOMACH_BCAST_SEND;
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
}
|
|
|
|
static void gomach_bcast_end(gnrc_netif_t *netif)
|
|
{
|
|
gnrc_gomach_set_netdev_state(netif, NETOPT_STATE_SLEEP);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_BCAST_INTERVAL);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_BCAST_FINISH);
|
|
|
|
if (netif->mac.tx.packet) {
|
|
gnrc_pktbuf_release(netif->mac.tx.packet);
|
|
netif->mac.tx.packet = NULL;
|
|
}
|
|
netif->mac.tx.current_neighbor = NULL;
|
|
|
|
/* Reset the t2u state. */
|
|
netif->mac.tx.bcast_state = GNRC_GOMACH_BCAST_INIT;
|
|
|
|
/* Switch to the listen mode. */
|
|
netif->mac.prot.gomach.basic_state = GNRC_GOMACH_LISTEN;
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_SLEEP;
|
|
gnrc_gomach_set_enter_new_cycle(netif, false);
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
|
|
static void gomach_bcast_update(gnrc_netif_t *netif)
|
|
{
|
|
/* State machine of GoMacH's broadcast procedure. */
|
|
switch (netif->mac.tx.bcast_state) {
|
|
case GNRC_GOMACH_BCAST_INIT: {
|
|
gomach_bcast_init(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_BCAST_SEND: {
|
|
gomach_send_bcast_packet(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_BCAST_WAIT_TX_FINISH: {
|
|
gomach_wait_bcast_tx_finish(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_BCAST_WAIT_NEXT_TX: {
|
|
gomach_wait_bcast_wait_next_tx(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_BCAST_END: {
|
|
gomach_bcast_end(netif);
|
|
break;
|
|
}
|
|
default: break;
|
|
}
|
|
}
|
|
|
|
static void gomach_init_prepare(gnrc_netif_t *netif)
|
|
{
|
|
rtt_clear_alarm();
|
|
|
|
/* Random delay for avoiding the same wake-up phase among devices. */
|
|
uint32_t random_backoff = random_uint32_range(0, CONFIG_GNRC_GOMACH_SUPERFRAME_DURATION_US);
|
|
xtimer_usleep(random_backoff);
|
|
|
|
gnrc_gomach_set_quit_cycle(netif, false);
|
|
netif->mac.prot.gomach.subchannel_occu_flags = 0;
|
|
|
|
gnrc_priority_pktqueue_flush(&netif->mac.rx.queue);
|
|
|
|
/* Since devices don't broadcast beacons on default, so no need to collect beacons.
|
|
* Go to announce its chosen sub-channel sequence. */
|
|
netif->mac.prot.gomach.init_state = GNRC_GOMACH_INIT_ANNC_SUBCHAN;
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
|
|
static void gomach_init_announce_subchannel(gnrc_netif_t *netif)
|
|
{
|
|
/* Choose a sub-channel for the device. */
|
|
gnrc_gomach_init_choose_subchannel(netif);
|
|
|
|
/* Announce the device's chosen sub-channel sequence to its neighbors. */
|
|
gnrc_gomach_bcast_subchann_seq(netif, NETOPT_ENABLE);
|
|
|
|
netif->mac.prot.gomach.init_state = GNRC_GOMACH_INIT_WAIT_FEEDBACK;
|
|
gnrc_gomach_set_update(netif, false);
|
|
}
|
|
|
|
static void gomach_init_wait_announce_feedback(gnrc_netif_t *netif)
|
|
{
|
|
if (gnrc_gomach_get_tx_finish(netif)) {
|
|
gnrc_priority_pktqueue_flush(&netif->mac.rx.queue);
|
|
netif->mac.prot.gomach.init_state = GNRC_GOMACH_INIT_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
}
|
|
|
|
static void gomach_init_end(gnrc_netif_t *netif)
|
|
{
|
|
/* Reset initialization state. */
|
|
netif->mac.prot.gomach.init_state = GNRC_GOMACH_INIT_PREPARE;
|
|
/* Switch to duty-cycle listen mode. */
|
|
netif->mac.prot.gomach.basic_state = GNRC_GOMACH_LISTEN;
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_CP_INIT;
|
|
|
|
/* Start duty-cycle scheme. */
|
|
gnrc_gomach_set_duty_cycle_start(netif, false);
|
|
_gomach_rtt_handler(GNRC_GOMACH_EVENT_RTT_NEW_CYCLE, netif);
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
|
|
static void gomach_t2k_init(gnrc_netif_t *netif)
|
|
{
|
|
/* Turn off radio to conserve power */
|
|
gnrc_gomach_set_netdev_state(netif, NETOPT_STATE_SLEEP);
|
|
|
|
gnrc_gomach_set_quit_cycle(netif, false);
|
|
|
|
/* Set waiting timer for the targeted device! */
|
|
long int wait_phase_duration = netif->mac.tx.current_neighbor->cp_phase -
|
|
gnrc_gomach_phase_now(netif);
|
|
|
|
if (wait_phase_duration < 0) {
|
|
wait_phase_duration += CONFIG_GNRC_GOMACH_SUPERFRAME_DURATION_US;
|
|
}
|
|
|
|
/* Upon several times of t2k failure, we now doubt that the phase-lock may fail due to drift.
|
|
* Here is the phase-lock auto-adjust scheme, trying to catch the neighbot's phase in case of
|
|
* phase-lock failure due to timer drift.
|
|
* Firstly, put the calculated phase ahead, check whether the neighbor's phase has gone ahead
|
|
* of the recorded one */
|
|
if (netif->mac.tx.no_ack_counter == (CONFIG_GNRC_GOMACH_REPHASELOCK_THRESHOLD - 2)) {
|
|
if ((uint32_t)wait_phase_duration < CONFIG_GNRC_GOMACH_CP_DURATION_US) {
|
|
wait_phase_duration = (wait_phase_duration +
|
|
CONFIG_GNRC_GOMACH_SUPERFRAME_DURATION_US) -
|
|
CONFIG_GNRC_GOMACH_CP_DURATION_US;
|
|
}
|
|
else {
|
|
wait_phase_duration = wait_phase_duration - CONFIG_GNRC_GOMACH_CP_DURATION_US;
|
|
}
|
|
}
|
|
/* If this is the last t2k trial, the phase-lock auto-adjust scheme delays the estimated phase
|
|
* a little bit, to see if the real phase is behind the original calculated one. */
|
|
if (netif->mac.tx.no_ack_counter == (CONFIG_GNRC_GOMACH_REPHASELOCK_THRESHOLD - 1)) {
|
|
wait_phase_duration = wait_phase_duration + CONFIG_GNRC_GOMACH_CP_DURATION_US;
|
|
if ((uint32_t)wait_phase_duration > CONFIG_GNRC_GOMACH_SUPERFRAME_DURATION_US) {
|
|
wait_phase_duration = wait_phase_duration - CONFIG_GNRC_GOMACH_SUPERFRAME_DURATION_US;
|
|
}
|
|
}
|
|
|
|
if ((uint32_t)wait_phase_duration > CONFIG_GNRC_GOMACH_SUPERFRAME_DURATION_US) {
|
|
wait_phase_duration = wait_phase_duration % CONFIG_GNRC_GOMACH_SUPERFRAME_DURATION_US;
|
|
}
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_CP, (uint32_t)wait_phase_duration);
|
|
|
|
/* Flush the rx-queue. */
|
|
gnrc_priority_pktqueue_flush(&netif->mac.rx.queue);
|
|
|
|
netif->mac.tx.tx_busy_count = 0;
|
|
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_WAIT_CP;
|
|
gnrc_gomach_set_update(netif, false);
|
|
}
|
|
|
|
static void gomach_t2k_wait_cp(gnrc_netif_t *netif)
|
|
{
|
|
if (gnrc_gomach_timeout_is_expired(netif, GNRC_GOMACH_TIMEOUT_WAIT_CP)) {
|
|
gnrc_gomach_set_netdev_state(netif, NETOPT_STATE_IDLE);
|
|
/* Turn radio onto the neighbor's public channel, which will not change in this cycle. */
|
|
gnrc_gomach_turn_channel(netif, netif->mac.tx.current_neighbor->pub_chanseq);
|
|
|
|
/* Disable auto-ack, don't try to receive packet! */
|
|
gnrc_gomach_set_autoack(netif, NETOPT_DISABLE);
|
|
/* Require ACK for the packet waiting to be sent! */
|
|
gnrc_gomach_set_ack_req(netif, NETOPT_ENABLE);
|
|
|
|
/* Enable csma for sending the packet! */
|
|
netopt_enable_t csma_enable = NETOPT_ENABLE;
|
|
netif->dev->driver->set(netif->dev, NETOPT_CSMA, &csma_enable,
|
|
sizeof(netopt_enable_t));
|
|
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_TRANS_IN_CP;
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
}
|
|
|
|
static void gomach_t2k_trans_in_cp(gnrc_netif_t *netif)
|
|
{
|
|
/* To-do: should we add a rx-start security check and quit t2k when found
|
|
* ongoing transmissions? */
|
|
|
|
/* If we are retransmitting the packet, use the same sequence number for the
|
|
* packet to avoid duplicate packet reception at the receiver side. */
|
|
if ((netif->mac.tx.no_ack_counter > 0) || (netif->mac.tx.tx_busy_count > 0)) {
|
|
netdev_t *netdev = netif->dev;
|
|
netdev_ieee802154_t *device_state = container_of(netdev, netdev_ieee802154_t, netdev);
|
|
device_state->seq = netif->mac.tx.tx_seq;
|
|
}
|
|
|
|
/* Send the data packet here. */
|
|
int res = gnrc_gomach_send_data(netif, NETOPT_ENABLE);
|
|
if (res < 0) {
|
|
LOG_ERROR("ERROR: [GOMACH] t2k transmission fail: %d, drop packet.\n", res);
|
|
netif->mac.tx.no_ack_counter = 0;
|
|
|
|
/* If res is < 0, the data packet will not be released in send().
|
|
* so need to release the data here. */
|
|
if (netif->mac.tx.packet != NULL) {
|
|
gnrc_pktbuf_release(netif->mac.tx.packet);
|
|
netif->mac.tx.packet = NULL;
|
|
}
|
|
|
|
netif->mac.tx.current_neighbor = NULL;
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
return;
|
|
}
|
|
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_NO_TX_ISR, CONFIG_GNRC_GOMACH_NO_TX_ISR_US);
|
|
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_WAIT_CPTX_FEEDBACK;
|
|
gnrc_gomach_set_update(netif, false);
|
|
}
|
|
|
|
static void _cp_tx_success(gnrc_netif_t *netif)
|
|
{
|
|
/* Since the packet will not be released by the sending function,
|
|
* so, here, if TX success, we first release the packet. */
|
|
gnrc_pktbuf_release(netif->mac.tx.packet);
|
|
netif->mac.tx.packet = NULL;
|
|
|
|
/* Here is the phase-lock auto-adjust scheme. Use the new adjusted
|
|
* phase upon success. Here the new phase will be put ahead to the
|
|
* original phase. */
|
|
if (netif->mac.tx.no_ack_counter == (CONFIG_GNRC_GOMACH_REPHASELOCK_THRESHOLD - 2)) {
|
|
if (netif->mac.tx.current_neighbor->cp_phase >= CONFIG_GNRC_GOMACH_CP_DURATION_US) {
|
|
netif->mac.tx.current_neighbor->cp_phase -= CONFIG_GNRC_GOMACH_CP_DURATION_US;
|
|
}
|
|
else {
|
|
netif->mac.tx.current_neighbor->cp_phase += CONFIG_GNRC_GOMACH_SUPERFRAME_DURATION_US;
|
|
netif->mac.tx.current_neighbor->cp_phase -= CONFIG_GNRC_GOMACH_CP_DURATION_US;
|
|
}
|
|
}
|
|
/* Here is the phase-lock auto-adjust scheme. Use the new adjusted
|
|
* phase upon success. Here the new phase will be put behind the original
|
|
* phase. */
|
|
if (netif->mac.tx.no_ack_counter == (CONFIG_GNRC_GOMACH_REPHASELOCK_THRESHOLD - 1)) {
|
|
netif->mac.tx.current_neighbor->cp_phase +=
|
|
(CONFIG_GNRC_GOMACH_CP_DURATION_US + 20 * US_PER_MS);
|
|
|
|
if (netif->mac.tx.current_neighbor->cp_phase >=
|
|
CONFIG_GNRC_GOMACH_SUPERFRAME_DURATION_US) {
|
|
netif->mac.tx.current_neighbor->cp_phase -=
|
|
CONFIG_GNRC_GOMACH_SUPERFRAME_DURATION_US;
|
|
}
|
|
}
|
|
|
|
netif->mac.tx.no_ack_counter = 0;
|
|
netif->mac.tx.t2u_fail_count = 0;
|
|
|
|
/* If has pending packets, join the vTDMA period, first wait for receiver's beacon. */
|
|
if (gnrc_priority_pktqueue_length(&netif->mac.tx.current_neighbor->queue) > 0) {
|
|
netif->mac.tx.vtdma_para.slots_num = 0;
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_BEACON,
|
|
GNRC_GOMACH_WAIT_BEACON_TIME_US);
|
|
gnrc_priority_pktqueue_flush(&netif->mac.rx.queue);
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_WAIT_BEACON;
|
|
}
|
|
else {
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_END;
|
|
}
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
|
|
static bool _cp_tx_busy(gnrc_netif_t *netif)
|
|
{
|
|
/* If the channel busy counter is below threshold, retry CSMA immediately,
|
|
* by knowing that the CP will be automatically extended. */
|
|
if (netif->mac.tx.tx_busy_count < CONFIG_GNRC_GOMACH_TX_BUSY_THRESHOLD) {
|
|
netif->mac.tx.tx_busy_count++;
|
|
|
|
/* Store the TX sequence number for this packet. Always use the same
|
|
* sequence number for sending the same packet, to avoid duplicated
|
|
* packet reception at the receiver. */
|
|
netdev_t *netdev = netif->dev;
|
|
netdev_ieee802154_t *device_state = container_of(netdev, netdev_ieee802154_t, netdev);
|
|
netif->mac.tx.tx_seq = device_state->seq - 1;
|
|
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_TRANS_IN_CP;
|
|
gnrc_gomach_set_update(netif, true);
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static void _cp_tx_default(gnrc_netif_t *netif)
|
|
{
|
|
netif->mac.tx.no_ack_counter++;
|
|
|
|
LOG_DEBUG("[GOMACH] t2k %d times No-ACK.\n", netif->mac.tx.no_ack_counter);
|
|
|
|
/* This packet will be retried. Store the TX sequence number for this packet.
|
|
* Always use the same sequence number for sending the same packet. */
|
|
netdev_t *netdev = netif->dev;
|
|
netdev_ieee802154_t *device_state = container_of(netdev, netdev_ieee802154_t, netdev);
|
|
netif->mac.tx.tx_seq = device_state->seq - 1;
|
|
|
|
/* If no_ack_counter reaches the threshold, regarded as phase-lock failed. So
|
|
* retry to send the packet in t2u, i.e., try to phase-lock with the receiver
|
|
* again. */
|
|
if (netif->mac.tx.no_ack_counter >= CONFIG_GNRC_GOMACH_REPHASELOCK_THRESHOLD) {
|
|
LOG_DEBUG("[GOMACH] t2k failed, go to t2u.\n");
|
|
/* Here, we don't queue the packet again, but keep it in tx.packet. */
|
|
netif->mac.tx.current_neighbor->mac_type = GNRC_GOMACH_TYPE_UNKNOWN;
|
|
netif->mac.tx.t2u_retry_counter = 0;
|
|
}
|
|
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
|
|
static void gomach_t2k_wait_cp_txfeedback(gnrc_netif_t *netif)
|
|
{
|
|
if (gnrc_gomach_timeout_is_expired(netif, GNRC_GOMACH_TIMEOUT_NO_TX_ISR)) {
|
|
/* No TX-ISR, go to sleep. */
|
|
netif->mac.tx.no_ack_counter++;
|
|
|
|
netdev_t *netdev = netif->dev;
|
|
netdev_ieee802154_t *device_state = container_of(netdev, netdev_ieee802154_t, netdev);
|
|
netif->mac.tx.tx_seq = device_state->seq - 1;
|
|
|
|
/* Here, we don't queue the packet again, but keep it in tx.packet. */
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_NO_TX_ISR);
|
|
return;
|
|
}
|
|
|
|
if (gnrc_gomach_get_tx_finish(netif)) {
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_NO_TX_ISR);
|
|
|
|
switch (gnrc_netif_get_tx_feedback(netif)) {
|
|
case TX_FEEDBACK_SUCCESS: {
|
|
_cp_tx_success(netif);
|
|
break;
|
|
}
|
|
case TX_FEEDBACK_BUSY:
|
|
if (!_cp_tx_busy(netif)) {
|
|
return;
|
|
}
|
|
/* Intentionally falls through */
|
|
case TX_FEEDBACK_NOACK:
|
|
default: {
|
|
_cp_tx_default(netif);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void gomach_t2k_wait_beacon(gnrc_netif_t *netif)
|
|
{
|
|
/* Process the beacon if we receive it. */
|
|
if (gnrc_gomach_get_pkt_received(netif)) {
|
|
gnrc_gomach_set_pkt_received(netif, false);
|
|
gnrc_gomach_packet_process_in_wait_beacon(netif);
|
|
}
|
|
|
|
/* If we need to quit t2k, don't release the current neighbor pointer. In the
|
|
* next cycle, we will try to send to the same receiver. */
|
|
if (gnrc_gomach_get_quit_cycle(netif)) {
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_BEACON);
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
return;
|
|
}
|
|
|
|
if (netif->mac.tx.vtdma_para.slots_num > 0) {
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_BEACON);
|
|
|
|
/* If the sender gets allocated slots, go to attend the receiver's vTDMA for
|
|
* burst sending all the pending packets to the receiver. */
|
|
if (netif->mac.tx.vtdma_para.slots_num > 0) {
|
|
/* Switch the radio to the sub-channel of the receiver. */
|
|
gnrc_gomach_turn_channel(netif, netif->mac.tx.vtdma_para.sub_channel_seq);
|
|
|
|
/* If the allocated slots period is not right behind the beacon, i.e., not the first
|
|
* one, turn off the radio and wait for its own slots period. */
|
|
if (netif->mac.tx.vtdma_para.slots_position > 0) {
|
|
gnrc_gomach_set_netdev_state(netif, NETOPT_STATE_SLEEP);
|
|
|
|
uint32_t wait_slots_duration = netif->mac.tx.vtdma_para.slots_position *
|
|
CONFIG_GNRC_GOMACH_VTDMA_SLOT_SIZE_US;
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_SLOTS,
|
|
wait_slots_duration);
|
|
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_WAIT_SLOTS;
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
else {
|
|
/* If the allocated slots period is the first one in vTDMA,
|
|
* start sending packets. */
|
|
gnrc_pktsnip_t *pkt =
|
|
gnrc_priority_pktqueue_pop(&(netif->mac.tx.current_neighbor->queue));
|
|
if (pkt != NULL) {
|
|
netif->mac.tx.packet = pkt;
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_VTDMA_TRANS;
|
|
}
|
|
else {
|
|
LOG_ERROR("ERROR: [GOMACH] t2k vTDMA: null packet.\n");
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_END;
|
|
}
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
}
|
|
else {
|
|
/* No slots get allocated, go to t2k end. */
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
return;
|
|
}
|
|
|
|
/* If no beacon during waiting period, go to t2k end.
|
|
* Or, if we have received beacon, but find no allocated slots,
|
|
* go to t2k as well. */
|
|
if (gnrc_gomach_timeout_is_expired(netif, GNRC_GOMACH_TIMEOUT_WAIT_BEACON) ||
|
|
!gnrc_gomach_timeout_is_running(netif, GNRC_GOMACH_TIMEOUT_WAIT_BEACON)) {
|
|
gnrc_priority_pktqueue_flush(&netif->mac.rx.queue);
|
|
LOG_DEBUG("[GOMACH] t2k: no beacon.\n");
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
}
|
|
|
|
static void gomach_t2k_wait_own_slots(gnrc_netif_t *netif)
|
|
{
|
|
if (gnrc_gomach_timeout_is_expired(netif, GNRC_GOMACH_TIMEOUT_WAIT_SLOTS)) {
|
|
/* The node is now in its scheduled slots period, start burst sending packets. */
|
|
gnrc_gomach_set_netdev_state(netif, NETOPT_STATE_IDLE);
|
|
|
|
gnrc_pktsnip_t *pkt = gnrc_priority_pktqueue_pop(&(netif->mac.tx.current_neighbor->queue));
|
|
if (pkt != NULL) {
|
|
netif->mac.tx.packet = pkt;
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_VTDMA_TRANS;
|
|
}
|
|
else {
|
|
LOG_ERROR("ERROR: [GOMACH] t2k vTDMA: null packet.\n");
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_END;
|
|
}
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
}
|
|
|
|
static void gomach_t2k_trans_in_slots(gnrc_netif_t *netif)
|
|
{
|
|
/* If this packet is being retransmitted, use the same recorded MAC sequence number. */
|
|
if (netif->mac.tx.no_ack_counter > 0) {
|
|
netdev_t *netdev = netif->dev;
|
|
netdev_ieee802154_t *device_state = container_of(netdev, netdev_ieee802154_t, netdev);
|
|
device_state->seq = netif->mac.tx.tx_seq;
|
|
}
|
|
|
|
/* Send data packet in its allocated slots (scheduled slots period). */
|
|
int res = gnrc_gomach_send_data(netif, NETOPT_DISABLE);
|
|
if (res < 0) {
|
|
LOG_ERROR("ERROR: [GOMACH] t2k vTDMA transmission fail: %d, drop packet.\n", res);
|
|
|
|
/* If res is < 0, the data packet will not be released in send().
|
|
* so need to release the data here. */
|
|
if (netif->mac.tx.packet != NULL) {
|
|
gnrc_pktbuf_release(netif->mac.tx.packet);
|
|
netif->mac.tx.packet = NULL;
|
|
}
|
|
netif->mac.tx.current_neighbor = NULL;
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
return;
|
|
}
|
|
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_NO_TX_ISR, CONFIG_GNRC_GOMACH_NO_TX_ISR_US);
|
|
|
|
netif->mac.tx.vtdma_para.slots_num--;
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_WAIT_VTDMA_FEEDBACK;
|
|
gnrc_gomach_set_update(netif, false);
|
|
}
|
|
|
|
static void _t2k_wait_vtdma_tx_success(gnrc_netif_t *netif)
|
|
{
|
|
/* First release the packet. */
|
|
gnrc_pktbuf_release(netif->mac.tx.packet);
|
|
netif->mac.tx.packet = NULL;
|
|
netif->mac.tx.no_ack_counter = 0;
|
|
|
|
/* If the sender has pending packets and scheduled slots,
|
|
* continue vTDMA transmission. */
|
|
if ((netif->mac.tx.vtdma_para.slots_num > 0) &&
|
|
(gnrc_priority_pktqueue_length(&netif->mac.tx.current_neighbor->queue) > 0)) {
|
|
gnrc_pktsnip_t *pkt = gnrc_priority_pktqueue_pop(&netif->mac.tx.current_neighbor->queue);
|
|
if (pkt != NULL) {
|
|
netif->mac.tx.packet = pkt;
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_VTDMA_TRANS;
|
|
}
|
|
else {
|
|
LOG_ERROR("ERROR: [GOMACH] t2k vTDMA: null packet.\n");
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_END;
|
|
}
|
|
}
|
|
else {
|
|
/* If no scheduled slots or pending packets, go to t2k end. */
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_END;
|
|
}
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
|
|
static void _t2k_wait_vtdma_tx_default(gnrc_netif_t *netif)
|
|
{
|
|
/* In case of transmission failure in vTDMA, retransmit the packet in the next
|
|
* scheduled slot, or the next cycle's t2k procedure. */
|
|
|
|
/* Firstly, mark the current TX packet as not ACKed and record the MAC sequence
|
|
* number, such that the MAC will use the same MAC sequence to send it.
|
|
* Also, by marking no_ack_counter as non-zero, the neighbor and packet pointers
|
|
* will then not be released in t2k-end. Then, the packet can be retried right in
|
|
* the following cycle. */
|
|
netif->mac.tx.no_ack_counter = 1;
|
|
|
|
netdev_t *netdev = netif->dev;
|
|
netdev_ieee802154_t *device_state = container_of(netdev, netdev_ieee802154_t, netdev);
|
|
netif->mac.tx.tx_seq = device_state->seq - 1;
|
|
|
|
/* Do not release the packet here, continue sending the same packet. ***/
|
|
if (netif->mac.tx.vtdma_para.slots_num > 0) {
|
|
LOG_DEBUG("[GOMACH] no ACK in vTDMA, retry in next slot.\n");
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_VTDMA_TRANS;
|
|
}
|
|
else {
|
|
/* If no slots for sending, retry in next cycle's t2r, without releasing
|
|
* tx.packet pointer. */
|
|
LOG_DEBUG("[GOMACH] no ACK in vTDMA, retry in next cycle.\n");
|
|
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_END;
|
|
}
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
|
|
static void gomach_t2k_wait_vtdma_transfeedback(gnrc_netif_t *netif)
|
|
{
|
|
if (gnrc_gomach_timeout_is_expired(netif, GNRC_GOMACH_TIMEOUT_NO_TX_ISR)) {
|
|
/* No TX-ISR, go to sleep. */
|
|
netif->mac.tx.no_ack_counter++;
|
|
|
|
netdev_t *netdev = netif->dev;
|
|
netdev_ieee802154_t *device_state = container_of(netdev, netdev_ieee802154_t, netdev);
|
|
netif->mac.tx.tx_seq = device_state->seq - 1;
|
|
|
|
/* Here, we don't queue the packet again, but keep it in tx.packet. */
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_NO_TX_ISR);
|
|
return;
|
|
}
|
|
|
|
if (gnrc_gomach_get_tx_finish(netif)) {
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_NO_TX_ISR);
|
|
|
|
switch (gnrc_netif_get_tx_feedback(netif)) {
|
|
case TX_FEEDBACK_SUCCESS: {
|
|
_t2k_wait_vtdma_tx_success(netif);
|
|
break;
|
|
}
|
|
case TX_FEEDBACK_BUSY:
|
|
case TX_FEEDBACK_NOACK:
|
|
default: {
|
|
_t2k_wait_vtdma_tx_default(netif);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void gomach_t2k_end(gnrc_netif_t *netif)
|
|
{
|
|
gnrc_gomach_set_netdev_state(netif, NETOPT_STATE_SLEEP);
|
|
|
|
/* In GoMacH, normally, in case of transmission failure, no packet will be released
|
|
* in t2k. Failed packet will only be released in t2u. In case of continuous t2k
|
|
* failures, the MAC will goto t2u to retry the packet without releasing it here. */
|
|
if ((netif->mac.tx.packet != NULL) && (netif->mac.tx.no_ack_counter == 0)) {
|
|
LOG_ERROR("ERROR: [GOMACH] t2k: releasing unexpected packet!\n");
|
|
gnrc_pktbuf_release(netif->mac.tx.packet);
|
|
netif->mac.tx.packet = NULL;
|
|
}
|
|
|
|
/* In case no_ack_counter is not 0 here, it means we will retry to send the packet
|
|
* in t2k or t2u, then, don't release the neighbor pointer. */
|
|
if (netif->mac.tx.no_ack_counter == 0) {
|
|
netif->mac.tx.current_neighbor = NULL;
|
|
}
|
|
|
|
/* Clear all timeouts. */
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_CP);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_BEACON);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_SLOTS);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_NO_TX_ISR);
|
|
|
|
/* Reset t2k_state to the initial state. */
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_INIT;
|
|
|
|
netif->mac.prot.gomach.basic_state = GNRC_GOMACH_LISTEN;
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_SLEEP;
|
|
gnrc_gomach_set_enter_new_cycle(netif, false);
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
|
|
static void gomach_t2k_update(gnrc_netif_t *netif)
|
|
{
|
|
/* State machine of GoMacH's t2k (transmit to phase-known device) procedure. */
|
|
switch (netif->mac.tx.t2k_state) {
|
|
case GNRC_GOMACH_T2K_INIT: {
|
|
gomach_t2k_init(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_T2K_WAIT_CP: {
|
|
gomach_t2k_wait_cp(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_T2K_TRANS_IN_CP: {
|
|
gomach_t2k_trans_in_cp(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_T2K_WAIT_CPTX_FEEDBACK: {
|
|
gomach_t2k_wait_cp_txfeedback(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_T2K_WAIT_BEACON: {
|
|
gomach_t2k_wait_beacon(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_T2K_WAIT_SLOTS: {
|
|
gomach_t2k_wait_own_slots(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_T2K_VTDMA_TRANS: {
|
|
gomach_t2k_trans_in_slots(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_T2K_WAIT_VTDMA_FEEDBACK: {
|
|
gomach_t2k_wait_vtdma_transfeedback(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_T2K_END: {
|
|
gomach_t2k_end(netif);
|
|
break;
|
|
}
|
|
default: break;
|
|
}
|
|
}
|
|
|
|
static void gomach_t2u_init(gnrc_netif_t *netif)
|
|
{
|
|
/* since t2u is right following CP period (wake-up period), the radio is still on,
|
|
* so we don't need to turn on it again. */
|
|
|
|
LOG_DEBUG("[GOMACH] t2u initialization.\n");
|
|
|
|
gnrc_netif_set_rx_started(netif, false);
|
|
gnrc_gomach_set_quit_cycle(netif, false);
|
|
gnrc_gomach_set_pkt_received(netif, false);
|
|
netif->mac.tx.preamble_sent = 0;
|
|
gnrc_gomach_set_got_preamble_ack(netif, false);
|
|
gnrc_gomach_set_buffer_full(netif, false);
|
|
|
|
/* Start sending the preamble firstly on public channel 1. */
|
|
gnrc_gomach_turn_channel(netif, netif->mac.prot.gomach.pub_channel_1);
|
|
|
|
/* Disable auto-ACK here! Don't try to reply ACK to any node. */
|
|
gnrc_gomach_set_autoack(netif, NETOPT_DISABLE);
|
|
|
|
gnrc_gomach_set_on_pubchan_1(netif, true);
|
|
|
|
gnrc_priority_pktqueue_flush(&netif->mac.rx.queue);
|
|
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_PREAMBLE_PREPARE;
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
|
|
static void gomach_t2u_send_preamble_prepare(gnrc_netif_t *netif)
|
|
{
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_MAX_PREAM_INTERVAL);
|
|
|
|
if (netif->mac.tx.preamble_sent != 0) {
|
|
/* Toggle the radio channel after each preamble transmission. */
|
|
if (gnrc_gomach_get_on_pubchan_1(netif)) {
|
|
gnrc_gomach_turn_channel(netif, netif->mac.prot.gomach.pub_channel_2);
|
|
gnrc_gomach_set_on_pubchan_1(netif, false);
|
|
}
|
|
else {
|
|
gnrc_gomach_turn_channel(netif, netif->mac.prot.gomach.pub_channel_1);
|
|
gnrc_gomach_set_on_pubchan_1(netif, true);
|
|
}
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_MAX_PREAM_INTERVAL,
|
|
CONFIG_GNRC_GOMACH_MAX_PREAM_INTERVAL_US);
|
|
}
|
|
else {
|
|
/* Here, for the first preamble, we set the pream_max_interval timeout to
|
|
* 5*MAX_PREAM_INTERVAL due to the fact that the first preamble is
|
|
* using csma for sending, and csma costs some time before actually sending
|
|
* the packet. */
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_MAX_PREAM_INTERVAL,
|
|
(5 * CONFIG_GNRC_GOMACH_MAX_PREAM_INTERVAL_US));
|
|
}
|
|
|
|
gnrc_gomach_set_max_pream_interv(netif, false);
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_SEND_PREAMBLE;
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
|
|
static void gomach_t2u_send_preamble(gnrc_netif_t *netif)
|
|
{
|
|
/* Now, start sending preamble. */
|
|
int res;
|
|
|
|
/* The first preamble is sent with csma for collision avoidance. */
|
|
if (netif->mac.tx.preamble_sent == 0) {
|
|
res = gnrc_gomach_send_preamble(netif, NETOPT_ENABLE);
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_PREAM_DURATION,
|
|
GNRC_GOMACH_PREAMBLE_DURATION_US);
|
|
}
|
|
else {
|
|
res = gnrc_gomach_send_preamble(netif, NETOPT_DISABLE);
|
|
}
|
|
|
|
if (res < 0) {
|
|
LOG_ERROR("ERROR: [GOMACH] t2u send preamble failed: %d\n", res);
|
|
}
|
|
|
|
/* In case that packet-buffer is full, quit t2u and release packet. */
|
|
if (res == -ENOBUFS) {
|
|
LOG_ERROR("ERROR: [GOMACH] t2u: no pkt-buffer for sending preamble.\n");
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_PREAM_DURATION);
|
|
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
return;
|
|
}
|
|
|
|
gnrc_netif_set_rx_started(netif, false);
|
|
netif->mac.tx.preamble_sent++;
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_WAIT_PREAMBLE_TX;
|
|
gnrc_gomach_set_update(netif, false);
|
|
}
|
|
|
|
static void gomach_t2u_wait_preamble_tx(gnrc_netif_t *netif)
|
|
{
|
|
if (gnrc_gomach_get_tx_finish(netif)) {
|
|
/* Set preamble interval timeout. This is a very short timeout (1ms),
|
|
* just to catch the rx-start event of receiving possible preamble-ACK. */
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_PREAMBLE,
|
|
CONFIG_GNRC_GOMACH_PREAMBLE_INTERVAL_US);
|
|
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_WAIT_PREAMBLE_ACK;
|
|
gnrc_gomach_set_update(netif, false);
|
|
return;
|
|
}
|
|
|
|
/* This is mainly to handle no-TX-complete error. Once the max preamble interval
|
|
* timeout expired here (i.e., no-TX-complete error), we will quit waiting here
|
|
* and go to send the next preamble, thus the MAC will not get stuck here. */
|
|
if (gnrc_gomach_timeout_is_expired(netif, GNRC_GOMACH_TIMEOUT_MAX_PREAM_INTERVAL)) {
|
|
gnrc_priority_pktqueue_flush(&netif->mac.rx.queue);
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_PREAMBLE_PREPARE;
|
|
gnrc_gomach_set_update(netif, true);
|
|
return;
|
|
}
|
|
}
|
|
|
|
static bool _handle_in_t2u_send_preamble(gnrc_netif_t *netif)
|
|
{
|
|
/* If packet buffer is full, release one packet to release memory,
|
|
* and reload the next packet.
|
|
* In t2u, we need at least some minimum memory to build the preamble packet. */
|
|
if (gnrc_gomach_get_buffer_full(netif)) {
|
|
gnrc_gomach_set_buffer_full(netif, false);
|
|
|
|
gnrc_gomach_set_update(netif, true);
|
|
|
|
/* To-do: should we release all the buffered packets in the queue to
|
|
* release memory in such a critical situation? */
|
|
LOG_DEBUG("[GOMACH] t2u: pkt-buffer full, release one pkt.\n");
|
|
|
|
if (netif->mac.tx.packet != NULL) {
|
|
gnrc_pktbuf_release(netif->mac.tx.packet);
|
|
netif->mac.tx.packet = NULL;
|
|
netif->mac.tx.no_ack_counter = 0;
|
|
}
|
|
/* Reload the next packet in the neighbor's queue. */
|
|
gnrc_pktsnip_t *pkt = gnrc_priority_pktqueue_pop(&netif->mac.tx.current_neighbor->queue);
|
|
|
|
if (pkt != NULL) {
|
|
netif->mac.tx.packet = pkt;
|
|
}
|
|
else {
|
|
LOG_DEBUG("[GOMACH] t2u: null packet, quit t2u.\n");
|
|
netif->mac.tx.current_neighbor = NULL;
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_END;
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (gnrc_gomach_timeout_is_expired(netif, GNRC_GOMACH_TIMEOUT_MAX_PREAM_INTERVAL)) {
|
|
gnrc_gomach_set_max_pream_interv(netif, true);
|
|
return true;
|
|
}
|
|
|
|
/* if we are receiving packet, wait until RX is completed. */
|
|
if ((!gnrc_gomach_timeout_is_running(netif, GNRC_GOMACH_TIMEOUT_WAIT_RX_END)) &&
|
|
gnrc_netif_get_rx_started(netif) &&
|
|
(!gnrc_gomach_get_max_pream_interv(netif))) {
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_PREAMBLE);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_MAX_PREAM_INTERVAL);
|
|
|
|
/* Set a timeout to wait for the complete of reception. */
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_RX_END);
|
|
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_RX_END,
|
|
CONFIG_GNRC_GOMACH_WAIT_RX_END_US);
|
|
return false;
|
|
}
|
|
|
|
if (gnrc_gomach_get_pkt_received(netif)) {
|
|
gnrc_gomach_set_pkt_received(netif, false);
|
|
gnrc_gomach_process_pkt_in_wait_preamble_ack(netif);
|
|
}
|
|
|
|
/* Quit t2u if we have to, e.g., the device found ongoing bcast of other devices. */
|
|
if (gnrc_gomach_get_quit_cycle(netif)) {
|
|
LOG_DEBUG("[GOMACH] quit t2u.\n");
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_RX_END);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_PREAMBLE);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_PREAM_DURATION);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_MAX_PREAM_INTERVAL);
|
|
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
return false;
|
|
}
|
|
|
|
if (gnrc_gomach_timeout_is_expired(netif, GNRC_GOMACH_TIMEOUT_WAIT_RX_END)) {
|
|
gnrc_gomach_set_max_pream_interv(netif, true);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static void gomach_t2u_wait_preamble_ack(gnrc_netif_t *netif)
|
|
{
|
|
if (!_handle_in_t2u_send_preamble(netif)) {
|
|
return;
|
|
}
|
|
|
|
if (gnrc_gomach_get_got_preamble_ack(netif)) {
|
|
/* Require ACK for the packet waiting to be sent! */
|
|
gnrc_gomach_set_ack_req(netif, NETOPT_ENABLE);
|
|
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_PREAMBLE);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_PREAM_DURATION);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_MAX_PREAM_INTERVAL);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_RX_END);
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_SEND_DATA;
|
|
netif->mac.tx.t2u_fail_count = 0;
|
|
gnrc_gomach_set_update(netif, true);
|
|
return;
|
|
}
|
|
|
|
if (gnrc_gomach_timeout_is_expired(netif, GNRC_GOMACH_TIMEOUT_PREAM_DURATION)) {
|
|
netif->mac.tx.t2u_retry_counter++;
|
|
|
|
/* If we reach the maximum t2u retry limit, release the data packet. */
|
|
if (netif->mac.tx.t2u_retry_counter >= CONFIG_GNRC_GOMACH_T2U_RETYR_THRESHOLD) {
|
|
LOG_DEBUG("[GOMACH] t2u failed: no preamble-ACK.\n");
|
|
netif->mac.tx.t2u_retry_counter = 0;
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_END;
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_PREAMBLE);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_RX_END);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_MAX_PREAM_INTERVAL);
|
|
|
|
netif->mac.tx.t2u_fail_count++;
|
|
}
|
|
else {
|
|
/* If we haven't reach the maximum t2u limit, try again. Set quit_current_cycle
|
|
* flag to true such that we will release the current neighbor pointer. */
|
|
gnrc_gomach_set_quit_cycle(netif, true);
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_END;
|
|
}
|
|
|
|
gnrc_gomach_set_update(netif, true);
|
|
return;
|
|
}
|
|
|
|
/* If we didn't catch the RX-start event, go to send the next preamble. */
|
|
if ((gnrc_gomach_timeout_is_expired(netif, GNRC_GOMACH_TIMEOUT_PREAMBLE)) ||
|
|
gnrc_gomach_get_max_pream_interv(netif)) {
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_PREAMBLE_PREPARE;
|
|
gnrc_gomach_set_update(netif, true);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_PREAMBLE);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_MAX_PREAM_INTERVAL);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_RX_END);
|
|
}
|
|
}
|
|
|
|
static void gomach_t2u_send_data(gnrc_netif_t *netif)
|
|
{
|
|
/* If we are retrying to send the data, reload its original MAC sequence. */
|
|
if (netif->mac.tx.no_ack_counter > 0) {
|
|
netdev_t *netdev = netif->dev;
|
|
netdev_ieee802154_t *device_state = container_of(netdev, netdev_ieee802154_t, netdev);
|
|
device_state->seq = netif->mac.tx.tx_seq;
|
|
}
|
|
|
|
/* Here, we send the data to the receiver. */
|
|
int res = gnrc_gomach_send_data(netif, NETOPT_ENABLE);
|
|
if (res < 0) {
|
|
LOG_ERROR("ERROR: [GOMACH] t2u data sending error: %d.\n", res);
|
|
|
|
/* If res is < 0, the data packet will not be released in send().
|
|
* so need to release the data here. */
|
|
if (netif->mac.tx.packet != NULL) {
|
|
gnrc_pktbuf_release(netif->mac.tx.packet);
|
|
netif->mac.tx.packet = NULL;
|
|
}
|
|
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
return;
|
|
}
|
|
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_NO_TX_ISR, CONFIG_GNRC_GOMACH_NO_TX_ISR_US);
|
|
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_WAIT_DATA_TX;
|
|
gnrc_gomach_set_update(netif, false);
|
|
}
|
|
|
|
static void _t2u_data_tx_success(gnrc_netif_t *netif)
|
|
{
|
|
/* If transmission succeeded, release the data. */
|
|
gnrc_pktbuf_release(netif->mac.tx.packet);
|
|
netif->mac.tx.packet = NULL;
|
|
|
|
netif->mac.tx.no_ack_counter = 0;
|
|
netif->mac.tx.t2u_retry_counter = 0;
|
|
|
|
/* Attend the vTDMA procedure if the sender has pending packets for the receiver. */
|
|
if (gnrc_priority_pktqueue_length(&netif->mac.tx.current_neighbor->queue) > 0) {
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_INIT;
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_PREAMBLE);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_PREAM_DURATION);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_RX_END);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_MAX_PREAM_INTERVAL);
|
|
|
|
/* Switch to t2k procedure and wait for the beacon of the receiver. */
|
|
netif->mac.tx.vtdma_para.slots_num = 0;
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_BEACON,
|
|
GNRC_GOMACH_WAIT_BEACON_TIME_US);
|
|
gnrc_priority_pktqueue_flush(&netif->mac.rx.queue);
|
|
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_WAIT_BEACON;
|
|
netif->mac.tx.transmit_state = GNRC_GOMACH_TRANS_TO_KNOWN;
|
|
}
|
|
else {
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_END;
|
|
}
|
|
}
|
|
|
|
static void _t2u_data_tx_fail(gnrc_netif_t *netif)
|
|
{
|
|
netif->mac.tx.t2u_retry_counter++;
|
|
/* If we meet t2u retry limit, release the packet. */
|
|
if (netif->mac.tx.t2u_retry_counter >= CONFIG_GNRC_GOMACH_T2U_RETYR_THRESHOLD) {
|
|
LOG_DEBUG("[GOMACH] t2u send data failed on channel %d,"
|
|
" drop packet.\n", netif->mac.tx.current_neighbor->pub_chanseq);
|
|
gnrc_pktbuf_release(netif->mac.tx.packet);
|
|
netif->mac.tx.packet = NULL;
|
|
netif->mac.tx.current_neighbor = NULL;
|
|
netif->mac.tx.no_ack_counter = 0;
|
|
netif->mac.tx.t2u_retry_counter = 0;
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_END;
|
|
}
|
|
else {
|
|
/* Record the MAC sequence of the data, retry t2u in next cycle. */
|
|
netif->mac.tx.no_ack_counter = CONFIG_GNRC_GOMACH_REPHASELOCK_THRESHOLD;
|
|
netdev_t *netdev = netif->dev;
|
|
netdev_ieee802154_t *device_state = container_of(netdev, netdev_ieee802154_t, netdev);
|
|
netif->mac.tx.tx_seq = device_state->seq - 1;
|
|
|
|
LOG_DEBUG("[GOMACH] t2u send data failed on channel %d.\n",
|
|
netif->mac.tx.current_neighbor->pub_chanseq);
|
|
/* Set quit_current_cycle to true, thus not to release current_neighbour pointer
|
|
* in t2u-end */
|
|
gnrc_gomach_set_quit_cycle(netif, true);
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_END;
|
|
}
|
|
}
|
|
|
|
static void gomach_t2u_wait_tx_feedback(gnrc_netif_t *netif)
|
|
{
|
|
if (gnrc_gomach_timeout_is_expired(netif, GNRC_GOMACH_TIMEOUT_NO_TX_ISR)) {
|
|
/* No TX-ISR, go to sleep. */
|
|
netif->mac.tx.t2u_retry_counter++;
|
|
|
|
netif->mac.tx.no_ack_counter = CONFIG_GNRC_GOMACH_REPHASELOCK_THRESHOLD;
|
|
netdev_t *netdev = netif->dev;
|
|
netdev_ieee802154_t *device_state = container_of(netdev, netdev_ieee802154_t, netdev);
|
|
netif->mac.tx.tx_seq = device_state->seq - 1;
|
|
|
|
gnrc_gomach_set_quit_cycle(netif, true);
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_END;
|
|
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_NO_TX_ISR);
|
|
gnrc_gomach_set_update(netif, true);
|
|
return;
|
|
}
|
|
|
|
if (gnrc_gomach_get_tx_finish(netif)) {
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_NO_TX_ISR);
|
|
|
|
if (gnrc_netif_get_tx_feedback(netif) == TX_FEEDBACK_SUCCESS) {
|
|
_t2u_data_tx_success(netif);
|
|
}
|
|
else {
|
|
_t2u_data_tx_fail(netif);
|
|
}
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
}
|
|
|
|
static void gomach_t2u_end(gnrc_netif_t *netif)
|
|
{
|
|
gnrc_gomach_set_netdev_state(netif, NETOPT_STATE_SLEEP);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_PREAMBLE);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_PREAM_DURATION);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_RX_END);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_MAX_PREAM_INTERVAL);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_NO_TX_ISR);
|
|
|
|
/* In case quit_current_cycle is true, don't release neighbor pointer,
|
|
* will retry t2u immediately in next cycle.*/
|
|
if (!gnrc_gomach_get_quit_cycle(netif)) {
|
|
if (netif->mac.tx.packet != NULL) {
|
|
gnrc_pktbuf_release(netif->mac.tx.packet);
|
|
netif->mac.tx.packet = NULL;
|
|
netif->mac.tx.no_ack_counter = 0;
|
|
LOG_WARNING("WARNING: [GOMACH] t2u: drop packet.\n");
|
|
}
|
|
netif->mac.tx.current_neighbor = NULL;
|
|
}
|
|
|
|
/* Reset t2u state. */
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_INIT;
|
|
|
|
/* Resume to listen state and go to sleep. */
|
|
netif->mac.prot.gomach.basic_state = GNRC_GOMACH_LISTEN;
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_SLEEP;
|
|
gnrc_gomach_set_enter_new_cycle(netif, false);
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
|
|
static void gomach_t2u_update(gnrc_netif_t *netif)
|
|
{
|
|
/* State machine of GoMacH's t2u (transmit to phase-unknown device) procedure. */
|
|
switch (netif->mac.tx.t2u_state) {
|
|
case GNRC_GOMACH_T2U_INIT: {
|
|
gomach_t2u_init(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_T2U_PREAMBLE_PREPARE: {
|
|
gomach_t2u_send_preamble_prepare(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_T2U_SEND_PREAMBLE: {
|
|
gomach_t2u_send_preamble(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_T2U_WAIT_PREAMBLE_TX: {
|
|
gomach_t2u_wait_preamble_tx(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_T2U_WAIT_PREAMBLE_ACK: {
|
|
gomach_t2u_wait_preamble_ack(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_T2U_SEND_DATA: {
|
|
gomach_t2u_send_data(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_T2U_WAIT_DATA_TX: {
|
|
gomach_t2u_wait_tx_feedback(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_T2U_END: {
|
|
gomach_t2u_end(netif);
|
|
break;
|
|
}
|
|
default: break;
|
|
}
|
|
}
|
|
|
|
static void _gomach_phase_backoff(gnrc_netif_t *netif)
|
|
{
|
|
/* Execute phase backoff for avoiding CP (wake-up period) overlap. */
|
|
rtt_clear_alarm();
|
|
xtimer_usleep(netif->mac.prot.gomach.backoff_phase_us);
|
|
|
|
rtt_set_counter(0);
|
|
netif->mac.prot.gomach.last_wakeup = rtt_get_counter();
|
|
|
|
uint32_t alarm = netif->mac.prot.gomach.last_wakeup +
|
|
RTT_US_TO_TICKS(CONFIG_GNRC_GOMACH_SUPERFRAME_DURATION_US);
|
|
|
|
rtt_set_alarm(alarm, _gomach_rtt_cb, NULL);
|
|
|
|
gnrc_gomach_update_neighbor_phase(netif);
|
|
|
|
LOG_INFO("INFO: [GOMACH] phase backoffed: %lu us.\n",
|
|
(unsigned long)netif->mac.prot.gomach.backoff_phase_us);
|
|
}
|
|
|
|
static void gomach_listen_init(gnrc_netif_t *netif)
|
|
{
|
|
/* Reset last_seq_info, for avoiding receiving duplicate packets.
|
|
* To-do: remove this in the future? */
|
|
for (uint8_t i = 0; i < GNRC_GOMACH_DUPCHK_BUFFER_SIZE; i++) {
|
|
if (netif->mac.rx.check_dup_pkt.last_nodes[i].node_addr.len != 0) {
|
|
netif->mac.rx.check_dup_pkt.last_nodes[i].life_cycle++;
|
|
if (netif->mac.rx.check_dup_pkt.last_nodes[i].life_cycle >=
|
|
CONFIG_GNRC_GOMACH_RX_DUPCHK_UNIT_LIFE) {
|
|
netif->mac.rx.check_dup_pkt.last_nodes[i].node_addr.len = 0;
|
|
netif->mac.rx.check_dup_pkt.last_nodes[i].node_addr.addr[0] = 0;
|
|
netif->mac.rx.check_dup_pkt.last_nodes[i].node_addr.addr[1] = 0;
|
|
netif->mac.rx.check_dup_pkt.last_nodes[i].seq = 0;
|
|
netif->mac.rx.check_dup_pkt.last_nodes[i].life_cycle = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (netif->mac.tx.t2u_fail_count >= CONFIG_GNRC_GOMACH_MAX_T2U_RETYR_THRESHOLD) {
|
|
netif->mac.tx.t2u_fail_count = 0;
|
|
LOG_DEBUG("[GOMACH]: Re-initialize radio.");
|
|
gomach_reinit_radio(netif);
|
|
}
|
|
gnrc_gomach_set_enter_new_cycle(netif, false);
|
|
|
|
/* Set listen period timeout. */
|
|
uint32_t listen_period = random_uint32_range(0, CONFIG_GNRC_GOMACH_CP_RANDOM_END_US) +
|
|
CONFIG_GNRC_GOMACH_CP_DURATION_US;
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_CP_END, listen_period);
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_CP_MAX, GNRC_GOMACH_CP_DURATION_MAX_US);
|
|
gnrc_netif_set_rx_started(netif, false);
|
|
gnrc_gomach_set_pkt_received(netif, false);
|
|
netif->mac.prot.gomach.cp_extend_count = 0;
|
|
gnrc_gomach_set_quit_cycle(netif, false);
|
|
gnrc_gomach_set_unintd_preamble(netif, false);
|
|
gnrc_gomach_set_beacon_fail(netif, false);
|
|
gnrc_gomach_set_cp_end(netif, false);
|
|
gnrc_gomach_set_got_preamble(netif, false);
|
|
|
|
/* Flush RX queue and turn on radio. */
|
|
gnrc_priority_pktqueue_flush(&netif->mac.rx.queue);
|
|
gnrc_gomach_set_netdev_state(netif, NETOPT_STATE_IDLE);
|
|
|
|
/* Turn to current public channel. */
|
|
gnrc_gomach_turn_channel(netif, netif->mac.prot.gomach.cur_pub_channel);
|
|
|
|
/* Enable Auto-ACK for data packet reception. */
|
|
gnrc_gomach_set_autoack(netif, NETOPT_ENABLE);
|
|
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_CP_LISTEN;
|
|
gnrc_gomach_set_update(netif, false);
|
|
}
|
|
|
|
static void _cp_listen_get_pkt(gnrc_netif_t *netif)
|
|
{
|
|
gnrc_gomach_cp_packet_process(netif);
|
|
|
|
/* If the device has replied a preamble-ACK, it must waits for the data.
|
|
* Here, we extend the CP. */
|
|
if (gnrc_gomach_get_got_preamble(netif)) {
|
|
gnrc_gomach_set_got_preamble(netif, false);
|
|
gnrc_gomach_set_cp_end(netif, false);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_CP_END);
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_CP_END,
|
|
CONFIG_GNRC_GOMACH_CP_DURATION_US);
|
|
}
|
|
else if ((!gnrc_gomach_get_unintd_preamble(netif)) &&
|
|
(!gnrc_gomach_get_quit_cycle(netif))) {
|
|
gnrc_gomach_set_got_preamble(netif, false);
|
|
gnrc_gomach_set_cp_end(netif, false);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_CP_END);
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_CP_END,
|
|
CONFIG_GNRC_GOMACH_CP_DURATION_US);
|
|
}
|
|
}
|
|
|
|
static void _cp_listen_end(gnrc_netif_t *netif)
|
|
{
|
|
/* If we found ongoing reception, wait for reception complete. */
|
|
if ((gnrc_gomach_get_netdev_state(netif) == NETOPT_STATE_RX) &&
|
|
(netif->mac.prot.gomach.cp_extend_count < CONFIG_GNRC_GOMACH_CP_EXTEND_THRESHOLD)) {
|
|
netif->mac.prot.gomach.cp_extend_count++;
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_RX_END);
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_RX_END,
|
|
CONFIG_GNRC_GOMACH_WAIT_RX_END_US);
|
|
}
|
|
else {
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_RX_END);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_CP_END);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_CP_MAX);
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_CP_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
}
|
|
|
|
static void gomach_listen_cp_listen(gnrc_netif_t *netif)
|
|
{
|
|
if (gnrc_gomach_get_pkt_received(netif)) {
|
|
gnrc_gomach_set_pkt_received(netif, false);
|
|
_cp_listen_get_pkt(netif);
|
|
}
|
|
|
|
/* If we have reached the maximum CP duration, quit CP. */
|
|
if (gnrc_gomach_timeout_is_expired(netif, GNRC_GOMACH_TIMEOUT_CP_MAX)) {
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_RX_END);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_CP_END);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_CP_MAX);
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_CP_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
return;
|
|
}
|
|
|
|
if ((gnrc_gomach_timeout_is_expired(netif, GNRC_GOMACH_TIMEOUT_CP_END))) {
|
|
gnrc_gomach_set_cp_end(netif, true);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_CP_END);
|
|
}
|
|
|
|
/* If CP duration timeouted or we must quit CP, go to CP end. */
|
|
if (gnrc_gomach_get_cp_end(netif) || gnrc_gomach_get_quit_cycle(netif)) {
|
|
_cp_listen_end(netif);
|
|
}
|
|
}
|
|
|
|
static void gomach_listen_cp_end(gnrc_netif_t *netif)
|
|
{
|
|
gnrc_priority_pktqueue_flush(&netif->mac.rx.queue);
|
|
gnrc_mac_dispatch(&netif->mac.rx);
|
|
|
|
/* If we need to quit communications in this cycle, go to sleep. */
|
|
if (gnrc_gomach_get_quit_cycle(netif)) {
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_SLEEP_INIT;
|
|
}
|
|
else {
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_SEND_BEACON;
|
|
}
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
|
|
static void gomach_listen_send_beacon(gnrc_netif_t *netif)
|
|
{
|
|
/* First check if there are slots needed to be allocated. */
|
|
uint8_t slot_num = 0;
|
|
|
|
for (uint8_t i = 0; i < GNRC_GOMACH_SLOSCH_UNIT_COUNT; i++) {
|
|
if (netif->mac.rx.slosch_list[i].queue_indicator > 0) {
|
|
slot_num += netif->mac.rx.slosch_list[i].queue_indicator;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (slot_num > 0) {
|
|
/* Disable auto-ACK. Thus not to receive packet (attempt to reply ACK) anymore. */
|
|
gnrc_gomach_set_autoack(netif, NETOPT_DISABLE);
|
|
|
|
/* Assemble and send the beacon. */
|
|
int res = gnrc_gomach_send_beacon(netif);
|
|
if (res < 0) {
|
|
LOG_ERROR("ERROR: [GOMACH] send beacon error: %d.\n", res);
|
|
gnrc_gomach_set_beacon_fail(netif, true);
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
else {
|
|
gnrc_gomach_set_update(netif, false);
|
|
}
|
|
}
|
|
else {
|
|
/* No need to send beacon, go to next state. */
|
|
gnrc_gomach_set_beacon_fail(netif, true);
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_WAIT_BEACON_TX;
|
|
}
|
|
|
|
static void _no_vtdma_after_cp(gnrc_netif_t *netif)
|
|
{
|
|
/* If the device hasn't allocated transmission slots, check whether it has packets
|
|
* to transmit to neighbor. */
|
|
if (gnrc_gomach_find_next_tx_neighbor(netif)) {
|
|
/* Now, we have packet to send. */
|
|
|
|
if (netif->mac.tx.current_neighbor == &netif->mac.tx.neighbors[0]) {
|
|
/* The packet is for broadcasting. */
|
|
|
|
/* If we didn't find ongoing preamble stream, go to send broadcast packet. */
|
|
if (!gnrc_gomach_get_unintd_preamble(netif)) {
|
|
netif->mac.prot.gomach.basic_state = GNRC_GOMACH_TRANSMIT;
|
|
netif->mac.tx.transmit_state = GNRC_GOMACH_BROADCAST;
|
|
}
|
|
else {
|
|
/* If we find ongoing preamble stream, go to sleep. */
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_SLEEP_INIT;
|
|
}
|
|
}
|
|
else {
|
|
/* The packet waiting to be sent is for unicast. */
|
|
switch (netif->mac.tx.current_neighbor->mac_type) {
|
|
case GNRC_GOMACH_TYPE_UNKNOWN: {
|
|
/* The neighbor's phase is unknown yet, try to run t2u (transmission
|
|
* to unknown device) procedure to phase-lock the neighbor. */
|
|
|
|
/* If we didn't find ongoing preamble stream, go to t2u procedure. */
|
|
if (!gnrc_gomach_get_unintd_preamble(netif)) {
|
|
netif->mac.prot.gomach.basic_state = GNRC_GOMACH_TRANSMIT;
|
|
netif->mac.tx.transmit_state = GNRC_GOMACH_TRANS_TO_UNKNOWN;
|
|
}
|
|
else {
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_SLEEP_INIT;
|
|
}
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_TYPE_KNOWN: {
|
|
/* If the neighbor's phase is known, go to t2k (transmission
|
|
* to known device) procedure. Here, we don't worry that the t2k
|
|
* unicast transmission will interrupt with possible ongoing
|
|
* preamble transmissions of other devices. */
|
|
netif->mac.prot.gomach.basic_state = GNRC_GOMACH_TRANSMIT;
|
|
netif->mac.tx.transmit_state = GNRC_GOMACH_TRANS_TO_KNOWN;
|
|
break;
|
|
}
|
|
default: {
|
|
LOG_ERROR("ERROR: [GOMACH] vTDMA: unknown MAC type of "
|
|
"the neighbor.\n");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
/* No packet to send, go to sleep. */
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_SLEEP_INIT;
|
|
}
|
|
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
|
|
static void gomach_listen_wait_beacon_tx(gnrc_netif_t *netif)
|
|
{
|
|
if ((gnrc_gomach_timeout_is_expired(netif, GNRC_GOMACH_TIMEOUT_NO_TX_ISR))) {
|
|
/* No TX-ISR, go to sleep. */
|
|
LOG_DEBUG("[GOMACH]: no TX-finish ISR.");
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_SLEEP_INIT;
|
|
gnrc_gomach_set_update(netif, true);
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_NO_TX_ISR);
|
|
return;
|
|
}
|
|
|
|
if (gnrc_gomach_get_tx_finish(netif) ||
|
|
gnrc_gomach_get_beacon_fail(netif)) {
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_NO_TX_ISR);
|
|
|
|
if ((netif->mac.rx.vtdma_manag.total_slots_num > 0) &&
|
|
(!gnrc_gomach_get_beacon_fail(netif))) {
|
|
/* If the device has allocated transmission slots to other nodes,
|
|
* switch to vTDMA period to receive packets. */
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_VTDMA_INIT;
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
else {
|
|
_no_vtdma_after_cp(netif);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void gomach_vtdma_init(gnrc_netif_t *netif)
|
|
{
|
|
/* Switch the radio to the device's sub-channel. */
|
|
gnrc_gomach_turn_channel(netif, netif->mac.prot.gomach.sub_channel_seq);
|
|
|
|
/* Enable Auto ACK again for data reception */
|
|
gnrc_gomach_set_autoack(netif, NETOPT_ENABLE);
|
|
|
|
/* Set the vTDMA period timeout. */
|
|
uint32_t vtdma_duration = netif->mac.rx.vtdma_manag.total_slots_num *
|
|
CONFIG_GNRC_GOMACH_VTDMA_SLOT_SIZE_US;
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_VTDMA, vtdma_duration);
|
|
|
|
gnrc_gomach_set_vTDMA_end(netif, false);
|
|
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_VTDMA;
|
|
gnrc_gomach_set_update(netif, false);
|
|
}
|
|
|
|
static void gomach_vtdma(gnrc_netif_t *netif)
|
|
{
|
|
/* Process received packet here. */
|
|
if (gnrc_gomach_get_pkt_received(netif)) {
|
|
gnrc_gomach_set_pkt_received(netif, false);
|
|
gnrc_gomach_packet_process_in_vtdma(netif);
|
|
}
|
|
|
|
if (gnrc_gomach_timeout_is_expired(netif, GNRC_GOMACH_TIMEOUT_VTDMA)) {
|
|
gnrc_gomach_set_vTDMA_end(netif, true);
|
|
}
|
|
|
|
/* Go to vTDMA end after vTDMA timeout expires. */
|
|
if (gnrc_gomach_get_vTDMA_end(netif)) {
|
|
/* Wait for reception complete if found ongoing transmission. */
|
|
if (gnrc_gomach_get_netdev_state(netif) == NETOPT_STATE_RX) {
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_RX_END);
|
|
gnrc_gomach_set_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_RX_END,
|
|
CONFIG_GNRC_GOMACH_WAIT_RX_END_US);
|
|
return;
|
|
}
|
|
|
|
gnrc_gomach_clear_timeout(netif, GNRC_GOMACH_TIMEOUT_WAIT_RX_END);
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_VTDMA_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
}
|
|
|
|
static void gomach_vtdma_end(gnrc_netif_t *netif)
|
|
{
|
|
gnrc_priority_pktqueue_flush(&netif->mac.rx.queue);
|
|
gnrc_mac_dispatch(&netif->mac.rx);
|
|
|
|
/* Switch the radio to the public-channel. */
|
|
gnrc_gomach_turn_channel(netif, netif->mac.prot.gomach.cur_pub_channel);
|
|
|
|
/* Check if there is packet to send. */
|
|
if (gnrc_gomach_find_next_tx_neighbor(netif)) {
|
|
if (netif->mac.tx.current_neighbor == &netif->mac.tx.neighbors[0]) {
|
|
/* The packet is for broadcasting. */
|
|
if (!gnrc_gomach_get_unintd_preamble(netif)) {
|
|
netif->mac.prot.gomach.basic_state = GNRC_GOMACH_TRANSMIT;
|
|
netif->mac.tx.transmit_state = GNRC_GOMACH_BROADCAST;
|
|
}
|
|
else {
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_SLEEP_INIT;
|
|
}
|
|
}
|
|
else {
|
|
switch (netif->mac.tx.current_neighbor->mac_type) {
|
|
/* The packet waiting to be sent is for unicast. */
|
|
case GNRC_GOMACH_TYPE_UNKNOWN: {
|
|
/* The neighbor's phase is unknown yet, try to run t2u (transmission
|
|
* to unknown device) procedure to phase-lock the neighbor. */
|
|
if (!gnrc_gomach_get_unintd_preamble(netif)) {
|
|
netif->mac.prot.gomach.basic_state = GNRC_GOMACH_TRANSMIT;
|
|
netif->mac.tx.transmit_state = GNRC_GOMACH_TRANS_TO_UNKNOWN;
|
|
}
|
|
else {
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_SLEEP_INIT;
|
|
}
|
|
} break;
|
|
case GNRC_GOMACH_TYPE_KNOWN: {
|
|
/* If the neighbor's phase is known, go to t2k (transmission
|
|
* to known device) procedure. Here, we don't worry that the t2k
|
|
* unicast transmission will interrupt with possible ongoing
|
|
* preamble transmissions of other devices. */
|
|
netif->mac.prot.gomach.basic_state = GNRC_GOMACH_TRANSMIT;
|
|
netif->mac.tx.transmit_state = GNRC_GOMACH_TRANS_TO_KNOWN;
|
|
} break;
|
|
default: {
|
|
LOG_ERROR("ERROR: [GOMACH] vTDMA: unknown MAC type of the neighbor.\n");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
/* No packet to send, go to sleep. */
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_SLEEP_INIT;
|
|
}
|
|
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
|
|
static void gomach_sleep_init(gnrc_netif_t *netif)
|
|
{
|
|
/* Turn off the radio during sleep period to conserve power. */
|
|
gnrc_gomach_set_netdev_state(netif, NETOPT_STATE_SLEEP);
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_SLEEP;
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
|
|
static void gomach_sleep(gnrc_netif_t *netif)
|
|
{
|
|
/* If we are entering a new cycle, quit sleeping. */
|
|
if (gnrc_gomach_get_enter_new_cycle(netif)) {
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_SLEEP_END;
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
}
|
|
|
|
static void gomach_sleep_end(gnrc_netif_t *netif)
|
|
{
|
|
/* Run phase-backoff if needed, select a new wake-up phase. */
|
|
if (gnrc_gomach_get_phase_backoff(netif)) {
|
|
gnrc_gomach_set_phase_backoff(netif, false);
|
|
_gomach_phase_backoff(netif);
|
|
}
|
|
|
|
/* Go to CP (start of the new cycle), start listening on the public-channel. */
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_CP_INIT;
|
|
gnrc_gomach_set_update(netif, true);
|
|
}
|
|
|
|
static void gomach_update(gnrc_netif_t *netif)
|
|
{
|
|
switch (netif->mac.prot.gomach.basic_state) {
|
|
case GNRC_GOMACH_INIT: {
|
|
/* State machine of GoMacH's initialization procedure. */
|
|
switch (netif->mac.prot.gomach.init_state) {
|
|
case GNRC_GOMACH_INIT_PREPARE: {
|
|
gomach_init_prepare(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_INIT_ANNC_SUBCHAN: {
|
|
gomach_init_announce_subchannel(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_INIT_WAIT_FEEDBACK: {
|
|
gomach_init_wait_announce_feedback(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_INIT_END: {
|
|
gomach_init_end(netif);
|
|
break;
|
|
}
|
|
default: break;
|
|
}
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_LISTEN: {
|
|
/* State machine of GoMacH's duty-cycled listen procedure. */
|
|
switch (netif->mac.rx.listen_state) {
|
|
case GNRC_GOMACH_LISTEN_CP_INIT: {
|
|
gomach_listen_init(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_LISTEN_CP_LISTEN: {
|
|
gomach_listen_cp_listen(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_LISTEN_CP_END: {
|
|
gomach_listen_cp_end(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_LISTEN_SEND_BEACON: {
|
|
gomach_listen_send_beacon(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_LISTEN_WAIT_BEACON_TX: {
|
|
gomach_listen_wait_beacon_tx(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_LISTEN_VTDMA_INIT: {
|
|
gomach_vtdma_init(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_LISTEN_VTDMA: {
|
|
gomach_vtdma(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_LISTEN_VTDMA_END: {
|
|
gomach_vtdma_end(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_LISTEN_SLEEP_INIT: {
|
|
gomach_sleep_init(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_LISTEN_SLEEP: {
|
|
gomach_sleep(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_LISTEN_SLEEP_END: {
|
|
gomach_sleep_end(netif);
|
|
break;
|
|
}
|
|
default: break;
|
|
}
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_TRANSMIT: {
|
|
/* State machine of GoMacH's basic transmission scheme. */
|
|
switch (netif->mac.tx.transmit_state) {
|
|
case GNRC_GOMACH_TRANS_TO_UNKNOWN: {
|
|
gomach_t2u_update(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_TRANS_TO_KNOWN: {
|
|
gomach_t2k_update(netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_BROADCAST: {
|
|
gomach_bcast_update(netif);
|
|
break;
|
|
}
|
|
default: break;
|
|
}
|
|
break;
|
|
}
|
|
default: break;
|
|
}
|
|
}
|
|
|
|
static int _send(gnrc_netif_t *netif, gnrc_pktsnip_t *pkt)
|
|
{
|
|
if (!gnrc_mac_queue_tx_packet(&netif->mac.tx, 0, pkt)) {
|
|
/* TX packet queue full, release the packet. */
|
|
DEBUG("[GOMACH] TX queue full, drop packet.\n");
|
|
gnrc_pktbuf_release(pkt);
|
|
}
|
|
gnrc_gomach_set_update(netif, true);
|
|
|
|
while (gnrc_gomach_get_update(netif)) {
|
|
gnrc_gomach_set_update(netif, false);
|
|
gomach_update(netif);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void _gomach_msg_handler(gnrc_netif_t *netif, msg_t *msg)
|
|
{
|
|
switch (msg->type) {
|
|
case GNRC_GOMACH_EVENT_RTT_TYPE: {
|
|
_gomach_rtt_handler(msg->content.value, netif);
|
|
break;
|
|
}
|
|
case GNRC_GOMACH_EVENT_TIMEOUT_TYPE: {
|
|
/* GoMacH timeout expires. */
|
|
gnrc_gomach_timeout_make_expire((gnrc_gomach_timeout_t *) msg->content.ptr);
|
|
gnrc_gomach_set_update(netif, true);
|
|
break;
|
|
}
|
|
#if (GNRC_MAC_ENABLE_DUTYCYCLE_RECORD == 1)
|
|
case GNRC_MAC_TYPE_GET_DUTYCYCLE: {
|
|
/* Output GoMacH's current radio duty-cycle. */
|
|
uint64_t duty;
|
|
duty = xtimer_now_usec64();
|
|
duty = (netif->mac.prot.gomach.awake_duration_sum_ms) * 100 /
|
|
(duty - netif->mac.prot.gomach.system_start_time_ms);
|
|
printf("[GoMacH]: achieved radio duty-cycle: %u %% \n", (unsigned)duty);
|
|
break;
|
|
}
|
|
#endif
|
|
default: {
|
|
DEBUG("[GoMacH]: Unknown command %" PRIu16 "\n", msg->type);
|
|
break;
|
|
}
|
|
}
|
|
|
|
while (gnrc_gomach_get_update(netif)) {
|
|
gnrc_gomach_set_update(netif, false);
|
|
gomach_update(netif);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief Function called by the device driver on device events
|
|
*
|
|
* @param[in] event type of event
|
|
*/
|
|
static void _gomach_event_cb(netdev_t *dev, netdev_event_t event)
|
|
{
|
|
gnrc_netif_t *netif = (gnrc_netif_t *) dev->context;
|
|
|
|
if (event == NETDEV_EVENT_ISR) {
|
|
event_post(&netif->evq[GNRC_NETIF_EVQ_INDEX_PRIO_LOW], &netif->event_isr);
|
|
}
|
|
else {
|
|
DEBUG("gnrc_netdev: event triggered -> %i\n", event);
|
|
switch (event) {
|
|
case NETDEV_EVENT_RX_STARTED: {
|
|
gnrc_netif_set_rx_started(netif, true);
|
|
gnrc_gomach_set_update(netif, true);
|
|
break;
|
|
}
|
|
case NETDEV_EVENT_RX_COMPLETE: {
|
|
gnrc_gomach_set_update(netif, true);
|
|
|
|
gnrc_pktsnip_t *pkt = netif->ops->recv(netif);
|
|
if (pkt == NULL) {
|
|
gnrc_gomach_set_buffer_full(netif, true);
|
|
|
|
LOG_DEBUG("[GOMACH] gnrc_netdev: packet is NULL, memory full?\n");
|
|
gnrc_gomach_set_pkt_received(netif, false);
|
|
gnrc_netif_set_rx_started(netif, false);
|
|
break;
|
|
}
|
|
|
|
if (!gnrc_netif_get_rx_started(netif)) {
|
|
LOG_DEBUG("[GOMACH] gnrc_netdev: maybe sending kicked in "
|
|
"and frame buffer is now corrupted?\n");
|
|
gnrc_pktbuf_release(pkt);
|
|
gnrc_netif_set_rx_started(netif, false);
|
|
break;
|
|
}
|
|
|
|
gnrc_netif_set_rx_started(netif, false);
|
|
|
|
if (!gnrc_mac_queue_rx_packet(&netif->mac.rx, 0, pkt)) {
|
|
LOG_ERROR("ERROR: [GOMACH] gnrc_netdev: can't push RX packet, queue full?\n");
|
|
gnrc_pktbuf_release(pkt);
|
|
gnrc_gomach_set_pkt_received(netif, false);
|
|
break;
|
|
}
|
|
else {
|
|
gnrc_gomach_set_pkt_received(netif, true);
|
|
}
|
|
break;
|
|
}
|
|
case NETDEV_EVENT_TX_COMPLETE: {
|
|
gnrc_netif_set_tx_feedback(netif, TX_FEEDBACK_SUCCESS);
|
|
gnrc_gomach_set_tx_finish(netif, true);
|
|
gnrc_gomach_set_netdev_state(netif, NETOPT_STATE_IDLE);
|
|
gnrc_gomach_set_update(netif, true);
|
|
break;
|
|
}
|
|
case NETDEV_EVENT_TX_NOACK: {
|
|
gnrc_netif_set_tx_feedback(netif, TX_FEEDBACK_NOACK);
|
|
gnrc_gomach_set_tx_finish(netif, true);
|
|
gnrc_gomach_set_netdev_state(netif, NETOPT_STATE_IDLE);
|
|
gnrc_gomach_set_update(netif, true);
|
|
break;
|
|
}
|
|
case NETDEV_EVENT_TX_MEDIUM_BUSY: {
|
|
gnrc_netif_set_tx_feedback(netif, TX_FEEDBACK_BUSY);
|
|
gnrc_gomach_set_tx_finish(netif, true);
|
|
gnrc_gomach_set_netdev_state(netif, NETOPT_STATE_IDLE);
|
|
gnrc_gomach_set_update(netif, true);
|
|
break;
|
|
}
|
|
default: {
|
|
DEBUG("WARNING [GoMacH]: unhandled event %u.\n", event);
|
|
}
|
|
}
|
|
}
|
|
|
|
while (gnrc_gomach_get_update(netif)) {
|
|
gnrc_gomach_set_update(netif, false);
|
|
gomach_update(netif);
|
|
}
|
|
}
|
|
|
|
static int _gomach_init(gnrc_netif_t *netif)
|
|
{
|
|
netdev_t *dev;
|
|
|
|
int res = gnrc_netif_default_init(netif);
|
|
if (res < 0) {
|
|
return res;
|
|
}
|
|
|
|
dev = netif->dev;
|
|
dev->event_callback = _gomach_event_cb;
|
|
|
|
/* Initialize RTT. */
|
|
rtt_init();
|
|
|
|
/* Store pid globally, so that IRQ can use it to send message. */
|
|
gomach_pid = netif->pid;
|
|
|
|
/* Set MAC address length. */
|
|
uint16_t src_len = IEEE802154_LONG_ADDRESS_LEN;
|
|
dev->driver->set(dev, NETOPT_SRC_LEN, &src_len, sizeof(src_len));
|
|
|
|
/* Get the MAC address of the device. */
|
|
netif->l2addr_len = dev->driver->get(dev,
|
|
NETOPT_ADDRESS_LONG,
|
|
netif->l2addr,
|
|
sizeof(netif->l2addr));
|
|
|
|
/* Initialize GoMacH's state machines. */
|
|
netif->mac.prot.gomach.basic_state = GNRC_GOMACH_INIT;
|
|
netif->mac.prot.gomach.init_state = GNRC_GOMACH_INIT_PREPARE;
|
|
netif->mac.rx.listen_state = GNRC_GOMACH_LISTEN_CP_INIT;
|
|
netif->mac.tx.transmit_state = GNRC_GOMACH_TRANS_TO_UNKNOWN;
|
|
netif->mac.tx.bcast_state = GNRC_GOMACH_BCAST_INIT;
|
|
netif->mac.tx.t2k_state = GNRC_GOMACH_T2K_INIT;
|
|
netif->mac.tx.t2u_state = GNRC_GOMACH_T2U_INIT;
|
|
|
|
/* Initialize GoMacH's channels. */
|
|
netif->mac.prot.gomach.sub_channel_seq = 13;
|
|
netif->mac.prot.gomach.pub_channel_1 = 26;
|
|
netif->mac.prot.gomach.pub_channel_2 = 11;
|
|
netif->mac.prot.gomach.cur_pub_channel = netif->mac.prot.gomach.pub_channel_1;
|
|
gnrc_gomach_turn_channel(netif, netif->mac.prot.gomach.cur_pub_channel);
|
|
|
|
/* Enable RX-start and TX-started and TX-END interrupts. */
|
|
netopt_enable_t enable = NETOPT_ENABLE;
|
|
dev->driver->set(dev, NETOPT_RX_START_IRQ, &enable, sizeof(enable));
|
|
dev->driver->set(dev, NETOPT_TX_END_IRQ, &enable, sizeof(enable));
|
|
|
|
/* Initialize broadcast sequence number. This at least differs from board
|
|
* to board. */
|
|
netif->mac.tx.broadcast_seq = netif->l2addr[netif->l2addr_len - 1];
|
|
|
|
/* Reset all timeouts just to be sure. */
|
|
gnrc_gomach_reset_timeouts(netif);
|
|
|
|
/* Initialize GoMacH's other key parameters. */
|
|
netif->mac.tx.no_ack_counter = 0;
|
|
gnrc_gomach_set_enter_new_cycle(netif, false);
|
|
netif->mac.rx.vtdma_manag.sub_channel_seq = 26;
|
|
netif->mac.prot.gomach.subchannel_occu_flags = 0;
|
|
gnrc_gomach_set_pkt_received(netif, false);
|
|
gnrc_gomach_set_update(netif, false);
|
|
gnrc_gomach_set_duty_cycle_start(netif, false);
|
|
gnrc_gomach_set_quit_cycle(netif, false);
|
|
|
|
gnrc_gomach_set_beacon_fail(netif, false);
|
|
gnrc_gomach_set_buffer_full(netif, false);
|
|
gnrc_gomach_set_phase_backoff(netif, false);
|
|
netif->mac.rx.check_dup_pkt.queue_head = 0;
|
|
netif->mac.tx.last_tx_neighbor_id = 0;
|
|
|
|
netdev_t *netdev = netif->dev;
|
|
netdev_ieee802154_t *device_state = container_of(netdev, netdev_ieee802154_t, netdev);
|
|
device_state->seq = netif->l2addr[netif->l2addr_len - 1];
|
|
|
|
/* Initialize GoMacH's duplicate-check scheme. */
|
|
for (uint8_t i = 0; i < GNRC_GOMACH_DUPCHK_BUFFER_SIZE; i++) {
|
|
netif->mac.rx.check_dup_pkt.last_nodes[i].node_addr.len = 0;
|
|
}
|
|
|
|
/* Set the random seed. */
|
|
uint32_t seed = 0;
|
|
seed = netif->l2addr[netif->l2addr_len - 2];
|
|
seed = seed << 8;
|
|
seed |= netif->l2addr[netif->l2addr_len - 1];
|
|
random_init(seed);
|
|
|
|
netif->mac.tx.t2u_fail_count = 0;
|
|
|
|
#if (GNRC_MAC_ENABLE_DUTYCYCLE_RECORD == 1)
|
|
/* Start duty cycle recording */
|
|
netif->mac.prot.gomach.system_start_time_ms = xtimer_now_usec64();
|
|
netif->mac.prot.gomach.last_radio_on_time_ms =
|
|
netif->mac.prot.gomach.system_start_time_ms;
|
|
netif->mac.prot.gomach.awake_duration_sum_ms = 0;
|
|
netif->mac.prot.gomach.gomach_info |= GNRC_GOMACH_INTERNAL_INFO_RADIO_IS_ON;
|
|
#endif
|
|
|
|
gnrc_gomach_set_update(netif, true);
|
|
|
|
while (gnrc_gomach_get_update(netif)) {
|
|
gnrc_gomach_set_update(netif, false);
|
|
gomach_update(netif);
|
|
}
|
|
|
|
return 0;
|
|
}
|