/* * Copyright (C) 2020 HAW Hamburg * * 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. */ /** * @{ * * @file * @author José I. Alamos */ #include "net/netdev/ieee802154_submac.h" #include "event/thread.h" static const ieee802154_submac_cb_t _cb; static const netdev_driver_t netdev_submac_driver; static void _ack_timeout(void *arg) { netdev_ieee802154_submac_t *netdev_submac = arg; netdev_t *netdev = arg; netdev_submac->isr_flags |= NETDEV_SUBMAC_FLAGS_ACK_TIMEOUT; netdev->event_callback(netdev, NETDEV_EVENT_ISR); } static netopt_state_t _get_submac_state(ieee802154_submac_t *submac) { if (ieee802154_submac_state_is_idle(submac)) { return NETOPT_STATE_SLEEP; } else { return NETOPT_STATE_IDLE; } } static int _get(netdev_t *netdev, netopt_t opt, void *value, size_t max_len) { netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev, netdev_ieee802154_t, netdev); netdev_ieee802154_submac_t *netdev_submac = container_of(netdev_ieee802154, netdev_ieee802154_submac_t, dev); ieee802154_submac_t *submac = &netdev_submac->submac; switch (opt) { case NETOPT_STATE: *((netopt_state_t*) value) = _get_submac_state(submac); return 0; case NETOPT_TX_RETRIES_NEEDED: if (netdev_submac->retrans < 0) { return -ENOTSUP; } *((uint8_t*) value) = netdev_submac->retrans; return 1; case NETOPT_IEEE802154_PHY: *((uint8_t*) value) = ieee802154_get_phy_mode(submac); return 1; default: break; } return netdev_ieee802154_get(container_of(netdev, netdev_ieee802154_t, netdev), opt, value, max_len); } static int _set_submac_state(ieee802154_submac_t *submac, netopt_state_t state) { switch (state) { case NETOPT_STATE_SLEEP: return ieee802154_set_idle(submac); break; case NETOPT_STATE_IDLE: return ieee802154_set_rx(submac); break; default: return -ENOTSUP; } } static int _set(netdev_t *netdev, netopt_t opt, const void *value, size_t value_len) { netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev, netdev_ieee802154_t, netdev); netdev_ieee802154_submac_t *netdev_submac = container_of(netdev_ieee802154, netdev_ieee802154_submac_t, dev); ieee802154_submac_t *submac = &netdev_submac->submac; int res; int16_t tx_power; switch (opt) { case NETOPT_ADDRESS: ieee802154_set_short_addr(submac, value); break; case NETOPT_ADDRESS_LONG: ieee802154_set_ext_addr(submac, value); break; case NETOPT_NID: ieee802154_set_panid(submac, value); break; case NETOPT_CHANNEL: ieee802154_set_channel_number(submac, *((uint16_t *)value)); break; case NETOPT_TX_POWER: tx_power = *((int16_t *)value); res = ieee802154_set_tx_power(submac, tx_power); if (res >= 0) { netdev_submac->dev.txpower = tx_power; } return res; case NETOPT_STATE: return _set_submac_state(submac, *((netopt_state_t*) value)); default: break; } return netdev_ieee802154_set(container_of(netdev, netdev_ieee802154_t, netdev), opt, value, value_len); } void ieee802154_submac_bh_request(ieee802154_submac_t *submac) { netdev_ieee802154_submac_t *netdev_submac = container_of(submac, netdev_ieee802154_submac_t, submac); netdev_t *netdev = &netdev_submac->dev.netdev; netdev_submac->isr_flags |= NETDEV_SUBMAC_FLAGS_BH_REQUEST; netdev->event_callback(netdev, NETDEV_EVENT_ISR); } void ieee802154_submac_ack_timer_set(ieee802154_submac_t *submac, uint16_t us) { netdev_ieee802154_submac_t *netdev_submac = container_of(submac, netdev_ieee802154_submac_t, submac); xtimer_set(&netdev_submac->ack_timer, us); } void ieee802154_submac_ack_timer_cancel(ieee802154_submac_t *submac) { netdev_ieee802154_submac_t *netdev_submac = container_of(submac, netdev_ieee802154_submac_t, submac); xtimer_remove(&netdev_submac->ack_timer); /* Prevent a race condition between the RX_DONE event and the ACK timeout */ netdev_submac->isr_flags &= ~NETDEV_SUBMAC_FLAGS_ACK_TIMEOUT; } static int _send(netdev_t *netdev, const iolist_t *pkt) { netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev, netdev_ieee802154_t, netdev); netdev_ieee802154_submac_t *netdev_submac = container_of(netdev_ieee802154, netdev_ieee802154_submac_t, dev); ieee802154_submac_t *submac = &netdev_submac->submac; int res = ieee802154_send(submac, pkt); if (res >= 0) { /* HACK: Used to mark a transmission when called * inside the TX Done callback */ netdev_submac->ev = NETDEV_EVENT_TX_STARTED; } return res; } static void _isr(netdev_t *netdev) { netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev, netdev_ieee802154_t, netdev); netdev_ieee802154_submac_t *netdev_submac = container_of(netdev_ieee802154, netdev_ieee802154_submac_t, dev); ieee802154_submac_t *submac = &netdev_submac->submac; bool can_dispatch = true; do { irq_disable(); int flags = netdev_submac->isr_flags; netdev_submac->isr_flags = 0; irq_enable(); if (flags & NETDEV_SUBMAC_FLAGS_BH_REQUEST) { ieee802154_submac_bh_process(submac); } if (flags & NETDEV_SUBMAC_FLAGS_ACK_TIMEOUT) { ieee802154_submac_ack_timeout_fired(&netdev_submac->submac); } if (flags & NETDEV_SUBMAC_FLAGS_TX_DONE) { ieee802154_submac_tx_done_cb(&netdev_submac->submac); } if (flags & NETDEV_SUBMAC_FLAGS_RX_DONE) { ieee802154_submac_rx_done_cb(submac); } if (flags & NETDEV_SUBMAC_FLAGS_CRC_ERROR) { ieee802154_submac_crc_error_cb(submac); } if (flags) { can_dispatch = false; } } while (netdev_submac->isr_flags != 0); if (netdev_submac->dispatch) { /* The SubMAC will not generate further events after calling TX Done * or RX Done, but there might be pending ISR events that might not be * caught by the previous loop. * This should be safe to make sure that all events are cached */ if (!can_dispatch) { netdev->event_callback(netdev, NETDEV_EVENT_ISR); return; } netdev_submac->dispatch = false; /* TODO: Prevent race condition when state goes to PREPARE */ netdev->event_callback(netdev, netdev_submac->ev); /* HACK: the TX_STARTED event is used to indicate a frame was * sent during the event callback. * If no frame was sent go back to RX */ if (netdev_submac->ev != NETDEV_EVENT_TX_STARTED) { ieee802154_set_rx(submac); } } } static int _recv(netdev_t *netdev, void *buf, size_t len, void *info) { netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev, netdev_ieee802154_t, netdev); netdev_ieee802154_submac_t *netdev_submac = container_of(netdev_ieee802154, netdev_ieee802154_submac_t, dev); ieee802154_submac_t *submac = &netdev_submac->submac; ieee802154_rx_info_t rx_info; if (buf == NULL && len == 0) { return ieee802154_get_frame_length(submac); } int res = ieee802154_read_frame(submac, buf, len, &rx_info); if (info) { netdev_ieee802154_rx_info_t *netdev_rx_info = info; /* The Radio HAL uses the IEEE 802.15.4 definition for RSSI. * Netdev uses dBm. Therefore we need a translation here */ netdev_rx_info->rssi = ieee802154_rssi_to_dbm(rx_info.rssi); netdev_rx_info->lqi = rx_info.lqi; } return res; } static void submac_tx_done(ieee802154_submac_t *submac, int status, ieee802154_tx_info_t *info) { (void)status; netdev_ieee802154_submac_t *netdev_submac = container_of(submac, netdev_ieee802154_submac_t, submac); if (info) { netdev_submac->retrans = info->retrans; } netdev_submac->dispatch = true; switch (status) { case TX_STATUS_SUCCESS: netdev_submac->ev = NETDEV_EVENT_TX_COMPLETE; break; case TX_STATUS_FRAME_PENDING: netdev_submac->ev = NETDEV_EVENT_TX_COMPLETE_DATA_PENDING; break; case TX_STATUS_MEDIUM_BUSY: netdev_submac->ev = NETDEV_EVENT_TX_MEDIUM_BUSY; break; case TX_STATUS_NO_ACK: netdev_submac->ev = NETDEV_EVENT_TX_NOACK; break; default: break; } } static void submac_rx_done(ieee802154_submac_t *submac) { netdev_ieee802154_submac_t *netdev_submac = container_of(submac, netdev_ieee802154_submac_t, submac); netdev_submac->dispatch = true; netdev_submac->ev = NETDEV_EVENT_RX_COMPLETE; } static const ieee802154_submac_cb_t _cb = { .rx_done = submac_rx_done, .tx_done = submac_tx_done, }; /* Event Notification callback */ static void _hal_radio_cb(ieee802154_dev_t *dev, ieee802154_trx_ev_t status) { ieee802154_submac_t *submac = container_of(dev, ieee802154_submac_t, dev); netdev_ieee802154_submac_t *netdev_submac = container_of(submac, netdev_ieee802154_submac_t, submac); netdev_t *netdev = &netdev_submac->dev.netdev; switch (status) { case IEEE802154_RADIO_CONFIRM_TX_DONE: netdev_submac->isr_flags |= NETDEV_SUBMAC_FLAGS_TX_DONE; break; case IEEE802154_RADIO_INDICATION_RX_DONE: netdev_submac->isr_flags |= NETDEV_SUBMAC_FLAGS_RX_DONE; break; case IEEE802154_RADIO_INDICATION_CRC_ERROR: netdev_submac->isr_flags |= NETDEV_SUBMAC_FLAGS_CRC_ERROR; break; default: break; } netdev->event_callback(netdev, NETDEV_EVENT_ISR); } static int _init(netdev_t *netdev) { netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev, netdev_ieee802154_t, netdev); netdev_ieee802154_submac_t *netdev_submac = container_of(netdev_ieee802154, netdev_ieee802154_submac_t, dev); ieee802154_submac_t *submac = &netdev_submac->submac; netdev_ieee802154_setup(netdev_ieee802154); int res = ieee802154_submac_init(submac, (network_uint16_t*) netdev_ieee802154->short_addr, (eui64_t*) netdev_ieee802154->long_addr); if (res < 0) { return res; } /* This function already sets the PAN ID to the default one */ netdev_ieee802154_reset(netdev_ieee802154); uint16_t chan = submac->channel_num; int16_t tx_power = submac->tx_pow; netopt_enable_t enable = NETOPT_ENABLE; /* Initialise netdev_ieee802154_t struct */ netdev_ieee802154_set(netdev_ieee802154, NETOPT_CHANNEL, &chan, sizeof(chan)); netdev_ieee802154_set(netdev_ieee802154, NETOPT_ACK_REQ, &enable, sizeof(enable)); netdev_submac->dev.txpower = tx_power; return 0; } int netdev_ieee802154_submac_init(netdev_ieee802154_submac_t *netdev_submac) { netdev_t *netdev = &netdev_submac->dev.netdev; netdev->driver = &netdev_submac_driver; ieee802154_submac_t *submac = &netdev_submac->submac; submac->cb = &_cb; /* Set the Event Notification */ submac->dev.cb = _hal_radio_cb; netdev_submac->ack_timer.callback = _ack_timeout; netdev_submac->ack_timer.arg = netdev_submac; return 0; } static const netdev_driver_t netdev_submac_driver = { .get = _get, .set = _set, .send = _send, .recv = _recv, .isr = _isr, .init = _init, }; /** @} */