2015-08-26 10:21:26 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 Freie Universität Berlin
|
|
|
|
* 2015 Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
*
|
|
|
|
* 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_cc110x
|
|
|
|
* @{
|
|
|
|
* @file
|
2017-02-15 13:07:34 +01:00
|
|
|
* @brief Implementation of netdev interface for cc110x
|
2015-08-26 10:21:26 +02:00
|
|
|
*
|
|
|
|
* @author Fabian Nack <nack@inf.fu-berlin.de>
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#include "cc110x.h"
|
2017-02-15 13:07:34 +01:00
|
|
|
#include "cc110x-netdev.h"
|
2015-11-19 16:37:16 +01:00
|
|
|
#include "cc110x-internal.h"
|
|
|
|
#include "cc110x-interface.h"
|
2017-08-30 21:10:10 +02:00
|
|
|
#include "cc110x-defines.h"
|
2015-08-26 10:21:26 +02:00
|
|
|
#include "net/eui64.h"
|
|
|
|
|
|
|
|
#include "periph/gpio.h"
|
2017-02-15 13:07:34 +01:00
|
|
|
#include "net/netdev.h"
|
2015-08-26 10:21:26 +02:00
|
|
|
#include "net/gnrc/nettype.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
2018-01-12 00:19:03 +01:00
|
|
|
static int _send(netdev_t *dev, const iolist_t *iolist)
|
2015-08-26 10:21:26 +02:00
|
|
|
{
|
|
|
|
DEBUG("%s:%u\n", __func__, __LINE__);
|
|
|
|
|
2018-01-12 00:19:03 +01:00
|
|
|
netdev_cc110x_t *netdev_cc110x = (netdev_cc110x_t *)dev;
|
|
|
|
cc110x_pkt_t *cc110x_pkt = iolist->iol_base;
|
2015-08-26 10:21:26 +02:00
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
return cc110x_send(&netdev_cc110x->cc110x, cc110x_pkt);
|
2015-08-26 10:21:26 +02:00
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _recv(netdev_t *dev, void *buf, size_t len, void *info)
|
2015-08-26 10:21:26 +02:00
|
|
|
{
|
|
|
|
DEBUG("%s:%u\n", __func__, __LINE__);
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
cc110x_t *cc110x = &((netdev_cc110x_t*) dev)->cc110x;
|
2015-08-26 10:21:26 +02:00
|
|
|
|
|
|
|
cc110x_pkt_t *cc110x_pkt = &cc110x->pkt_buf.packet;
|
|
|
|
if (cc110x_pkt->length > len) {
|
|
|
|
return -ENOSPC;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(buf, (void*)cc110x_pkt, cc110x_pkt->length);
|
2016-02-23 15:51:10 +01:00
|
|
|
if (info != NULL) {
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_cc110x_rx_info_t *cc110x_info = info;
|
2016-02-23 15:51:10 +01:00
|
|
|
|
2017-08-30 21:10:10 +02:00
|
|
|
cc110x_info->rssi = (int16_t)cc110x->pkt_buf.rssi/2 - CC110X_RSSI_OFFSET;
|
2016-02-23 15:51:10 +01:00
|
|
|
cc110x_info->lqi = cc110x->pkt_buf.lqi;
|
|
|
|
}
|
2015-08-26 10:21:26 +02:00
|
|
|
return cc110x_pkt->length;
|
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static inline int _get_iid(netdev_t *netdev, eui64_t *value, size_t max_len)
|
2015-08-26 10:21:26 +02:00
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
cc110x_t *cc110x = &((netdev_cc110x_t*) netdev)->cc110x;
|
2015-08-26 10:21:26 +02:00
|
|
|
uint8_t *eui64 = (uint8_t*) value;
|
|
|
|
|
2016-03-26 19:16:05 +01:00
|
|
|
if (max_len < sizeof(eui64_t)) {
|
|
|
|
return -EOVERFLOW;
|
2015-08-26 10:21:26 +02:00
|
|
|
}
|
|
|
|
|
2016-03-26 19:16:05 +01:00
|
|
|
/* make address compatible to https://tools.ietf.org/html/rfc6282#section-3.2.2*/
|
|
|
|
memset(eui64, 0, sizeof(eui64_t));
|
|
|
|
eui64[3] = 0xff;
|
|
|
|
eui64[4] = 0xfe;
|
|
|
|
eui64[7] = cc110x->radio_address;
|
2015-08-26 10:21:26 +02:00
|
|
|
|
|
|
|
return sizeof(eui64_t);
|
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _get(netdev_t *dev, netopt_t opt, void *value, size_t value_len)
|
2015-08-26 10:21:26 +02:00
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
cc110x_t *cc110x = &((netdev_cc110x_t*) dev)->cc110x;
|
2015-08-26 10:21:26 +02:00
|
|
|
|
|
|
|
switch (opt) {
|
|
|
|
case NETOPT_DEVICE_TYPE:
|
|
|
|
assert(value_len == 2);
|
2017-02-15 13:07:34 +01:00
|
|
|
*((uint16_t *) value) = NETDEV_TYPE_CC110X;
|
2015-08-26 10:21:26 +02:00
|
|
|
return 2;
|
2017-01-15 20:36:48 +01:00
|
|
|
#ifdef MODULE_GNRC_NETIF
|
2015-08-26 10:21:26 +02:00
|
|
|
case NETOPT_PROTO:
|
|
|
|
assert(value_len == sizeof(gnrc_nettype_t));
|
2015-10-31 11:20:26 +01:00
|
|
|
*((gnrc_nettype_t *)value) = cc110x->proto;
|
2015-08-26 10:21:26 +02:00
|
|
|
return sizeof(gnrc_nettype_t);
|
2017-01-15 20:36:48 +01:00
|
|
|
#endif
|
2015-08-26 10:21:26 +02:00
|
|
|
case NETOPT_CHANNEL:
|
|
|
|
assert(value_len > 1);
|
|
|
|
*((uint16_t *)value) = (uint16_t)cc110x->radio_channel;
|
2017-11-14 09:30:23 +01:00
|
|
|
return sizeof(uint16_t);
|
2015-08-26 10:21:26 +02:00
|
|
|
case NETOPT_ADDRESS:
|
|
|
|
assert(value_len > 0);
|
|
|
|
*((uint8_t *)value) = cc110x->radio_address;
|
2017-11-14 09:30:23 +01:00
|
|
|
return sizeof(uint8_t);
|
2015-08-26 10:21:26 +02:00
|
|
|
case NETOPT_MAX_PACKET_SIZE:
|
|
|
|
assert(value_len > 0);
|
2019-02-01 12:24:34 +01:00
|
|
|
*((uint16_t *)value) = CC110X_PACKET_LENGTH - CC110X_L2_HDR_SIZE;
|
2017-11-14 09:30:23 +01:00
|
|
|
return sizeof(uint16_t);
|
2015-08-26 10:21:26 +02:00
|
|
|
case NETOPT_IPV6_IID:
|
|
|
|
return _get_iid(dev, value, value_len);
|
2017-11-22 13:36:21 +01:00
|
|
|
case NETOPT_ADDR_LEN:
|
|
|
|
case NETOPT_SRC_LEN:
|
|
|
|
*((uint16_t *)value) = sizeof(cc110x->radio_address);
|
|
|
|
return sizeof(uint16_t);
|
2015-08-26 10:21:26 +02:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
|
2017-08-25 07:34:03 +02:00
|
|
|
static int _set(netdev_t *dev, netopt_t opt, const void *value, size_t value_len)
|
2015-08-26 10:21:26 +02:00
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
cc110x_t *cc110x = &((netdev_cc110x_t*) dev)->cc110x;
|
2015-08-26 10:21:26 +02:00
|
|
|
|
|
|
|
switch (opt) {
|
|
|
|
case NETOPT_CHANNEL:
|
|
|
|
{
|
2017-11-14 09:30:23 +01:00
|
|
|
const uint16_t *arg = value;
|
|
|
|
uint8_t channel = (uint8_t)(*arg);
|
make: fix various compile errors with Wextra
pkg, nordic_softdevice_ble: disable CFLAGS to omit compiler error
sys, pm_layered: fix casting nonscalar to the same type
cpu, stm32_common: fix type-limits, remove always true assert
cpu, stm32f4: fix pointer arithmetic in periph/i2c
drivers, at86rf2xx: fix type-limits where condition always true
saul, gpio: fix if no gpio configured for saul
cpu, saml21: add frequency check to periph/timer
driver, cc110x: fix unused param and type-limts errors
boards, wsn430-common: fix old-style-declaration
make: fix old style definition
drivers, sdcard_spi: fix old style typedef
driver, at30tse: remove unnecessary check
driver, nrf24: fix type-limit
driver, pn532: change buffer from char to uint8_t
tests/driver_sdcard: fix type limits
boards, feather-m0: add missing field inits
driver, tcs37727: fix type limits
pkg, emb6: disable some compiler warnings
tests/emb6: disable some compiler warings
pkg, openthread: fix sign compare and unused params
tests/trickle: fix struct init
tests/pthread_cooperation: fix type limits
board, mips-malta: remove feature periph_uart
shell: fix var size for netif command
gnrc, netif: fix sign-compare
gnrc, nib: fix sign-compare
shell: fix output in netif command
posix: fix type-limits in pthread_cond
2017-10-31 11:52:18 +01:00
|
|
|
#if CC110X_MIN_CHANNR
|
|
|
|
if (channel < CC110X_MIN_CHANNR) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
#endif /* CC110X_MIN_CHANNR */
|
|
|
|
if (channel > CC110X_MAX_CHANNR) {
|
2015-08-26 10:21:26 +02:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
if (cc110x_set_channel(cc110x, channel) == -1) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
2017-11-14 09:30:23 +01:00
|
|
|
return sizeof(uint16_t);
|
2015-08-26 10:21:26 +02:00
|
|
|
}
|
|
|
|
case NETOPT_ADDRESS:
|
|
|
|
if (value_len < 1) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
2017-08-25 07:34:03 +02:00
|
|
|
if (!cc110x_set_address(cc110x, *(const uint8_t*)value)) {
|
2015-08-26 10:21:26 +02:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
2017-11-14 09:30:23 +01:00
|
|
|
return sizeof(uint8_t);
|
2017-01-15 20:36:48 +01:00
|
|
|
#ifdef MODULE_GNRC_NETIF
|
2015-10-31 11:20:26 +01:00
|
|
|
case NETOPT_PROTO:
|
|
|
|
if (value_len != sizeof(gnrc_nettype_t)) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
cc110x->proto = (gnrc_nettype_t) value;
|
|
|
|
return sizeof(gnrc_nettype_t);
|
|
|
|
}
|
|
|
|
break;
|
2017-01-15 20:36:48 +01:00
|
|
|
#endif
|
2015-08-26 10:21:26 +02:00
|
|
|
default:
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static void _netdev_cc110x_isr(void *arg)
|
2015-08-26 10:21:26 +02:00
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_t *netdev = (netdev_t*) arg;
|
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_ISR);
|
2015-08-26 10:21:26 +02:00
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static void _netdev_cc110x_rx_callback(void *arg)
|
2015-08-26 10:21:26 +02:00
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_t *netdev = (netdev_t*) arg;
|
|
|
|
cc110x_t *cc110x = &((netdev_cc110x_t*) arg)->cc110x;
|
2015-08-26 10:21:26 +02:00
|
|
|
gpio_irq_disable(cc110x->params.gdo2);
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_RX_COMPLETE);
|
2015-08-26 10:21:26 +02:00
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static void _isr(netdev_t *dev)
|
2015-08-26 10:21:26 +02:00
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
cc110x_t *cc110x = &((netdev_cc110x_t*) dev)->cc110x;
|
|
|
|
cc110x_isr_handler(cc110x, _netdev_cc110x_rx_callback, (void*)dev);
|
2015-08-26 10:21:26 +02:00
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _init(netdev_t *dev)
|
2015-08-26 10:21:26 +02:00
|
|
|
{
|
|
|
|
DEBUG("%s:%u\n", __func__, __LINE__);
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
cc110x_t *cc110x = &((netdev_cc110x_t*) dev)->cc110x;
|
2015-08-26 10:21:26 +02:00
|
|
|
|
2016-02-20 17:10:02 +01:00
|
|
|
gpio_init_int(cc110x->params.gdo2, GPIO_IN, GPIO_BOTH,
|
2018-05-24 19:28:46 +02:00
|
|
|
&_netdev_cc110x_isr, (void*)dev);
|
2015-08-26 10:21:26 +02:00
|
|
|
|
|
|
|
gpio_set(cc110x->params.gdo2);
|
|
|
|
gpio_irq_disable(cc110x->params.gdo2);
|
|
|
|
|
|
|
|
/* Switch to RX mode */
|
|
|
|
cc110x_rd_set_mode(cc110x, RADIO_MODE_ON);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
const netdev_driver_t netdev_cc110x_driver = {
|
2015-08-26 10:21:26 +02:00
|
|
|
.send=_send,
|
|
|
|
.recv=_recv,
|
|
|
|
.init=_init,
|
|
|
|
.get=_get,
|
|
|
|
.set=_set,
|
|
|
|
.isr=_isr
|
|
|
|
};
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
int netdev_cc110x_setup(netdev_cc110x_t *netdev_cc110x, const cc110x_params_t *params)
|
2015-08-26 10:21:26 +02:00
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
DEBUG("netdev_cc110x_setup()\n");
|
|
|
|
netdev_cc110x->netdev.driver = &netdev_cc110x_driver;
|
2015-08-26 10:21:26 +02:00
|
|
|
|
2015-10-31 11:20:26 +01:00
|
|
|
/* set default protocol */
|
|
|
|
#ifdef MODULE_GNRC_NETIF
|
|
|
|
# ifdef MODULE_GNRC_SIXLOWPAN
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_cc110x->cc110x.proto = GNRC_NETTYPE_SIXLOWPAN;
|
2015-10-31 11:20:26 +01:00
|
|
|
# else
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_cc110x->cc110x.proto = GNRC_NETTYPE_UNDEF;
|
2015-10-31 11:20:26 +01:00
|
|
|
# endif
|
|
|
|
#endif
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
return cc110x_setup(&netdev_cc110x->cc110x, params);
|
2015-08-26 10:21:26 +02:00
|
|
|
}
|