2016-07-01 15:31:10 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 Freie Universität Berlin
|
|
|
|
* 2016 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 drivers_cc2420
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Netdev adaption for the cc2420 driver
|
|
|
|
*
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
* @author Francisco Acosta <francisco.acosta@inria.fr>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <assert.h>
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#include "net/eui64.h"
|
|
|
|
#include "net/ieee802154.h"
|
2017-02-15 13:07:34 +01:00
|
|
|
#include "net/netdev.h"
|
|
|
|
#include "net/netdev/ieee802154.h"
|
2016-07-01 15:31:10 +02:00
|
|
|
#include "xtimer.h"
|
|
|
|
|
|
|
|
#include "cc2420.h"
|
|
|
|
#include "cc2420_netdev.h"
|
|
|
|
#include "cc2420_internal.h"
|
|
|
|
#include "cc2420_registers.h"
|
|
|
|
|
2020-10-22 11:34:31 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2016-07-01 15:31:10 +02:00
|
|
|
#include "debug.h"
|
|
|
|
|
2018-01-12 00:19:03 +01:00
|
|
|
static int _send(netdev_t *netdev, const iolist_t *iolist);
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _recv(netdev_t *netdev, void *buf, size_t len, void *info);
|
|
|
|
static int _init(netdev_t *netdev);
|
|
|
|
static void _isr(netdev_t *netdev);
|
|
|
|
static int _get(netdev_t *netdev, netopt_t opt, void *val, size_t max_len);
|
2017-08-25 07:34:03 +02:00
|
|
|
static int _set(netdev_t *netdev, netopt_t opt, const void *val, size_t len);
|
2016-07-01 15:31:10 +02:00
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
const netdev_driver_t cc2420_driver = {
|
2016-07-01 15:31:10 +02:00
|
|
|
.send = _send,
|
|
|
|
.recv = _recv,
|
|
|
|
.init = _init,
|
|
|
|
.isr = _isr,
|
|
|
|
.get = _get,
|
|
|
|
.set = _set,
|
|
|
|
};
|
|
|
|
|
|
|
|
static void _irq_handler(void *arg)
|
|
|
|
{
|
2021-06-21 17:05:07 +02:00
|
|
|
netdev_t *dev = arg;
|
2016-07-01 15:31:10 +02:00
|
|
|
|
2020-03-05 15:11:20 +01:00
|
|
|
netdev_trigger_event_isr(dev);
|
2016-07-01 15:31:10 +02:00
|
|
|
}
|
|
|
|
|
2017-08-25 07:34:03 +02:00
|
|
|
static inline uint16_t to_u16(const void *buf)
|
2016-07-01 15:31:10 +02:00
|
|
|
{
|
2017-08-25 07:34:03 +02:00
|
|
|
return *((const uint16_t *)buf);
|
2016-07-01 15:31:10 +02:00
|
|
|
}
|
|
|
|
|
2017-08-25 07:34:03 +02:00
|
|
|
static inline int16_t to_i16(const void *buf)
|
2016-07-01 15:31:10 +02:00
|
|
|
{
|
2017-08-25 07:34:03 +02:00
|
|
|
return *((const int16_t *)buf);
|
2016-07-01 15:31:10 +02:00
|
|
|
}
|
|
|
|
|
2017-08-25 07:34:03 +02:00
|
|
|
static inline bool to_bool(const void *buf)
|
2016-07-01 15:31:10 +02:00
|
|
|
{
|
2017-08-25 07:34:03 +02:00
|
|
|
return *((const bool *)buf);
|
2016-07-01 15:31:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline int w_u16(void *buf, uint16_t val)
|
|
|
|
{
|
|
|
|
memcpy(buf, &val, sizeof(uint16_t));
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int w_i16(void *buf, int16_t val)
|
|
|
|
{
|
|
|
|
memcpy(buf, &val, sizeof(int16_t));
|
|
|
|
return sizeof(int16_t);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int opt_state(void *buf, bool cond)
|
|
|
|
{
|
|
|
|
*((netopt_enable_t *)buf) = !!(cond);
|
|
|
|
return sizeof(netopt_enable_t);
|
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _init(netdev_t *netdev)
|
2016-07-01 15:31:10 +02:00
|
|
|
{
|
2021-06-21 17:05:07 +02:00
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev, netdev_ieee802154_t, netdev);
|
|
|
|
cc2420_t *dev = container_of(netdev_ieee802154, cc2420_t, netdev);
|
2016-07-01 15:31:10 +02:00
|
|
|
|
|
|
|
uint16_t reg;
|
|
|
|
|
|
|
|
/* initialize power and reset pins -> put the device into reset state */
|
|
|
|
gpio_init(dev->params.pin_reset, GPIO_OUT);
|
|
|
|
gpio_set(dev->params.pin_reset);
|
|
|
|
gpio_init(dev->params.pin_vrefen, GPIO_OUT);
|
|
|
|
gpio_clear(dev->params.pin_vrefen);
|
|
|
|
|
|
|
|
/* initialize the input lines */
|
|
|
|
gpio_init(dev->params.pin_cca, GPIO_IN);
|
|
|
|
gpio_init(dev->params.pin_sfd, GPIO_IN);
|
|
|
|
gpio_init(dev->params.pin_fifo, GPIO_IN);
|
|
|
|
gpio_init_int(dev->params.pin_fifop, GPIO_IN, GPIO_RISING, _irq_handler, dev);
|
|
|
|
|
|
|
|
/* initialize the chip select line and the SPI bus */
|
2016-11-08 18:40:41 +01:00
|
|
|
spi_init_cs(dev->params.spi, dev->params.pin_cs);
|
2016-07-01 15:31:10 +02:00
|
|
|
|
|
|
|
/* power on and toggle reset */
|
|
|
|
gpio_set(dev->params.pin_vrefen);
|
|
|
|
gpio_clear(dev->params.pin_reset);
|
|
|
|
xtimer_usleep(CC2420_RESET_DELAY);
|
|
|
|
gpio_set(dev->params.pin_reset);
|
|
|
|
|
|
|
|
/* test the connection to the device by reading MANFIDL register */
|
|
|
|
reg = cc2420_reg_read(dev, CC2420_REG_MANFIDL);
|
|
|
|
if (reg != CC2420_MANFIDL_VAL) {
|
|
|
|
DEBUG("cc2420: init: unable to communicate with device\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* turn on the oscillator and wait for it to be stable */
|
|
|
|
cc2420_en_xosc(dev);
|
|
|
|
if (!(cc2420_status(dev) & CC2420_STATUS_XOSC_STABLE)) {
|
|
|
|
DEBUG("cc2420: init: oscillator did not stabilize\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2022-08-18 12:02:47 +02:00
|
|
|
int res = cc2420_init(dev);
|
|
|
|
if (res == 0) {
|
|
|
|
/* signal link UP */
|
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_LINK_UP);
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
2016-07-01 15:31:10 +02:00
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static void _isr(netdev_t *netdev)
|
2016-07-01 15:31:10 +02:00
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_RX_COMPLETE);
|
2016-07-01 15:31:10 +02:00
|
|
|
}
|
|
|
|
|
2018-01-12 00:19:03 +01:00
|
|
|
static int _send(netdev_t *netdev, const iolist_t *iolist)
|
2016-07-01 15:31:10 +02:00
|
|
|
{
|
2021-06-21 17:05:07 +02:00
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev, netdev_ieee802154_t, netdev);
|
|
|
|
cc2420_t *dev = container_of(netdev_ieee802154, cc2420_t, netdev);
|
2018-01-12 00:19:03 +01:00
|
|
|
return (int)cc2420_send(dev, iolist);
|
2016-07-01 15:31:10 +02:00
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _recv(netdev_t *netdev, void *buf, size_t len, void *info)
|
2016-07-01 15:31:10 +02:00
|
|
|
{
|
2021-06-21 17:05:07 +02:00
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev, netdev_ieee802154_t, netdev);
|
|
|
|
cc2420_t *dev = container_of(netdev_ieee802154, cc2420_t, netdev);
|
2016-08-02 16:56:15 +02:00
|
|
|
return (int)cc2420_rx(dev, buf, len, info);
|
2016-07-01 15:31:10 +02:00
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _get(netdev_t *netdev, netopt_t opt, void *val, size_t max_len)
|
2016-07-01 15:31:10 +02:00
|
|
|
{
|
|
|
|
if (netdev == NULL) {
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
2021-06-21 17:05:07 +02:00
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev, netdev_ieee802154_t, netdev);
|
|
|
|
cc2420_t *dev = container_of(netdev_ieee802154, cc2420_t, netdev);
|
2016-07-01 15:31:10 +02:00
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
int ext = netdev_ieee802154_get(&dev->netdev, opt, val, max_len);
|
2016-07-01 15:31:10 +02:00
|
|
|
if (ext > 0) {
|
|
|
|
return ext;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (opt) {
|
|
|
|
|
|
|
|
case NETOPT_ADDRESS:
|
|
|
|
assert(max_len >= sizeof(uint16_t));
|
|
|
|
cc2420_get_addr_short(dev, val);
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
|
|
|
case NETOPT_ADDRESS_LONG:
|
|
|
|
assert(max_len >= 8);
|
|
|
|
cc2420_get_addr_long(dev, val);
|
|
|
|
return 8;
|
|
|
|
|
|
|
|
case NETOPT_NID:
|
|
|
|
assert(max_len >= sizeof(uint16_t));
|
|
|
|
return w_u16(val, cc2420_get_pan(dev));
|
|
|
|
|
|
|
|
case NETOPT_CHANNEL:
|
|
|
|
assert(max_len >= sizeof(uint16_t));
|
|
|
|
return w_u16(val, cc2420_get_chan(dev));
|
|
|
|
|
|
|
|
case NETOPT_TX_POWER:
|
|
|
|
assert(max_len >= sizeof(int16_t));
|
|
|
|
return w_i16(val, cc2420_get_txpower(dev));
|
|
|
|
|
|
|
|
case NETOPT_STATE:
|
|
|
|
assert(max_len >= sizeof(netopt_state_t));
|
|
|
|
*((netopt_state_t *)val) = cc2420_get_state(dev);
|
|
|
|
return sizeof(netopt_state_t);
|
|
|
|
|
|
|
|
case NETOPT_IS_CHANNEL_CLR:
|
|
|
|
return opt_state(val, cc2420_cca(dev));
|
|
|
|
|
|
|
|
case NETOPT_AUTOACK:
|
|
|
|
return opt_state(val, (dev->options & CC2420_OPT_AUTOACK));
|
|
|
|
|
|
|
|
case NETOPT_CSMA:
|
|
|
|
return opt_state(val, (dev->options & CC2420_OPT_CSMA));
|
|
|
|
|
|
|
|
case NETOPT_PRELOADING:
|
|
|
|
return opt_state(val, (dev->options & CC2420_OPT_PRELOADING));
|
|
|
|
|
|
|
|
case NETOPT_PROMISCUOUSMODE:
|
|
|
|
return opt_state(val, (dev->options & CC2420_OPT_PROMISCUOUS));
|
|
|
|
|
|
|
|
case NETOPT_RX_START_IRQ:
|
|
|
|
case NETOPT_TX_START_IRQ:
|
|
|
|
case NETOPT_TX_END_IRQ:
|
2021-02-26 11:39:18 +01:00
|
|
|
*((netopt_enable_t *)val) = NETOPT_ENABLE;
|
|
|
|
return sizeof(netopt_enable_t);
|
2016-07-01 15:31:10 +02:00
|
|
|
|
|
|
|
default:
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-25 07:34:03 +02:00
|
|
|
static int _set(netdev_t *netdev, netopt_t opt, const void *val, size_t val_len)
|
2016-07-01 15:31:10 +02:00
|
|
|
{
|
|
|
|
if (netdev == NULL) {
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
2021-06-21 17:05:07 +02:00
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev, netdev_ieee802154_t, netdev);
|
|
|
|
cc2420_t *dev = container_of(netdev_ieee802154, cc2420_t, netdev);
|
2016-07-01 15:31:10 +02:00
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
int ext = netdev_ieee802154_set(&dev->netdev, opt, val, val_len);
|
2016-07-01 15:31:10 +02:00
|
|
|
|
|
|
|
switch (opt) {
|
|
|
|
case NETOPT_ADDRESS:
|
|
|
|
assert(val_len == 2);
|
2017-08-25 07:34:03 +02:00
|
|
|
cc2420_set_addr_short(dev, val);
|
2016-07-01 15:31:10 +02:00
|
|
|
return 2;
|
|
|
|
|
|
|
|
case NETOPT_ADDRESS_LONG:
|
|
|
|
assert(val_len == 8);
|
2017-08-25 07:34:03 +02:00
|
|
|
cc2420_set_addr_long(dev, val);
|
2016-07-01 15:31:10 +02:00
|
|
|
return 8;
|
|
|
|
|
|
|
|
case NETOPT_NID:
|
|
|
|
assert(val_len == sizeof(uint16_t));
|
|
|
|
cc2420_set_pan(dev, to_u16(val));
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
|
|
|
case NETOPT_CHANNEL:
|
|
|
|
assert(val_len == sizeof(uint16_t));
|
|
|
|
return cc2420_set_chan(dev, to_u16(val));
|
|
|
|
|
|
|
|
case NETOPT_TX_POWER:
|
|
|
|
assert(val_len == sizeof(int16_t));
|
|
|
|
cc2420_set_txpower(dev, to_i16(val));
|
|
|
|
return sizeof(int16_t);
|
|
|
|
|
|
|
|
case NETOPT_STATE:
|
|
|
|
assert(val_len == sizeof(netopt_state_t));
|
|
|
|
return cc2420_set_state(dev, *((netopt_state_t *)val));
|
|
|
|
|
|
|
|
case NETOPT_AUTOACK:
|
|
|
|
return cc2420_set_option(dev, CC2420_OPT_AUTOACK, to_bool(val));
|
|
|
|
|
|
|
|
case NETOPT_CSMA:
|
|
|
|
return cc2420_set_option(dev, CC2420_OPT_CSMA, to_bool(val));
|
|
|
|
|
|
|
|
case NETOPT_PRELOADING:
|
|
|
|
return cc2420_set_option(dev, CC2420_OPT_PRELOADING, to_bool(val));
|
|
|
|
|
|
|
|
case NETOPT_PROMISCUOUSMODE:
|
|
|
|
return cc2420_set_option(dev, CC2420_OPT_PROMISCUOUS, to_bool(val));
|
|
|
|
|
|
|
|
default:
|
|
|
|
return ext;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|