mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
624 lines
20 KiB
C
624 lines
20 KiB
C
|
/*
|
||
|
* Copyright (C) 2018 Otto-von-Guericke-Universität Magdeburg
|
||
|
*
|
||
|
* 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
|
||
|
* @brief Implementation of RIOT's netdev_driver API for the CC1100/CC1101
|
||
|
* transceiver
|
||
|
*
|
||
|
* @author Marian Buschsieweke <marian.buschsieweke@ovgu.de>
|
||
|
* @}
|
||
|
*/
|
||
|
|
||
|
#include <errno.h>
|
||
|
#include <string.h>
|
||
|
|
||
|
#include "assert.h"
|
||
|
#include "iolist.h"
|
||
|
#include "irq.h"
|
||
|
#include "luid.h"
|
||
|
#include "mutex.h"
|
||
|
#include "net/eui64.h"
|
||
|
#include "net/netdev.h"
|
||
|
#include "xtimer.h"
|
||
|
|
||
|
#include "cc110x.h"
|
||
|
#include "cc110x_internal.h"
|
||
|
|
||
|
#define ENABLE_DEBUG (0)
|
||
|
#include "debug.h"
|
||
|
|
||
|
static int cc110x_init(netdev_t *netdev);
|
||
|
static int cc110x_recv(netdev_t *netdev, void *buf, size_t len, void *info);
|
||
|
static int cc110x_send(netdev_t *netdev, const iolist_t *iolist);
|
||
|
static int cc110x_get(netdev_t *netdev, netopt_t opt,
|
||
|
void *val, size_t max_len);
|
||
|
static int cc110x_set(netdev_t *netdev, netopt_t opt,
|
||
|
const void *val, size_t len);
|
||
|
|
||
|
/**
|
||
|
* @brief A lookup table to convert from dBm value to the best matching
|
||
|
* @ref cc110x_tx_power_t value
|
||
|
*/
|
||
|
static const int8_t tx_power_from_dbm[] = {
|
||
|
[CC110X_TX_POWER_MINUS_30_DBM] = -25,
|
||
|
[CC110X_TX_POWER_MINUS_20_DBM] = -17,
|
||
|
[CC110X_TX_POWER_MINUS_15_DBM] = -12,
|
||
|
[CC110X_TX_POWER_MINUS_10_DBM] = -5,
|
||
|
[CC110X_TX_POWER_0_DBM] = 3,
|
||
|
[CC110X_TX_POWER_PLUS_5_DBM] = 6,
|
||
|
[CC110X_TX_POWER_PLUS_7_DBM] = 9,
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* @brief A lookup table to convert an @ref cc110x_tx_power_t value to dBm
|
||
|
*/
|
||
|
static const int8_t dbm_from_tx_power[] = {
|
||
|
[CC110X_TX_POWER_MINUS_30_DBM] = -30,
|
||
|
[CC110X_TX_POWER_MINUS_20_DBM] = -20,
|
||
|
[CC110X_TX_POWER_MINUS_15_DBM] = -15,
|
||
|
[CC110X_TX_POWER_MINUS_10_DBM] = -10,
|
||
|
[CC110X_TX_POWER_0_DBM] = 0,
|
||
|
[CC110X_TX_POWER_PLUS_5_DBM] = 5,
|
||
|
[CC110X_TX_POWER_PLUS_7_DBM] = 7,
|
||
|
[CC110X_TX_POWER_PLUS_10_DBM] = 10,
|
||
|
};
|
||
|
|
||
|
const netdev_driver_t cc110x_driver = {
|
||
|
.init = cc110x_init,
|
||
|
.recv = cc110x_recv,
|
||
|
.send = cc110x_send,
|
||
|
.isr = cc110x_isr,
|
||
|
.get = cc110x_get,
|
||
|
.set = cc110x_set,
|
||
|
};
|
||
|
|
||
|
void cc110x_on_gdo(void *_dev)
|
||
|
{
|
||
|
cc110x_t *dev = _dev;
|
||
|
if ((dev->state & 0x07) == CC110X_STATE_TX_MODE) {
|
||
|
/* Unlock mutex to unblock netdev thread */
|
||
|
mutex_unlock(&dev->isr_signal);
|
||
|
}
|
||
|
else {
|
||
|
dev->netdev.event_callback(&dev->netdev, NETDEV_EVENT_ISR);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static int identify_device(cc110x_t *dev)
|
||
|
{
|
||
|
uint8_t partnum, version, status;
|
||
|
int is_ready;
|
||
|
cc110x_state_t state;
|
||
|
|
||
|
cc110x_read(dev, CC110X_REG_VERSION, &version);
|
||
|
/* Retrieving the status is reliable for non-transient status */
|
||
|
status = cc110x_read(dev, CC110X_REG_PARTNUM, &partnum);
|
||
|
state = cc110x_state_from_status(status);
|
||
|
/* Most significant bit should be zero, otherwise chip is not ready */
|
||
|
is_ready = cc110x_is_ready_from_status(status);
|
||
|
|
||
|
DEBUG("[cc110x] PARTNUM = %i, VERSION = %i, STATUS = 0x%02x, READY = %i\n",
|
||
|
(int)partnum, (int)version, (int)status, is_ready);
|
||
|
|
||
|
if ((state != CC110X_STATE_IDLE) || (!is_ready)) {
|
||
|
DEBUG("[cc110x] IC not ready or in invalid state\n");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
/* Source: https://e2e.ti.com/support/wireless-connectivity/other-wireless/f/667/t/370643 */
|
||
|
if (partnum != 0) {
|
||
|
DEBUG("[cc110x] Device not a CC110x transceiver\n");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
switch (version) {
|
||
|
case 3:
|
||
|
DEBUG("[cc110x] Detected CC1100 transceiver\n");
|
||
|
/* RSSI offset is 78dBm @ 868MHz & 250kBaud.
|
||
|
* Depends on the symbol rate and base band and ranges from
|
||
|
* 74dBm to 79dBm.
|
||
|
*/
|
||
|
dev->rssi_offset = 78;
|
||
|
return 0;
|
||
|
case 5:
|
||
|
DEBUG("[cc110x] Detected CC1100E transceiver\n");
|
||
|
/* RSSI offset is 79 dBm @ 250kbps & 250 kbps.
|
||
|
* Depends on base band and symbol rate and ranges from
|
||
|
* 75dBm to 79dBm
|
||
|
*/
|
||
|
dev->rssi_offset = 79;
|
||
|
return 0;
|
||
|
case 4:
|
||
|
/* falls through */
|
||
|
case 14:
|
||
|
/* falls through */
|
||
|
case 20:
|
||
|
/* RSSI offset for the CC1101 is independent of symbol rate and
|
||
|
* base 74 dBm
|
||
|
*/
|
||
|
dev->rssi_offset = 74;
|
||
|
DEBUG("[cc110x] Detected CC1101 transceiver\n");
|
||
|
return 0;
|
||
|
default:
|
||
|
DEBUG("[cc110x] Device not a CC110x transceiver\n");
|
||
|
return -1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static int check_config(cc110x_t *dev)
|
||
|
{
|
||
|
char buf[CC110X_CONF_SIZE];
|
||
|
|
||
|
/* Verify content of main config registers */
|
||
|
cc110x_burst_read(dev, CC110X_CONF_START, buf, sizeof(buf));
|
||
|
if (memcmp(buf, cc110x_conf, sizeof(buf))) {
|
||
|
DEBUG("[cc110x] ERROR: Verification of main config registers failed\n"
|
||
|
" Check SPI wiring or reduce SPI clock\n");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
/* Verify content of "magic number" config registers */
|
||
|
cc110x_burst_read(dev, CC110X_REG_TEST2, buf,
|
||
|
sizeof(cc110x_magic_registers));
|
||
|
if (memcmp(buf, cc110x_magic_registers, sizeof(cc110x_magic_registers))) {
|
||
|
DEBUG("[cc110x] ERROR: Verification of \"magic\" registers failed\n"
|
||
|
" Check SPI wiring or reduce SPI clock\n");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
/* Verify content of PA_TABLE */
|
||
|
cc110x_burst_read(dev, CC110X_MULTIREG_PATABLE, buf, CC110X_PATABLE_LEN);
|
||
|
if (memcmp(buf, dev->params.patable->data, CC110X_PATABLE_LEN)) {
|
||
|
DEBUG("[cc110x] ERROR: Verification of PA_TABLE failed\n"
|
||
|
" Check SPI wiring or reduce SPI clock\n");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
DEBUG("[cc110x] Content of configuration registers verified\n");
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int check_gdo_pins(cc110x_t *dev)
|
||
|
{
|
||
|
/* GPIOs connected to GDOs are not yet configured, so we do this now.
|
||
|
* This configuration is just temporarily for testing, as gpio_init_int()
|
||
|
* will reconfigure them later again. We do not want to set up the
|
||
|
* interrupts before validating the transceiver, so we effectively configure
|
||
|
* the GPIOs twice.
|
||
|
*/
|
||
|
if (gpio_init(dev->params.gdo2, GPIO_IN)) {
|
||
|
DEBUG("[cc110x] Configuring GDO2 failed");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
if (gpio_init(dev->params.gdo0, GPIO_IN)) {
|
||
|
DEBUG("[cc110x] Configuring GDO0 failed");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
/* Validate that GDO2 responds to configuration updates */
|
||
|
cc110x_write(dev, CC110X_REG_IOCFG2, CC110X_GDO_CONSTANT_HIGH);
|
||
|
if (!gpio_read(dev->params.gdo2)) {
|
||
|
DEBUG("[cc110x] GDO2 does not respond (check wiring!)\n");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
cc110x_write(dev, CC110X_REG_IOCFG2, CC110X_GDO_CONSTANT_LOW);
|
||
|
if (gpio_read(dev->params.gdo2)) {
|
||
|
DEBUG("[cc110x] GDO2 does not respond (check wiring!)\n");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
/* Validate that GDO0 responds to configuration updates */
|
||
|
cc110x_write(dev, CC110X_REG_IOCFG0, CC110X_GDO_CONSTANT_HIGH);
|
||
|
if (!gpio_read(dev->params.gdo0)) {
|
||
|
DEBUG("[cc110x] GDO0 does not respond (check wiring!)\n");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
cc110x_write(dev, CC110X_REG_IOCFG0, CC110X_GDO_CONSTANT_LOW);
|
||
|
if (gpio_read(dev->params.gdo0)) {
|
||
|
DEBUG("[cc110x] GDO0 does not respond (check wiring!)\n");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
/* Restore default GDO2 & GDO0 config */
|
||
|
cc110x_write(dev, CC110X_REG_IOCFG2, cc110x_conf[CC110X_REG_IOCFG2]);
|
||
|
cc110x_write(dev, CC110X_REG_IOCFG0, cc110x_conf[CC110X_REG_IOCFG0]);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int cc110x_init(netdev_t *netdev)
|
||
|
{
|
||
|
cc110x_t *dev = (cc110x_t *)netdev;
|
||
|
/* Use locked mutex to block thread on TX and un-block from ISR */
|
||
|
mutex_init(&dev->isr_signal);
|
||
|
mutex_lock(&dev->isr_signal);
|
||
|
|
||
|
/* Make sure the crystal is stable and the chip ready. This is needed as
|
||
|
* the reset is done via an SPI command, but the SPI interface must not be
|
||
|
* used unless the chip is ready according to the data sheet. After the
|
||
|
* reset, a second call to cc110x_power_on() is needed to finally have
|
||
|
* the transceiver in a known state and ready for SPI communication.
|
||
|
*/
|
||
|
if (cc110x_power_on(dev)) {
|
||
|
DEBUG("[cc110x] netdev_driver_t::init(): Failed to pull CS pin low\n");
|
||
|
return -EIO;
|
||
|
}
|
||
|
|
||
|
if (cc110x_acquire(dev) != SPI_OK) {
|
||
|
DEBUG("[cc110x] netdev_driver_t::init(): Failed to setup/acquire SPI "
|
||
|
"interface\n");
|
||
|
return -EIO;
|
||
|
}
|
||
|
|
||
|
/* Performing a reset of the transceiver to get it in a known state */
|
||
|
cc110x_cmd(dev, CC110X_STROBE_RESET);
|
||
|
cc110x_release(dev);
|
||
|
|
||
|
/* Again, make sure the crystal is stable and the chip ready */
|
||
|
if (cc110x_power_on(dev)) {
|
||
|
DEBUG("[cc110x] netdev_driver_t::init(): Failed to pull CS pin low "
|
||
|
"after reset\n");
|
||
|
return -EIO;
|
||
|
}
|
||
|
|
||
|
if (cc110x_acquire(dev) != SPI_OK) {
|
||
|
DEBUG("[cc110x] netdev_driver_t::init(): Failed to setup/acquire SPI "
|
||
|
"interface after reset\n");
|
||
|
return -EIO;
|
||
|
}
|
||
|
|
||
|
if (identify_device(dev)) {
|
||
|
DEBUG("[cc110x] netdev_driver_t::init(): Device identification failed\n");
|
||
|
cc110x_release(dev);
|
||
|
return -ENOTSUP;
|
||
|
}
|
||
|
|
||
|
/* Upload the main configuration */
|
||
|
cc110x_burst_write(dev, CC110X_CONF_START, cc110x_conf, CC110X_CONF_SIZE);
|
||
|
/* Set TX power to match uploaded configuration */
|
||
|
dev->tx_power = CC110X_TX_POWER_0_DBM;
|
||
|
|
||
|
/* Upload the poorly documented magic numbers obtained via SmartRF Studio */
|
||
|
cc110x_burst_write(dev, CC110X_REG_TEST2, cc110x_magic_registers,
|
||
|
sizeof(cc110x_magic_registers));
|
||
|
|
||
|
/* Setup the selected PA_TABLE */
|
||
|
cc110x_burst_write(dev, CC110X_MULTIREG_PATABLE,
|
||
|
dev->params.patable->data, CC110X_PATABLE_LEN);
|
||
|
|
||
|
/* Verify main config, magic numbers and PA_TABLE correctly uploaded */
|
||
|
if (check_config(dev)) {
|
||
|
cc110x_release(dev);
|
||
|
return -EIO;
|
||
|
}
|
||
|
|
||
|
/* Verify that pins GDO2 and GDO0 are correctly connected */
|
||
|
if (check_gdo_pins(dev)) {
|
||
|
cc110x_release(dev);
|
||
|
return -EIO;
|
||
|
}
|
||
|
|
||
|
/* Setup the layer 2 address, but do not accept CC110X_L2ADDR_AUTO (which
|
||
|
* has the value 0x00 and is used for broadcast)
|
||
|
*/
|
||
|
dev->addr = dev->params.l2addr;
|
||
|
while (dev->addr == CC110X_L2ADDR_AUTO) {
|
||
|
luid_get(&dev->addr, 1);
|
||
|
}
|
||
|
cc110x_write(dev, CC110X_REG_ADDR, dev->addr);
|
||
|
|
||
|
/* Setup interrupt on GDO0 */
|
||
|
if (gpio_init_int(dev->params.gdo0, GPIO_IN, GPIO_BOTH,
|
||
|
cc110x_on_gdo, dev)) {
|
||
|
cc110x_release(dev);
|
||
|
DEBUG("[cc110x] netdev_driver_t::init(): Failed to setup interrupt on "
|
||
|
"GDO0 pin\n");
|
||
|
return -EIO;
|
||
|
}
|
||
|
|
||
|
/* Setup interrupt on GDO2 */
|
||
|
if (gpio_init_int(dev->params.gdo2, GPIO_IN, GPIO_BOTH,
|
||
|
cc110x_on_gdo, dev)) {
|
||
|
gpio_irq_disable(dev->params.gdo0);
|
||
|
cc110x_release(dev);
|
||
|
DEBUG("[cc110x] netdev_driver_t::init(): Failed to setup interrupt on "
|
||
|
"GDO2 pin\n");
|
||
|
return -EIO;
|
||
|
}
|
||
|
|
||
|
/* Update the state of the driver/transceiver */
|
||
|
dev->state = CC110X_STATE_IDLE;
|
||
|
cc110x_release(dev);
|
||
|
|
||
|
|
||
|
int retval; /*< Store return value to be able to pass through error code */
|
||
|
/* Apply configuration (if non-NULL) and channel map, which also calls
|
||
|
* cc110x_full_calibration
|
||
|
*/
|
||
|
retval = cc110x_apply_config(dev, dev->params.config, dev->params.channels);
|
||
|
if (retval) {
|
||
|
gpio_irq_disable(dev->params.gdo0);
|
||
|
gpio_irq_disable(dev->params.gdo2);
|
||
|
DEBUG("[cc110x] netdev_driver_t::init(): cc110x_apply_config() "
|
||
|
"failed\n");
|
||
|
/* Pass through received error code */
|
||
|
return retval;
|
||
|
}
|
||
|
else {
|
||
|
DEBUG("[cc110x] netdev_driver_t::init(): Success\n");
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int cc110x_recv(netdev_t *netdev, void *buf, size_t len, void *info)
|
||
|
{
|
||
|
cc110x_t *dev = (cc110x_t *)netdev;
|
||
|
|
||
|
/* Call to cc110x_enter_rx_mode() will clear dev->buf.len, so back up it first */
|
||
|
int size = dev->buf.len;
|
||
|
|
||
|
if (cc110x_acquire(dev) != SPI_OK) {
|
||
|
DEBUG("[cc110x] netdev_driver_t::recv(): cc110x_acquire() "
|
||
|
"failed\n");
|
||
|
return -EIO;
|
||
|
}
|
||
|
|
||
|
/* Copy RX info on last frame (if requested) */
|
||
|
if (info != NULL) {
|
||
|
*((cc1xxx_rx_info_t *)info) = dev->rx_info;
|
||
|
}
|
||
|
|
||
|
if (!buf) {
|
||
|
/* Get the size of the frame; if len > 0 then also drop the frame */
|
||
|
if (len > 0) {
|
||
|
/* Drop frame requested */
|
||
|
cc110x_enter_rx_mode(dev);
|
||
|
}
|
||
|
cc110x_release(dev);
|
||
|
return size;
|
||
|
}
|
||
|
|
||
|
if (len < (size_t)size) {
|
||
|
/* Drop frame and return -ENOBUFS */
|
||
|
cc110x_enter_rx_mode(dev);
|
||
|
cc110x_release(dev);
|
||
|
return -ENOBUFS;
|
||
|
}
|
||
|
|
||
|
memcpy(buf, dev->buf.data, (size_t)size);
|
||
|
|
||
|
cc110x_enter_rx_mode(dev);
|
||
|
cc110x_release(dev);
|
||
|
return size;
|
||
|
}
|
||
|
|
||
|
static int cc110x_send(netdev_t *netdev, const iolist_t *iolist)
|
||
|
{
|
||
|
cc110x_t *dev = (cc110x_t *)netdev;
|
||
|
|
||
|
/* assert that cc110x_send was called with valid parameters */
|
||
|
assert(netdev && iolist && (iolist->iol_len == sizeof(cc1xxx_l2hdr_t)));
|
||
|
|
||
|
if (cc110x_acquire(dev) != SPI_OK) {
|
||
|
DEBUG("[cc110x] netdev_driver_t::send(): cc110x_acquire() failed\n");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
switch (dev->state) {
|
||
|
case CC110X_STATE_FSTXON:
|
||
|
/* falls through */
|
||
|
case CC110X_STATE_RX_MODE:
|
||
|
break;
|
||
|
case CC110X_STATE_RECEIVING:
|
||
|
cc110x_release(dev);
|
||
|
DEBUG("[cc110x] netdev_driver_t::send(): Refusing to send while "
|
||
|
"receiving a frame\n");
|
||
|
return -EBUSY;
|
||
|
default:
|
||
|
cc110x_release(dev);
|
||
|
DEBUG("[cc110x] netdev_driver_t::send(): Driver state %i prevents "
|
||
|
"sending\n", (int)dev->state);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
/* Copy data to send into frame buffer */
|
||
|
size_t size = sizeof(cc1xxx_l2hdr_t);
|
||
|
memcpy(dev->buf.data, iolist->iol_base, sizeof(cc1xxx_l2hdr_t));
|
||
|
|
||
|
for (const iolist_t *iol = iolist->iol_next; iol; iol = iol->iol_next) {
|
||
|
if (iol->iol_len) {
|
||
|
if (size + iol->iol_len > CC110X_MAX_FRAME_SIZE) {
|
||
|
cc110x_release(dev);
|
||
|
DEBUG("[cc110x] netdev_driver_t::send(): Frame size of %uB "
|
||
|
"exceeds maximum supported size of %uB\n",
|
||
|
(unsigned)(size + iol->iol_len),
|
||
|
(unsigned)CC110X_MAX_FRAME_SIZE);
|
||
|
return -1;
|
||
|
}
|
||
|
memcpy(dev->buf.data + size, iol->iol_base, iol->iol_len);
|
||
|
size += iol->iol_len;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
dev->buf.len = (uint8_t)size;
|
||
|
|
||
|
/* Disable IRQs, as GDO configuration will be changed now */
|
||
|
gpio_irq_disable(dev->params.gdo0);
|
||
|
gpio_irq_disable(dev->params.gdo2);
|
||
|
|
||
|
/* Fill the TX FIFO: First write the length, then the frame */
|
||
|
dev->buf.pos = (size > CC110X_FIFO_SIZE - 1) ? CC110X_FIFO_SIZE - 1 : size;
|
||
|
/* cc110x_framebuf_t has the same memory layout as the device expects */
|
||
|
cc110x_burst_write(dev, CC110X_MULTIREG_FIFO,
|
||
|
&dev->buf, dev->buf.pos + 1);
|
||
|
|
||
|
/* Go to TX */
|
||
|
cc110x_cmd(dev, CC110X_STROBE_TX);
|
||
|
|
||
|
/* Configure GDO2 and update state */
|
||
|
if (dev->buf.pos < dev->buf.len) {
|
||
|
/* We need to keep feeding TX FIFO */
|
||
|
cc110x_write(dev, CC110X_REG_IOCFG2, CC110X_GDO_ON_TX_DATA);
|
||
|
dev->state = CC110X_STATE_TX_MODE;
|
||
|
}
|
||
|
else {
|
||
|
/* All data in TX FIFO, just waiting for transceiver to finish */
|
||
|
cc110x_write(dev, CC110X_REG_IOCFG2, CC110X_GDO_CONSTANT_LOW);
|
||
|
dev->state = CC110X_STATE_TX_COMPLETING;
|
||
|
}
|
||
|
|
||
|
cc110x_release(dev);
|
||
|
|
||
|
/* Restore IRQs */
|
||
|
gpio_irq_enable(dev->params.gdo0);
|
||
|
gpio_irq_enable(dev->params.gdo2);
|
||
|
|
||
|
while ((dev->state & 0x07) == CC110X_STATE_TX_MODE) {
|
||
|
/* Block until mutex is unlocked from ISR */
|
||
|
mutex_lock(&dev->isr_signal);
|
||
|
cc110x_isr(&dev->netdev);
|
||
|
}
|
||
|
|
||
|
return (int)size;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Generate an IPv6 interface identifier for a CC110X transceiver
|
||
|
*
|
||
|
* @param dev Transceiver to create the IPv6 interface identifier (IID)
|
||
|
* @param iid Store the generated IID here
|
||
|
*
|
||
|
* @return Returns the size of @ref eui64_t to confirm with the API
|
||
|
* in @ref netdev_driver_t::get
|
||
|
*/
|
||
|
static int cc110x_get_iid(cc110x_t *dev, eui64_t *iid)
|
||
|
{
|
||
|
static const eui64_t empty_iid = {
|
||
|
.uint8 = { 0x00, 0x00, 0x00, 0xff, 0xfe, 0x00, 0x00, 0x00 }
|
||
|
};
|
||
|
|
||
|
*iid = empty_iid;
|
||
|
iid->uint8[7] = dev->addr;
|
||
|
return sizeof(eui64_t);
|
||
|
}
|
||
|
|
||
|
static int cc110x_get(netdev_t *netdev, netopt_t opt,
|
||
|
void *val, size_t max_len)
|
||
|
{
|
||
|
cc110x_t *dev = (cc110x_t *)netdev;
|
||
|
|
||
|
switch (opt) {
|
||
|
case NETOPT_DEVICE_TYPE:
|
||
|
assert(max_len == sizeof(uint16_t));
|
||
|
*((uint16_t *)val) = NETDEV_TYPE_CC110X;
|
||
|
return sizeof(uint16_t);
|
||
|
case NETOPT_PROTO:
|
||
|
assert(max_len == sizeof(gnrc_nettype_t));
|
||
|
*((gnrc_nettype_t *)val) = CC110X_DEFAULT_PROTOCOL;
|
||
|
return sizeof(gnrc_nettype_t);
|
||
|
case NETOPT_MAX_PACKET_SIZE:
|
||
|
assert(max_len == sizeof(uint16_t));
|
||
|
*((uint16_t *)val) = CC110X_MAX_FRAME_SIZE - sizeof(cc1xxx_l2hdr_t);
|
||
|
return sizeof(uint16_t);
|
||
|
case NETOPT_ADDR_LEN:
|
||
|
/* falls through */
|
||
|
case NETOPT_SRC_LEN:
|
||
|
assert(max_len == sizeof(uint16_t));
|
||
|
*((uint16_t *)val) = CC1XXX_ADDR_SIZE;
|
||
|
return sizeof(uint16_t);
|
||
|
case NETOPT_ADDRESS:
|
||
|
assert(max_len >= CC1XXX_ADDR_SIZE);
|
||
|
*((uint8_t *)val) = dev->addr;
|
||
|
return CC1XXX_ADDR_SIZE;
|
||
|
case NETOPT_IPV6_IID:
|
||
|
if (max_len < sizeof(eui64_t)) {
|
||
|
return -EOVERFLOW;
|
||
|
}
|
||
|
return cc110x_get_iid(dev, val);
|
||
|
case NETOPT_CHANNEL:
|
||
|
assert(max_len == sizeof(uint16_t));
|
||
|
*((uint16_t *)val) = dev->channel;
|
||
|
return sizeof(uint16_t);
|
||
|
case NETOPT_TX_POWER:
|
||
|
assert(max_len == sizeof(uint16_t));
|
||
|
*((uint16_t *)val) = dbm_from_tx_power[dev->tx_power];
|
||
|
return sizeof(uint16_t);
|
||
|
default:
|
||
|
return -ENOTSUP;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Set the given address as the device's layer 2 address
|
||
|
*
|
||
|
* @param dev Device descripter of the transceiver
|
||
|
* @param addr Address to set
|
||
|
*/
|
||
|
static int cc110x_set_addr(cc110x_t *dev, uint8_t addr)
|
||
|
{
|
||
|
if (cc110x_acquire(dev) != SPI_OK) {
|
||
|
return -EIO;
|
||
|
}
|
||
|
|
||
|
dev->addr = addr;
|
||
|
cc110x_write(dev, CC110X_REG_ADDR, addr);
|
||
|
cc110x_release(dev);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
static int cc110x_set(netdev_t *netdev, netopt_t opt,
|
||
|
const void *val, size_t len)
|
||
|
{
|
||
|
(void)len;
|
||
|
cc110x_t *dev = (cc110x_t *)netdev;
|
||
|
|
||
|
switch (opt) {
|
||
|
case NETOPT_ADDRESS:
|
||
|
assert(len == CC1XXX_ADDR_SIZE);
|
||
|
return cc110x_set_addr(dev, *((uint8_t *)val));
|
||
|
case NETOPT_CHANNEL:
|
||
|
{
|
||
|
assert(len == sizeof(uint16_t));
|
||
|
int retval;
|
||
|
uint16_t channel = *((uint16_t *)val);
|
||
|
if (channel >= CC110X_MAX_CHANNELS) {
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
if ((retval = cc110x_set_channel(dev, (uint8_t)channel))) {
|
||
|
return retval;
|
||
|
}
|
||
|
}
|
||
|
return sizeof(uint16_t);
|
||
|
case NETOPT_TX_POWER:
|
||
|
{
|
||
|
assert(len == sizeof(int16_t));
|
||
|
int16_t dbm = *((int16_t *)val);
|
||
|
cc110x_tx_power_t power = CC110X_TX_POWER_MINUS_30_DBM;
|
||
|
for ( ; power < CC110X_TX_POWER_PLUS_10_DBM; power++) {
|
||
|
if ((int16_t)tx_power_from_dbm[power] >= dbm) {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (cc110x_set_tx_power(dev, power)) {
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
}
|
||
|
return sizeof(uint16_t);
|
||
|
default:
|
||
|
return -ENOTSUP;
|
||
|
}
|
||
|
}
|