2015-04-23 18:36:55 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 Freie Universität Berlin
|
|
|
|
*
|
|
|
|
* 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 drivers_ng_at86rf2xx
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Netdev adaption for the AT86RF2xx drivers
|
|
|
|
*
|
|
|
|
* @author Thomas Eichinger <thomas.eichinger@fu-berlin.de>
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2015-06-03 17:08:03 +02:00
|
|
|
#include "net/eui64.h"
|
2015-04-23 18:36:55 +02:00
|
|
|
#include "net/ng_ieee802154.h"
|
|
|
|
#include "net/ng_netbase.h"
|
|
|
|
#include "ng_at86rf2xx.h"
|
|
|
|
#include "ng_at86rf2xx_netdev.h"
|
|
|
|
#include "ng_at86rf2xx_internal.h"
|
|
|
|
#include "ng_at86rf2xx_registers.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
/* TODO: generalize and move to ng_ieee802154 */
|
|
|
|
static size_t _make_data_frame_hdr(ng_at86rf2xx_t *dev, uint8_t *buf,
|
|
|
|
ng_netif_hdr_t *hdr)
|
|
|
|
{
|
|
|
|
int pos = 0;
|
|
|
|
|
|
|
|
/* we are building a data frame here */
|
|
|
|
buf[0] = NG_IEEE802154_FCF_TYPE_DATA;
|
|
|
|
buf[1] = 0x88; /* use short src and dst addresses as starting point */
|
|
|
|
|
|
|
|
/* if AUTOACK is enabled, then we also expect ACKs for this packet */
|
2015-06-02 21:26:02 +02:00
|
|
|
if (!(hdr->flags & NG_NETIF_HDR_FLAGS_BROADCAST) &&
|
|
|
|
!(hdr->flags & NG_NETIF_HDR_FLAGS_MULTICAST) &&
|
|
|
|
(dev->options & NG_AT86RF2XX_OPT_AUTOACK)) {
|
2015-04-23 18:36:55 +02:00
|
|
|
buf[0] |= NG_IEEE802154_FCF_ACK_REQ;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* fill in destination PAN ID */
|
|
|
|
pos = 3;
|
|
|
|
buf[pos++] = (uint8_t)((dev->pan) & 0xff);
|
|
|
|
buf[pos++] = (uint8_t)((dev->pan) >> 8);
|
|
|
|
|
|
|
|
/* fill in destination address */
|
|
|
|
if (hdr->flags &
|
|
|
|
(NG_NETIF_HDR_FLAGS_BROADCAST | NG_NETIF_HDR_FLAGS_MULTICAST)) {
|
|
|
|
buf[pos++] = 0xff;
|
|
|
|
buf[pos++] = 0xff;
|
|
|
|
}
|
|
|
|
else if (hdr->dst_l2addr_len == 2) {
|
|
|
|
uint8_t *dst_addr = ng_netif_hdr_get_dst_addr(hdr);
|
|
|
|
buf[pos++] = dst_addr[1];
|
|
|
|
buf[pos++] = dst_addr[0];
|
|
|
|
}
|
|
|
|
else if (hdr->dst_l2addr_len == 8) {
|
|
|
|
buf[1] |= 0x04;
|
|
|
|
uint8_t *dst_addr = ng_netif_hdr_get_dst_addr(hdr);
|
|
|
|
for (int i = 7; i >= 0; i--) {
|
|
|
|
buf[pos++] = dst_addr[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* unsupported address length */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* fill in source PAN ID (if applicable */
|
|
|
|
if (dev->options & NG_AT86RF2XX_OPT_USE_SRC_PAN) {
|
|
|
|
buf[pos++] = (uint8_t)((dev->pan) & 0xff);
|
|
|
|
buf[pos++] = (uint8_t)((dev->pan) >> 8);
|
|
|
|
} else {
|
|
|
|
buf[0] |= NG_IEEE802154_FCF_PAN_COMP;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* fill in source address */
|
|
|
|
if (dev->options & NG_AT86RF2XX_OPT_SRC_ADDR_LONG) {
|
|
|
|
buf[1] |= 0x40;
|
|
|
|
memcpy(&(buf[pos]), dev->addr_long, 8);
|
|
|
|
pos += 8;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
buf[pos++] = dev->addr_short[0];
|
|
|
|
buf[pos++] = dev->addr_short[1];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set sequence number */
|
|
|
|
buf[2] = dev->seq_nr++;
|
|
|
|
/* return actual header length */
|
|
|
|
return pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO: generalize and move to ng_ieee802154 */
|
|
|
|
/* TODO: include security header implications */
|
|
|
|
static size_t _get_frame_hdr_len(uint8_t *mhr)
|
|
|
|
{
|
|
|
|
uint8_t tmp;
|
|
|
|
size_t len = 3;
|
|
|
|
|
|
|
|
/* figure out address sizes */
|
|
|
|
tmp = (mhr[1] & NG_IEEE802154_FCF_DST_ADDR_MASK);
|
|
|
|
if (tmp == NG_IEEE802154_FCF_DST_ADDR_SHORT) {
|
|
|
|
len += 4;
|
|
|
|
}
|
|
|
|
else if (tmp == NG_IEEE802154_FCF_DST_ADDR_LONG) {
|
|
|
|
len += 10;
|
|
|
|
}
|
|
|
|
else if (tmp != NG_IEEE802154_FCF_DST_ADDR_VOID) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
tmp = (mhr[1] & NG_IEEE802154_FCF_SRC_ADDR_MASK);
|
|
|
|
if (tmp == NG_IEEE802154_FCF_SRC_ADDR_VOID) {
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (!(mhr[0] & NG_IEEE802154_FCF_PAN_COMP)) {
|
|
|
|
len += 2;
|
|
|
|
}
|
|
|
|
if (tmp == NG_IEEE802154_FCF_SRC_ADDR_SHORT) {
|
|
|
|
return (len + 2);
|
|
|
|
}
|
|
|
|
else if (tmp == NG_IEEE802154_FCF_SRC_ADDR_LONG) {
|
|
|
|
return (len + 8);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO: generalize and move to ng_ieee802154 */
|
|
|
|
static ng_pktsnip_t *_make_netif_hdr(uint8_t *mhr)
|
|
|
|
{
|
|
|
|
uint8_t tmp;
|
|
|
|
uint8_t *addr;
|
|
|
|
uint8_t src_len, dst_len;
|
|
|
|
ng_pktsnip_t *snip;
|
|
|
|
ng_netif_hdr_t *hdr;
|
|
|
|
|
|
|
|
/* figure out address sizes */
|
|
|
|
tmp = mhr[1] & NG_IEEE802154_FCF_SRC_ADDR_MASK;
|
|
|
|
if (tmp == NG_IEEE802154_FCF_SRC_ADDR_SHORT) {
|
|
|
|
src_len = 2;
|
|
|
|
}
|
|
|
|
else if (tmp == NG_IEEE802154_FCF_SRC_ADDR_LONG) {
|
|
|
|
src_len = 8;
|
|
|
|
}
|
|
|
|
else if (tmp == 0) {
|
|
|
|
src_len = 0;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
tmp = mhr[1] & NG_IEEE802154_FCF_DST_ADDR_MASK;
|
|
|
|
if (tmp == NG_IEEE802154_FCF_DST_ADDR_SHORT) {
|
|
|
|
dst_len = 2;
|
|
|
|
}
|
|
|
|
else if (tmp == NG_IEEE802154_FCF_DST_ADDR_LONG) {
|
|
|
|
dst_len = 8;
|
|
|
|
}
|
|
|
|
else if (tmp == 0) {
|
|
|
|
dst_len = 0;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
/* allocate space for header */
|
|
|
|
snip = ng_pktbuf_add(NULL, NULL, sizeof(ng_netif_hdr_t) + src_len + dst_len,
|
|
|
|
NG_NETTYPE_NETIF);
|
|
|
|
if (snip == NULL) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
/* fill header */
|
|
|
|
hdr = (ng_netif_hdr_t *)snip->data;
|
|
|
|
ng_netif_hdr_init(hdr, src_len, dst_len);
|
|
|
|
if (dst_len > 0) {
|
|
|
|
tmp = 5 + dst_len;
|
|
|
|
addr = ng_netif_hdr_get_dst_addr(hdr);
|
|
|
|
for (int i = 0; i < dst_len; i++) {
|
|
|
|
addr[i] = mhr[5 + (dst_len - i) - 1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
tmp = 3;
|
|
|
|
}
|
|
|
|
if (!(mhr[0] & NG_IEEE802154_FCF_PAN_COMP)) {
|
|
|
|
tmp += 2;
|
|
|
|
}
|
|
|
|
if (src_len > 0) {
|
|
|
|
addr = ng_netif_hdr_get_src_addr(hdr);
|
|
|
|
for (int i = 0; i < src_len; i++) {
|
|
|
|
addr[i] = mhr[tmp + (src_len - i) - 1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return snip;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int _send(ng_netdev_t *netdev, ng_pktsnip_t *pkt)
|
|
|
|
{
|
|
|
|
ng_at86rf2xx_t *dev = (ng_at86rf2xx_t *)netdev;
|
|
|
|
ng_pktsnip_t *snip;
|
|
|
|
uint8_t mhr[NG_IEEE802154_MAX_HDR_LEN];
|
|
|
|
size_t len;
|
|
|
|
|
|
|
|
if (pkt == NULL) {
|
|
|
|
return -ENOMSG;
|
|
|
|
}
|
|
|
|
if (dev == NULL) {
|
|
|
|
ng_pktbuf_release(pkt);
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* create 802.15.4 header */
|
|
|
|
len = _make_data_frame_hdr(dev, mhr, (ng_netif_hdr_t *)pkt->data);
|
|
|
|
if (len == 0) {
|
|
|
|
DEBUG("[ng_at86rf2xx] error: unable to create 802.15.4 header\n");
|
|
|
|
ng_pktbuf_release(pkt);
|
|
|
|
return -ENOMSG;
|
|
|
|
}
|
|
|
|
/* check if packet (header + payload + FCS) fits into FIFO */
|
|
|
|
snip = pkt->next;
|
|
|
|
if ((ng_pkt_len(snip) + len + 2) > NG_AT86RF2XX_MAX_PKT_LENGTH) {
|
|
|
|
DEBUG("[ng_at86rf2xx] error: packet too large to be send\n");
|
|
|
|
ng_pktbuf_release(pkt);
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
|
|
|
|
ng_at86rf2xx_tx_prepare(dev);
|
|
|
|
/* put header into FIFO */
|
|
|
|
len = ng_at86rf2xx_tx_load(dev, mhr, len, 0);
|
|
|
|
/* load packet data into FIFO */
|
|
|
|
while (snip) {
|
2015-05-10 14:02:42 +02:00
|
|
|
len = ng_at86rf2xx_tx_load(dev, snip->data, snip->size, len);
|
2015-04-23 18:36:55 +02:00
|
|
|
snip = snip->next;
|
|
|
|
}
|
|
|
|
/* send data out directly if pre-loading id disabled */
|
|
|
|
if (!(dev->options & NG_AT86RF2XX_OPT_PRELOADING)) {
|
|
|
|
ng_at86rf2xx_tx_exec(dev);
|
|
|
|
}
|
|
|
|
/* release packet */
|
|
|
|
ng_pktbuf_release(pkt);
|
|
|
|
/* return the number of bytes that were actually send out */
|
|
|
|
return (int)len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _receive_data(ng_at86rf2xx_t *dev)
|
|
|
|
{
|
|
|
|
uint8_t mhr[NG_IEEE802154_MAX_HDR_LEN];
|
|
|
|
size_t pkt_len, hdr_len;
|
|
|
|
ng_pktsnip_t *hdr, *payload = NULL;
|
|
|
|
ng_netif_hdr_t *netif;
|
|
|
|
|
|
|
|
/* get the size of the received packet (unlocks frame buffer protection) */
|
|
|
|
pkt_len = ng_at86rf2xx_rx_len(dev);
|
|
|
|
|
|
|
|
/* abort here already if no event callback is registered */
|
|
|
|
if (!dev->event_cb) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* in raw mode, just read the binary dump into the packet buffer */
|
|
|
|
if (dev->options & NG_AT86RF2XX_OPT_RAWDUMP) {
|
|
|
|
payload = ng_pktbuf_add(NULL, NULL, pkt_len, NG_NETTYPE_UNDEF);
|
|
|
|
if (payload == NULL ) {
|
|
|
|
DEBUG("[ng_at86rf2xx] error: unable to allocate RAW data\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ng_at86rf2xx_rx_read(dev, payload->data, pkt_len, 0);
|
|
|
|
dev->event_cb(NETDEV_EVENT_RX_COMPLETE, payload);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* get FCF field and compute 802.15.4 header length */
|
|
|
|
ng_at86rf2xx_rx_read(dev, mhr, 2, 0);
|
|
|
|
hdr_len = _get_frame_hdr_len(mhr);
|
|
|
|
if (hdr_len == 0) {
|
|
|
|
DEBUG("[ng_at86rf2xx] error: unable parse incoming frame header\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
/* read the rest of the header and parse the netif header from it */
|
|
|
|
ng_at86rf2xx_rx_read(dev, &(mhr[2]), hdr_len - 2, 2);
|
|
|
|
hdr = _make_netif_hdr(mhr);
|
|
|
|
if (hdr == NULL) {
|
|
|
|
DEBUG("[ng_at86rf2xx] error: unable to allocate netif header\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
/* fill missing fields in netif header */
|
|
|
|
netif = (ng_netif_hdr_t *)hdr->data;
|
|
|
|
netif->if_pid = dev->mac_pid;
|
|
|
|
ng_at86rf2xx_rx_read(dev, &(netif->lqi), 1, pkt_len);
|
|
|
|
netif->rssi = ng_at86rf2xx_reg_read(dev, NG_AT86RF2XX_REG__PHY_ED_LEVEL);
|
|
|
|
|
|
|
|
/* allocate payload */
|
|
|
|
payload = ng_pktbuf_add(hdr, NULL, (pkt_len - hdr_len), dev->proto);
|
|
|
|
if (payload == NULL) {
|
|
|
|
DEBUG("[ng_at86rf2xx] error: unable to allocate incoming payload\n");
|
|
|
|
ng_pktbuf_release(hdr);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
/* copy payload */
|
|
|
|
ng_at86rf2xx_rx_read(dev, payload->data, payload->size, hdr_len);
|
|
|
|
/* finish up and send data to upper layers */
|
|
|
|
dev->event_cb(NETDEV_EVENT_RX_COMPLETE, payload);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _set_state(ng_at86rf2xx_t *dev, ng_netconf_state_t state)
|
|
|
|
{
|
|
|
|
switch (state) {
|
|
|
|
case NETCONF_STATE_SLEEP:
|
|
|
|
ng_at86rf2xx_set_state(dev, NG_AT86RF2XX_STATE_TRX_OFF);
|
|
|
|
break;
|
|
|
|
case NETCONF_STATE_IDLE:
|
|
|
|
ng_at86rf2xx_set_state(dev, NG_AT86RF2XX_STATE_RX_AACK_ON);
|
|
|
|
break;
|
|
|
|
case NETCONF_STATE_TX:
|
|
|
|
if (dev->options & NG_AT86RF2XX_OPT_PRELOADING) {
|
|
|
|
ng_at86rf2xx_tx_exec(dev);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case NETCONF_STATE_RESET:
|
|
|
|
ng_at86rf2xx_reset(dev);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
return sizeof(ng_netconf_state_t);
|
|
|
|
}
|
|
|
|
|
|
|
|
ng_netconf_state_t _get_state(ng_at86rf2xx_t *dev)
|
|
|
|
{
|
|
|
|
switch (ng_at86rf2xx_get_state(dev)) {
|
|
|
|
case NG_AT86RF2XX_STATE_SLEEP:
|
|
|
|
return NETCONF_STATE_SLEEP;
|
|
|
|
case NG_AT86RF2XX_STATE_BUSY_RX_AACK:
|
|
|
|
return NETCONF_STATE_RX;
|
|
|
|
case NG_AT86RF2XX_STATE_BUSY_TX_ARET:
|
|
|
|
case NG_AT86RF2XX_STATE_TX_ARET_ON:
|
|
|
|
return NETCONF_STATE_TX;
|
|
|
|
case NG_AT86RF2XX_STATE_RX_AACK_ON:
|
|
|
|
default:
|
|
|
|
return NETCONF_STATE_IDLE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _get(ng_netdev_t *device, ng_netconf_opt_t opt,
|
|
|
|
void *val, size_t max_len)
|
|
|
|
{
|
|
|
|
if (device == NULL) {
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
ng_at86rf2xx_t *dev = (ng_at86rf2xx_t *) device;
|
|
|
|
|
|
|
|
switch (opt) {
|
|
|
|
|
|
|
|
case NETCONF_OPT_ADDRESS:
|
|
|
|
if (max_len < sizeof(uint16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
*((uint16_t *)val) = ng_at86rf2xx_get_addr_short(dev);
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_ADDRESS_LONG:
|
|
|
|
if (max_len < sizeof(uint64_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
*((uint64_t *)val) = ng_at86rf2xx_get_addr_long(dev);
|
|
|
|
return sizeof(uint64_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_ADDR_LEN:
|
|
|
|
if (max_len < sizeof(uint16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
*((uint16_t *)val) = 2;
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_SRC_LEN:
|
|
|
|
if (max_len < sizeof(uint16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
if (dev->options & NG_AT86RF2XX_OPT_SRC_ADDR_LONG) {
|
|
|
|
*((uint16_t *)val) = 8;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*((uint16_t *)val) = 2;
|
|
|
|
}
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_NID:
|
|
|
|
if (max_len < sizeof(uint16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
*((uint16_t *)val) = dev->pan;
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
2015-06-03 17:08:03 +02:00
|
|
|
case NETCONF_OPT_IPV6_IID:
|
|
|
|
if (max_len < sizeof(eui64_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
if (dev->options & NG_AT86RF2XX_OPT_SRC_ADDR_LONG) {
|
|
|
|
uint64_t addr = ng_at86rf2xx_get_addr_long(dev);
|
|
|
|
ng_ieee802154_get_iid(val, (uint8_t *)&addr, 8);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
uint16_t addr = ng_at86rf2xx_get_addr_short(dev);
|
|
|
|
ng_ieee802154_get_iid(val, (uint8_t *)&addr, 2);
|
|
|
|
}
|
|
|
|
return sizeof(eui64_t);
|
|
|
|
|
2015-05-09 15:59:28 +02:00
|
|
|
case NETCONF_OPT_PROTO:
|
|
|
|
if (max_len < sizeof(ng_nettype_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
*((ng_nettype_t *)val) = dev->proto;
|
|
|
|
return sizeof(ng_nettype_t);
|
|
|
|
|
2015-04-23 18:36:55 +02:00
|
|
|
case NETCONF_OPT_CHANNEL:
|
|
|
|
if (max_len < sizeof(uint16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
((uint8_t *)val)[1] = 0;
|
|
|
|
((uint8_t *)val)[0] = ng_at86rf2xx_get_chan(dev);
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_TX_POWER:
|
|
|
|
if (max_len < sizeof(int16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
*((uint16_t *)val) = ng_at86rf2xx_get_txpower(dev);
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_MAX_PACKET_SIZE:
|
|
|
|
if (max_len < sizeof(int16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
*((uint16_t *)val) = NG_AT86RF2XX_MAX_PKT_LENGTH;
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_STATE:
|
|
|
|
if (max_len < sizeof(ng_netconf_state_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
*((ng_netconf_state_t*)val) = _get_state(dev);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETCONF_OPT_PRELOADING:
|
|
|
|
if (dev->options & NG_AT86RF2XX_OPT_PRELOADING) {
|
|
|
|
*((ng_netconf_enable_t *)val) = NETCONF_ENABLE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*((ng_netconf_enable_t *)val) = NETCONF_DISABLE;
|
|
|
|
}
|
|
|
|
return sizeof(ng_netconf_enable_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_AUTOACK:
|
|
|
|
if (dev->options & NG_AT86RF2XX_OPT_AUTOACK) {
|
|
|
|
*((ng_netconf_enable_t *)val) = NETCONF_ENABLE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*((ng_netconf_enable_t *)val) = NETCONF_DISABLE;
|
|
|
|
}
|
|
|
|
return sizeof(ng_netconf_enable_t);
|
|
|
|
|
2015-05-18 21:19:03 +02:00
|
|
|
case NETCONF_OPT_RETRANS:
|
|
|
|
if (max_len < sizeof(uint8_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
*((uint8_t *)val) = ng_at86rf2xx_get_max_retries(dev);
|
|
|
|
return sizeof(uint8_t);
|
|
|
|
|
2015-04-23 18:36:55 +02:00
|
|
|
case NETCONF_OPT_PROMISCUOUSMODE:
|
|
|
|
if (dev->options & NG_AT86RF2XX_OPT_PROMISCUOUS) {
|
|
|
|
*((ng_netconf_enable_t *)val) = NETCONF_ENABLE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*((ng_netconf_enable_t *)val) = NETCONF_DISABLE;
|
|
|
|
}
|
|
|
|
return sizeof(ng_netconf_enable_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_RAWMODE:
|
|
|
|
if (dev->options & NG_AT86RF2XX_OPT_RAWDUMP) {
|
|
|
|
*((ng_netconf_enable_t *)val) = NETCONF_ENABLE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*((ng_netconf_enable_t *)val) = NETCONF_DISABLE;
|
|
|
|
}
|
|
|
|
return sizeof(ng_netconf_enable_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_IS_CHANNEL_CLR:
|
|
|
|
if (ng_at86rf2xx_cca(dev)) {
|
|
|
|
*((ng_netconf_enable_t *)val) = NETCONF_ENABLE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*((ng_netconf_enable_t *)val) = NETCONF_DISABLE;
|
|
|
|
}
|
|
|
|
return sizeof(ng_netconf_enable_t);
|
|
|
|
|
2015-05-05 22:50:14 +02:00
|
|
|
case NETCONF_OPT_RX_START_IRQ:
|
|
|
|
*((ng_netconf_enable_t *)val) =
|
|
|
|
!!(dev->options & NG_AT86RF2XX_OPT_TELL_RX_START);
|
|
|
|
return sizeof(ng_netconf_enable_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_RX_END_IRQ:
|
|
|
|
*((ng_netconf_enable_t *)val) =
|
|
|
|
!!(dev->options & NG_AT86RF2XX_OPT_TELL_RX_END);
|
|
|
|
return sizeof(ng_netconf_enable_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_TX_START_IRQ:
|
|
|
|
*((ng_netconf_enable_t *)val) =
|
|
|
|
!!(dev->options & NG_AT86RF2XX_OPT_TELL_TX_START);
|
|
|
|
return sizeof(ng_netconf_enable_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_TX_END_IRQ:
|
|
|
|
*((ng_netconf_enable_t *)val) =
|
|
|
|
!!(dev->options & NG_AT86RF2XX_OPT_TELL_TX_END);
|
2015-05-05 22:49:04 +02:00
|
|
|
return sizeof(ng_netconf_enable_t);
|
|
|
|
|
2015-04-23 18:36:55 +02:00
|
|
|
default:
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _set(ng_netdev_t *device, ng_netconf_opt_t opt,
|
|
|
|
void *val, size_t len)
|
|
|
|
{
|
|
|
|
ng_at86rf2xx_t *dev = (ng_at86rf2xx_t *) device;
|
|
|
|
|
|
|
|
if (dev == NULL) {
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (opt) {
|
|
|
|
case NETCONF_OPT_ADDRESS:
|
|
|
|
if (len > sizeof(uint16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
ng_at86rf2xx_set_addr_short(dev, *((uint16_t*)val));
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_ADDRESS_LONG:
|
|
|
|
if (len > sizeof(uint64_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
ng_at86rf2xx_set_addr_long(dev, *((uint64_t*)val));
|
|
|
|
return sizeof(uint64_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_SRC_LEN:
|
|
|
|
if (len > sizeof(uint16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
if (*((uint16_t *)val) == 2) {
|
|
|
|
ng_at86rf2xx_set_option(dev, NG_AT86RF2XX_OPT_SRC_ADDR_LONG,
|
|
|
|
false);
|
|
|
|
}
|
|
|
|
else if (*((uint16_t *)val) == 8) {
|
|
|
|
ng_at86rf2xx_set_option(dev, NG_AT86RF2XX_OPT_SRC_ADDR_LONG,
|
|
|
|
true);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_NID:
|
|
|
|
if (len > sizeof(uint16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
ng_at86rf2xx_set_pan(dev, *((uint16_t *)val));
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_CHANNEL:
|
|
|
|
if (len != sizeof(uint16_t)) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
uint8_t chan = ((uint8_t *)val)[0];
|
|
|
|
if (chan < NG_AT86RF2XX_MIN_CHANNEL ||
|
|
|
|
chan > NG_AT86RF2XX_MAX_CHANNEL) {
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
ng_at86rf2xx_set_chan(dev, chan);
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_TX_POWER:
|
|
|
|
if (len > sizeof(int16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
ng_at86rf2xx_set_txpower(dev, *((int16_t *)val));
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_STATE:
|
|
|
|
if (len > sizeof(ng_netconf_state_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
return _set_state(dev, *((ng_netconf_state_t *)val));
|
|
|
|
|
|
|
|
case NETCONF_OPT_AUTOACK:
|
|
|
|
ng_at86rf2xx_set_option(dev, NG_AT86RF2XX_OPT_AUTOACK,
|
|
|
|
((bool *)val)[0]);
|
|
|
|
return sizeof(ng_netconf_enable_t);
|
|
|
|
|
2015-05-18 21:19:03 +02:00
|
|
|
case NETCONF_OPT_RETRANS:
|
|
|
|
if (len > sizeof(uint8_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
ng_at86rf2xx_set_max_retries(dev, *((uint8_t *)val));
|
|
|
|
return sizeof(uint8_t);
|
|
|
|
|
2015-04-23 18:36:55 +02:00
|
|
|
case NETCONF_OPT_PRELOADING:
|
|
|
|
ng_at86rf2xx_set_option(dev, NG_AT86RF2XX_OPT_PRELOADING,
|
|
|
|
((bool *)val)[0]);
|
|
|
|
return sizeof(ng_netconf_enable_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_PROMISCUOUSMODE:
|
|
|
|
ng_at86rf2xx_set_option(dev, NG_AT86RF2XX_OPT_PROMISCUOUS,
|
|
|
|
((bool *)val)[0]);
|
|
|
|
return sizeof(ng_netconf_enable_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_RAWMODE:
|
|
|
|
ng_at86rf2xx_set_option(dev, NG_AT86RF2XX_OPT_RAWDUMP,
|
|
|
|
((bool *)val)[0]);
|
|
|
|
return sizeof(ng_netconf_enable_t);
|
|
|
|
|
2015-05-05 22:50:14 +02:00
|
|
|
case NETCONF_OPT_RX_START_IRQ:
|
|
|
|
ng_at86rf2xx_set_option(dev, NG_AT86RF2XX_OPT_TELL_RX_START,
|
|
|
|
((bool *)val)[0]);
|
|
|
|
return sizeof(ng_netconf_enable_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_RX_END_IRQ:
|
|
|
|
ng_at86rf2xx_set_option(dev, NG_AT86RF2XX_OPT_TELL_RX_END,
|
|
|
|
((bool *)val)[0]);
|
|
|
|
return sizeof(ng_netconf_enable_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_TX_START_IRQ:
|
|
|
|
ng_at86rf2xx_set_option(dev, NG_AT86RF2XX_OPT_TELL_TX_START,
|
|
|
|
((bool *)val)[0]);
|
|
|
|
return sizeof(ng_netconf_enable_t);
|
|
|
|
|
|
|
|
case NETCONF_OPT_TX_END_IRQ:
|
|
|
|
ng_at86rf2xx_set_option(dev, NG_AT86RF2XX_OPT_TELL_TX_END,
|
|
|
|
((bool *)val)[0]);
|
|
|
|
return sizeof(ng_netconf_enable_t);
|
|
|
|
|
2015-04-23 18:36:55 +02:00
|
|
|
default:
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _add_event_cb(ng_netdev_t *dev, ng_netdev_event_cb_t cb)
|
|
|
|
{
|
|
|
|
if (dev == NULL) {
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
if (dev->event_cb) {
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
|
|
|
|
dev->event_cb = cb;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _rem_event_cb(ng_netdev_t *dev, ng_netdev_event_cb_t cb)
|
|
|
|
{
|
|
|
|
if (dev == NULL) {
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
if (dev->event_cb != cb) {
|
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
dev->event_cb = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _isr_event(ng_netdev_t *device, uint32_t event_type)
|
|
|
|
{
|
|
|
|
ng_at86rf2xx_t *dev = (ng_at86rf2xx_t *) device;
|
|
|
|
uint8_t irq_mask;
|
|
|
|
uint8_t state;
|
|
|
|
|
|
|
|
/* read (consume) device status */
|
|
|
|
irq_mask = ng_at86rf2xx_reg_read(dev, NG_AT86RF2XX_REG__IRQ_STATUS);
|
|
|
|
state = ng_at86rf2xx_get_state(dev);
|
|
|
|
|
|
|
|
if (irq_mask & NG_AT86RF2XX_IRQ_STATUS_MASK__RX_START) {
|
2015-05-05 22:50:14 +02:00
|
|
|
dev->event_cb(NETDEV_EVENT_RX_STARTED, NULL);
|
2015-04-23 18:36:55 +02:00
|
|
|
DEBUG("[ng_at86rf2xx] EVT - RX_START\n");
|
|
|
|
}
|
|
|
|
if (irq_mask & NG_AT86RF2XX_IRQ_STATUS_MASK__TRX_END) {
|
|
|
|
if (state == NG_AT86RF2XX_STATE_RX_AACK_ON || state == NG_AT86RF2XX_STATE_BUSY_RX_AACK) {
|
|
|
|
DEBUG("[ng_at86rf2xx] EVT - RX_END\n");
|
|
|
|
if (!(dev->options & NG_AT86RF2XX_OPT_TELL_RX_END)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
_receive_data(dev);
|
|
|
|
}
|
|
|
|
else if (state == NG_AT86RF2XX_STATE_TX_ARET_ON) {
|
|
|
|
if (dev->event_cb && (dev->options & NG_AT86RF2XX_OPT_TELL_TX_END)) {
|
|
|
|
dev->event_cb(NETDEV_EVENT_TX_COMPLETE, NULL);
|
|
|
|
}
|
|
|
|
DEBUG("[ng_at86rf2xx] EVT - TX_END\n");
|
|
|
|
ng_at86rf2xx_set_state(dev, dev->idle_state);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const ng_netdev_driver_t ng_at86rf2xx_driver = {
|
|
|
|
.send_data = _send,
|
|
|
|
.add_event_callback = _add_event_cb,
|
|
|
|
.rem_event_callback = _rem_event_cb,
|
|
|
|
.get = _get,
|
|
|
|
.set = _set,
|
|
|
|
.isr_event = _isr_event,
|
|
|
|
};
|