2014-07-03 10:36:46 +02:00
|
|
|
/*
|
2014-07-31 19:59:02 +02:00
|
|
|
* Copyright (C) 2013 Alaeddine Weslati <alaeddine.weslati@inria.fr>
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
2013-08-15 19:13:00 +02:00
|
|
|
|
2014-07-03 10:36:46 +02:00
|
|
|
/**
|
|
|
|
* @ingroup drivers_at86rf231
|
|
|
|
* @{
|
|
|
|
*
|
2014-09-29 20:34:38 +02:00
|
|
|
* @file
|
|
|
|
* @brief Driver implementation of the AT86RF231 device driver
|
2014-07-03 10:36:46 +02:00
|
|
|
*
|
|
|
|
* @author Alaeddine Weslati <alaeddine.weslati@inria.fr>
|
2014-07-17 15:42:30 +02:00
|
|
|
* @author Thomas Eichinger <thomas.eichinger@fu-berlin.de>
|
2014-07-03 10:36:46 +02:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2013-12-16 17:54:58 +01:00
|
|
|
#include "at86rf231.h"
|
2014-02-20 03:05:35 +01:00
|
|
|
#include "at86rf231_spi.h"
|
2014-09-29 20:34:38 +02:00
|
|
|
#include "board.h"
|
|
|
|
#include "periph/gpio.h"
|
|
|
|
#include "periph/spi.h"
|
|
|
|
#include "kernel_types.h"
|
|
|
|
#include "transceiver.h"
|
2014-10-02 00:33:08 +02:00
|
|
|
#include "hwtimer.h"
|
2014-10-31 10:32:16 +01:00
|
|
|
#include "config.h"
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2013-12-16 17:54:58 +01:00
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2014-11-11 16:24:26 +01:00
|
|
|
#ifndef AT86RF231_SPI_SPEED
|
|
|
|
#define SPI_SPEED SPI_SPEED_5MHZ
|
|
|
|
#else
|
|
|
|
#define SPI_SPEED AT86RF231_SPI_SPEED
|
|
|
|
#endif
|
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
#define _MAX_RETRIES (100)
|
|
|
|
|
2013-07-12 12:31:16 +02:00
|
|
|
static uint16_t radio_pan;
|
|
|
|
static uint8_t radio_channel;
|
|
|
|
static uint16_t radio_address;
|
|
|
|
static uint64_t radio_address_long;
|
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
netdev_802154_raw_packet_cb_t at86rf231_raw_packet_cb;
|
|
|
|
netdev_rcv_data_cb_t at86rf231_data_packet_cb;
|
|
|
|
|
|
|
|
/* default source address length for sending in number of byte */
|
|
|
|
static size_t _default_src_addr_len = 2;
|
|
|
|
|
2014-07-17 15:42:30 +02:00
|
|
|
uint8_t driver_state;
|
|
|
|
|
2014-09-29 20:34:38 +02:00
|
|
|
void at86rf231_gpio_spi_interrupts_init(void);
|
|
|
|
void at86rf231_reset(void);
|
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
#ifdef MODULE_TRANSCEIVER
|
2014-07-06 22:57:56 +02:00
|
|
|
void at86rf231_init(kernel_pid_t tpid)
|
2013-07-12 12:31:16 +02:00
|
|
|
{
|
2013-08-15 19:13:00 +02:00
|
|
|
transceiver_pid = tpid;
|
2014-10-20 16:46:53 +02:00
|
|
|
at86rf231_initialize(NULL);
|
|
|
|
}
|
|
|
|
#endif
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
int at86rf231_initialize(netdev_t *dev)
|
|
|
|
{
|
2013-08-15 19:13:00 +02:00
|
|
|
at86rf231_gpio_spi_interrupts_init();
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2013-08-15 19:13:00 +02:00
|
|
|
at86rf231_reset();
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
at86rf231_on();
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
/* TODO :
|
|
|
|
* and configure security, power
|
|
|
|
*/
|
|
|
|
#ifdef MODULE_CONFIG
|
|
|
|
at86rf231_set_pan(sysconfig.radio_pan_id);
|
|
|
|
#else
|
|
|
|
at86rf231_set_pan(0x0001);
|
|
|
|
#endif
|
2014-01-21 14:39:13 +01:00
|
|
|
|
|
|
|
radio_channel = at86rf231_reg_read(AT86RF231_REG__PHY_CC_CCA) & AT86RF231_PHY_CC_CCA_MASK__CHANNEL;
|
|
|
|
|
|
|
|
radio_address = 0x00FF & (uint16_t)at86rf231_reg_read(AT86RF231_REG__SHORT_ADDR_0);
|
|
|
|
radio_address |= at86rf231_reg_read(AT86RF231_REG__SHORT_ADDR_1) << 8;
|
|
|
|
|
|
|
|
radio_address_long = 0x00000000000000FF & (uint64_t)at86rf231_reg_read(AT86RF231_REG__IEEE_ADDR_0);
|
|
|
|
radio_address_long |= ((uint64_t)at86rf231_reg_read(AT86RF231_REG__IEEE_ADDR_1)) << 8;
|
|
|
|
radio_address_long |= ((uint64_t)at86rf231_reg_read(AT86RF231_REG__IEEE_ADDR_1)) << 16;
|
|
|
|
radio_address_long |= ((uint64_t)at86rf231_reg_read(AT86RF231_REG__IEEE_ADDR_1)) << 24;
|
|
|
|
radio_address_long |= ((uint64_t)at86rf231_reg_read(AT86RF231_REG__IEEE_ADDR_1)) << 32;
|
|
|
|
radio_address_long |= ((uint64_t)at86rf231_reg_read(AT86RF231_REG__IEEE_ADDR_1)) << 40;
|
|
|
|
radio_address_long |= ((uint64_t)at86rf231_reg_read(AT86RF231_REG__IEEE_ADDR_1)) << 48;
|
|
|
|
radio_address_long |= ((uint64_t)at86rf231_reg_read(AT86RF231_REG__IEEE_ADDR_1)) << 56;
|
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int at86rf231_on(void)
|
|
|
|
{
|
|
|
|
/* Send a FORCE TRX OFF command */
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__TRX_STATE, AT86RF231_TRX_STATE__FORCE_TRX_OFF);
|
|
|
|
|
|
|
|
/* busy wait for TRX_OFF state */
|
|
|
|
do {
|
|
|
|
int delay = _MAX_RETRIES;
|
|
|
|
if (!--delay) {
|
|
|
|
DEBUG("at86rf231 : ERROR : could not enter TRX_OFF mode\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
} while (at86rf231_get_status() != AT86RF231_TRX_STATUS__TRX_OFF);
|
|
|
|
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__TRX_STATE, AT86RF231_TRX_STATE__RX_ON);
|
|
|
|
|
|
|
|
/* change into basic reception mode to initialise CSMA seed by RNG */
|
|
|
|
do {
|
|
|
|
int delay = _MAX_RETRIES;
|
|
|
|
if (!--delay) {
|
|
|
|
DEBUG("at86rf231 : ERROR : could not enter RX_ON mode\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
} while (at86rf231_get_status() != AT86RF231_TRX_STATUS__RX_ON);
|
|
|
|
|
|
|
|
/* read RNG values into CSMA_SEED_0 */
|
|
|
|
for (int i=0; i<7; i+=2) {
|
|
|
|
uint8_t tmp = at86rf231_reg_read(AT86RF231_REG__PHY_CC_CCA);
|
|
|
|
tmp = ((tmp&0x60)>>5);
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__CSMA_SEED_0, tmp << i);
|
|
|
|
}
|
|
|
|
/* read CSMA_SEED_1 and write back with RNG value */
|
|
|
|
uint8_t tmp = at86rf231_reg_read(AT86RF231_REG__CSMA_SEED_1);
|
2014-10-31 13:48:51 +01:00
|
|
|
tmp |= ((at86rf231_reg_read(AT86RF231_REG__PHY_CC_CCA)&0x60)>>5);
|
2014-10-20 16:46:53 +02:00
|
|
|
at86rf231_reg_write(AT86RF231_REG__CSMA_SEED_1, tmp);
|
|
|
|
|
|
|
|
/* change into reception mode */
|
2013-08-15 19:13:00 +02:00
|
|
|
at86rf231_switch_to_rx();
|
2014-10-20 16:46:53 +02:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void at86rf231_off(void)
|
|
|
|
{
|
|
|
|
/* TODO */
|
|
|
|
}
|
|
|
|
|
|
|
|
int at86rf231_is_on(void)
|
|
|
|
{
|
|
|
|
return ((at86rf231_get_status() & 0x1f) != 0);
|
2013-07-12 12:31:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void at86rf231_switch_to_rx(void)
|
|
|
|
{
|
2014-09-29 20:34:38 +02:00
|
|
|
gpio_irq_disable(AT86RF231_INT);
|
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
/* now change to PLL_ON state for extended operating mode */
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__TRX_STATE, AT86RF231_TRX_STATE__PLL_ON);
|
|
|
|
|
|
|
|
do {
|
|
|
|
int max_wait = _MAX_RETRIES;
|
|
|
|
if (!--max_wait) {
|
|
|
|
DEBUG("at86rf231 : ERROR : could not enter PLL_ON mode\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} while (at86rf231_get_status() != AT86RF231_TRX_STATUS__PLL_ON);
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2014-11-25 17:49:25 +01:00
|
|
|
#ifndef MODULE_OPENWSN
|
2014-10-20 16:46:53 +02:00
|
|
|
/* Reset IRQ to TRX END only */
|
2013-08-15 19:13:00 +02:00
|
|
|
at86rf231_reg_write(AT86RF231_REG__IRQ_MASK, AT86RF231_IRQ_STATUS_MASK__TRX_END);
|
2014-11-25 17:49:25 +01:00
|
|
|
#else
|
|
|
|
/* OpenWSN also needs RX_START IRQ */
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__IRQ_MASK, ( AT86RF231_IRQ_STATUS_MASK__RX_START | AT86RF231_IRQ_STATUS_MASK__TRX_END));
|
|
|
|
#endif
|
|
|
|
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
/* Read IRQ to clear it */
|
2013-08-15 19:13:00 +02:00
|
|
|
at86rf231_reg_read(AT86RF231_REG__IRQ_STATUS);
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
/* Enable IRQ interrupt */
|
2014-09-29 20:34:38 +02:00
|
|
|
gpio_irq_enable(AT86RF231_INT);
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
/* Enter RX_AACK_ON state */
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__TRX_STATE, AT86RF231_TRX_STATE__RX_AACK_ON);
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2013-08-15 19:13:00 +02:00
|
|
|
do {
|
2014-10-20 16:46:53 +02:00
|
|
|
int max_wait = _MAX_RETRIES;
|
2013-08-15 19:13:00 +02:00
|
|
|
if (!--max_wait) {
|
2014-10-20 16:46:53 +02:00
|
|
|
DEBUG("at86rf231 : ERROR : could not enter RX_AACK_ON mode\n");
|
2013-08-15 19:13:00 +02:00
|
|
|
break;
|
|
|
|
}
|
2014-10-20 16:46:53 +02:00
|
|
|
} while (at86rf231_get_status() != AT86RF231_TRX_STATUS__RX_AACK_ON);
|
2013-07-12 12:31:16 +02:00
|
|
|
}
|
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
void at86rf231_rxoverflow_irq(void)
|
2013-07-12 12:31:16 +02:00
|
|
|
{
|
2014-10-20 16:46:53 +02:00
|
|
|
/* TODO */
|
|
|
|
}
|
2014-09-29 20:34:38 +02:00
|
|
|
|
2014-11-25 17:49:25 +01:00
|
|
|
#ifndef MODULE_OPENWSN
|
2014-10-20 16:46:53 +02:00
|
|
|
void at86rf231_rx_irq(void)
|
|
|
|
{
|
2014-07-17 15:42:30 +02:00
|
|
|
/* check if we are in sending state */
|
|
|
|
if (driver_state == AT_DRIVER_STATE_SENDING) {
|
|
|
|
/* Read IRQ to clear it */
|
|
|
|
at86rf231_reg_read(AT86RF231_REG__IRQ_STATUS);
|
2014-08-19 13:41:53 +02:00
|
|
|
/* tx done, listen again */
|
|
|
|
at86rf231_switch_to_rx();
|
2014-07-17 15:42:30 +02:00
|
|
|
/* clear internal state */
|
|
|
|
driver_state = AT_DRIVER_STATE_DEFAULT;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* handle receive */
|
|
|
|
at86rf231_rx_handler();
|
|
|
|
}
|
2013-07-12 12:31:16 +02:00
|
|
|
}
|
2014-11-25 17:49:25 +01:00
|
|
|
#endif
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
int at86rf231_add_raw_recv_callback(netdev_t *dev,
|
|
|
|
netdev_802154_raw_packet_cb_t recv_cb)
|
|
|
|
{
|
|
|
|
(void)dev;
|
|
|
|
|
|
|
|
if (at86rf231_raw_packet_cb == NULL){
|
|
|
|
at86rf231_raw_packet_cb = recv_cb;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -ENOBUFS;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
int at86rf231_rem_raw_recv_callback(netdev_t *dev,
|
|
|
|
netdev_802154_raw_packet_cb_t recv_cb)
|
|
|
|
{
|
|
|
|
(void)dev;
|
|
|
|
|
|
|
|
at86rf231_raw_packet_cb = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int at86rf231_add_data_recv_callback(netdev_t *dev,
|
|
|
|
netdev_rcv_data_cb_t recv_cb)
|
|
|
|
{
|
|
|
|
(void)dev;
|
|
|
|
|
|
|
|
if (at86rf231_data_packet_cb == NULL){
|
|
|
|
at86rf231_data_packet_cb = recv_cb;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
|
|
|
|
int at86rf231_rem_data_recv_callback(netdev_t *dev,
|
|
|
|
netdev_rcv_data_cb_t recv_cb)
|
|
|
|
{
|
|
|
|
(void)dev;
|
|
|
|
|
|
|
|
at86rf231_data_packet_cb = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-01-31 09:50:34 +01:00
|
|
|
radio_address_t at86rf231_set_address(radio_address_t address)
|
2013-07-12 12:31:16 +02:00
|
|
|
{
|
2013-08-15 19:13:00 +02:00
|
|
|
radio_address = address;
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2014-11-04 15:10:18 +01:00
|
|
|
uint8_t old_state = at86rf231_get_status() & AT86RF231_TRX_STATUS_MASK__TRX_STATUS;
|
|
|
|
|
|
|
|
/* Go to state PLL_ON */
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__TRX_STATE, AT86RF231_TRX_STATE__PLL_ON);
|
|
|
|
|
|
|
|
/* wait until it is on PLL_ON state */
|
|
|
|
do {
|
|
|
|
int max_wait = _MAX_RETRIES;
|
|
|
|
if (!--max_wait) {
|
|
|
|
DEBUG("at86rf231 : ERROR : could not enter PLL_ON mode\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} while ((at86rf231_get_status() & AT86RF231_TRX_STATUS_MASK__TRX_STATUS)
|
|
|
|
!= AT86RF231_TRX_STATUS__PLL_ON);
|
|
|
|
|
2014-01-24 12:38:37 +01:00
|
|
|
at86rf231_reg_write(AT86RF231_REG__SHORT_ADDR_0, (uint8_t)(0x00FF & radio_address));
|
2013-08-15 19:13:00 +02:00
|
|
|
at86rf231_reg_write(AT86RF231_REG__SHORT_ADDR_1, (uint8_t)(radio_address >> 8));
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2014-11-04 15:10:18 +01:00
|
|
|
/* Go to state old state */
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__TRX_STATE, old_state);
|
|
|
|
|
|
|
|
/* wait until it is on old state */
|
|
|
|
do {
|
|
|
|
int max_wait = _MAX_RETRIES;
|
|
|
|
if (!--max_wait) {
|
|
|
|
DEBUG("at86rf231 : ERROR : could not enter old state %x\n", old_state);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} while ((at86rf231_get_status() & AT86RF231_TRX_STATUS_MASK__TRX_STATUS)
|
|
|
|
!= old_state);
|
|
|
|
|
2013-08-15 19:13:00 +02:00
|
|
|
return radio_address;
|
2013-07-12 12:31:16 +02:00
|
|
|
}
|
|
|
|
|
2014-01-31 09:50:34 +01:00
|
|
|
radio_address_t at86rf231_get_address(void)
|
2013-07-12 12:31:16 +02:00
|
|
|
{
|
2013-08-15 19:13:00 +02:00
|
|
|
return radio_address;
|
2013-07-12 12:31:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t at86rf231_set_address_long(uint64_t address)
|
|
|
|
{
|
2013-08-15 19:13:00 +02:00
|
|
|
radio_address_long = address;
|
|
|
|
|
2014-11-04 15:10:18 +01:00
|
|
|
uint8_t old_state = at86rf231_get_status() & AT86RF231_TRX_STATUS_MASK__TRX_STATUS;
|
|
|
|
|
|
|
|
/* Go to state PLL_ON */
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__TRX_STATE, AT86RF231_TRX_STATE__PLL_ON);
|
|
|
|
|
|
|
|
/* wait until it is on PLL_ON state */
|
|
|
|
do {
|
|
|
|
int max_wait = _MAX_RETRIES;
|
|
|
|
if (!--max_wait) {
|
|
|
|
DEBUG("at86rf231 : ERROR : could not enter PLL_ON mode\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} while ((at86rf231_get_status() & AT86RF231_TRX_STATUS_MASK__TRX_STATUS)
|
|
|
|
!= AT86RF231_TRX_STATUS__PLL_ON);
|
|
|
|
|
2014-01-24 12:38:37 +01:00
|
|
|
at86rf231_reg_write(AT86RF231_REG__IEEE_ADDR_0, (uint8_t)(0x00000000000000FF & radio_address_long));
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__IEEE_ADDR_1, (uint8_t)(0x000000000000FF00 & radio_address_long >> 8));
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__IEEE_ADDR_2, (uint8_t)(0x0000000000FF0000 & radio_address_long >> 16));
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__IEEE_ADDR_3, (uint8_t)(0x00000000FF000000 & radio_address_long >> 24));
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__IEEE_ADDR_4, (uint8_t)(0x000000FF00000000 & radio_address_long >> 32));
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__IEEE_ADDR_5, (uint8_t)(0x0000FF0000000000 & radio_address_long >> 40));
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__IEEE_ADDR_6, (uint8_t)(0x00FF000000000000 & radio_address_long >> 48));
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__IEEE_ADDR_7, (uint8_t)(radio_address_long >> 56));
|
2013-08-15 19:13:00 +02:00
|
|
|
|
2014-11-04 15:10:18 +01:00
|
|
|
/* Go to state old state */
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__TRX_STATE, old_state);
|
|
|
|
|
|
|
|
/* wait until it is on old state */
|
|
|
|
do {
|
|
|
|
int max_wait = _MAX_RETRIES;
|
|
|
|
if (!--max_wait) {
|
|
|
|
DEBUG("at86rf231 : ERROR : could not enter old state %x\n", old_state);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} while ((at86rf231_get_status() & AT86RF231_TRX_STATUS_MASK__TRX_STATUS)
|
|
|
|
!= old_state);
|
|
|
|
|
2013-08-15 19:13:00 +02:00
|
|
|
return radio_address_long;
|
2013-07-12 12:31:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t at86rf231_get_address_long(void)
|
|
|
|
{
|
2013-08-15 19:13:00 +02:00
|
|
|
return radio_address_long;
|
2013-07-12 12:31:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t at86rf231_set_pan(uint16_t pan)
|
|
|
|
{
|
2013-08-15 19:13:00 +02:00
|
|
|
radio_pan = pan;
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2014-01-24 12:38:37 +01:00
|
|
|
at86rf231_reg_write(AT86RF231_REG__PAN_ID_0, (uint8_t)(0x00FF & radio_pan));
|
2013-08-15 19:13:00 +02:00
|
|
|
at86rf231_reg_write(AT86RF231_REG__PAN_ID_1, (uint8_t)(radio_pan >> 8));
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2013-08-15 19:13:00 +02:00
|
|
|
return radio_pan;
|
2013-07-12 12:31:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t at86rf231_get_pan(void)
|
|
|
|
{
|
2013-08-15 19:13:00 +02:00
|
|
|
return radio_pan;
|
2013-07-12 12:31:16 +02:00
|
|
|
}
|
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
int at86rf231_set_channel(unsigned int channel)
|
2013-07-12 12:31:16 +02:00
|
|
|
{
|
2013-08-15 19:13:00 +02:00
|
|
|
radio_channel = channel;
|
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
if (channel < AT86RF231_MIN_CHANNEL ||
|
|
|
|
channel > AT86RF231_MAX_CHANNEL) {
|
2014-09-15 18:22:01 +02:00
|
|
|
#if DEVELHELP
|
|
|
|
puts("[at86rf231] channel out of range!");
|
|
|
|
#endif
|
|
|
|
return -1;
|
2013-08-15 19:13:00 +02:00
|
|
|
}
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
at86rf231_reg_write(AT86RF231_REG__PHY_CC_CCA, (radio_channel & AT86RF231_PHY_CC_CCA_MASK__CHANNEL));
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2013-08-15 19:13:00 +02:00
|
|
|
return radio_channel;
|
2013-07-12 12:31:16 +02:00
|
|
|
}
|
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
unsigned int at86rf231_get_channel(void)
|
2013-08-15 19:13:00 +02:00
|
|
|
{
|
|
|
|
return radio_channel;
|
2013-07-12 12:31:16 +02:00
|
|
|
}
|
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
void at86rf231_set_monitor(int mode)
|
2013-07-12 12:31:16 +02:00
|
|
|
{
|
2014-10-20 16:46:53 +02:00
|
|
|
/* read register */
|
|
|
|
uint8_t tmp = at86rf231_reg_read(AT86RF231_REG__XAH_CTRL_1);
|
|
|
|
/* set promicuous mode depending on *mode* */
|
|
|
|
if (mode) {
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__XAH_CTRL_1, (tmp|AT86RF231_XAH_CTRL_1__AACK_PROM_MODE));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__XAH_CTRL_1, (tmp&(~AT86RF231_XAH_CTRL_1__AACK_PROM_MODE)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int at86rf231_get_monitor(void)
|
|
|
|
{
|
|
|
|
return (at86rf231_reg_read(AT86RF231_REG__XAH_CTRL_1) & AT86RF231_XAH_CTRL_1__AACK_PROM_MODE);
|
2013-07-12 12:31:16 +02:00
|
|
|
}
|
2014-09-29 20:34:38 +02:00
|
|
|
|
|
|
|
void at86rf231_gpio_spi_interrupts_init(void)
|
|
|
|
{
|
|
|
|
/* SPI init */
|
2015-01-16 13:46:58 +01:00
|
|
|
spi_acquire(AT86RF231_SPI);
|
2014-11-11 16:24:26 +01:00
|
|
|
spi_init_master(AT86RF231_SPI, SPI_CONF_FIRST_RISING, SPI_SPEED);
|
2015-01-16 13:46:58 +01:00
|
|
|
spi_release(AT86RF231_SPI);
|
2014-09-29 20:34:38 +02:00
|
|
|
/* IRQ0 */
|
2014-10-20 16:46:53 +02:00
|
|
|
gpio_init_int(AT86RF231_INT, GPIO_NOPULL, GPIO_RISING, (gpio_cb_t)at86rf231_rx_irq, NULL);
|
2014-09-29 20:34:38 +02:00
|
|
|
/* CS */
|
|
|
|
gpio_init_out(AT86RF231_CS, GPIO_NOPULL);
|
|
|
|
/* SLEEP */
|
|
|
|
gpio_init_out(AT86RF231_SLEEP, GPIO_NOPULL);
|
|
|
|
/* RESET */
|
|
|
|
gpio_init_out(AT86RF231_RESET, GPIO_NOPULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
void at86rf231_reset(void)
|
|
|
|
{
|
|
|
|
/* force reset */
|
|
|
|
gpio_clear(AT86RF231_RESET);
|
|
|
|
|
|
|
|
/* put pins to default values */
|
|
|
|
gpio_set(AT86RF231_CS);
|
|
|
|
gpio_clear(AT86RF231_SLEEP);
|
|
|
|
|
|
|
|
/* additional waiting to comply to min rst pulse width */
|
2014-11-11 16:24:26 +01:00
|
|
|
uint8_t volatile delay = 50; /* volatile to ensure it isn't optimized away */
|
|
|
|
while (--delay);
|
2014-09-29 20:34:38 +02:00
|
|
|
gpio_set(AT86RF231_RESET);
|
2014-10-20 16:46:53 +02:00
|
|
|
}
|
2014-09-29 20:34:38 +02:00
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
int at86rf231_get_option(netdev_t *dev, netdev_opt_t opt, void *value,
|
|
|
|
size_t *value_len)
|
|
|
|
{
|
|
|
|
/* XXX: first check only for backwards compatibility with transceiver
|
|
|
|
* (see at86rf231_init) remove when adapter for transceiver exists */
|
|
|
|
if (dev != &at86rf231_netdev) {
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
2014-09-29 20:34:38 +02:00
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
switch (opt) {
|
|
|
|
case NETDEV_OPT_CHANNEL:
|
|
|
|
if (*value_len < sizeof(unsigned int)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
if (*value_len > sizeof(unsigned int)) {
|
|
|
|
*value_len = sizeof(unsigned int);
|
|
|
|
}
|
|
|
|
*((unsigned int *)value) = at86rf231_get_channel();
|
|
|
|
break;
|
2014-09-29 20:34:38 +02:00
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
case NETDEV_OPT_ADDRESS:
|
|
|
|
if (*value_len < sizeof(uint16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
if (*value_len > sizeof(uint16_t)) {
|
|
|
|
*value_len = sizeof(uint16_t);
|
|
|
|
}
|
|
|
|
*((uint16_t *)value) = at86rf231_get_address();
|
|
|
|
break;
|
2014-09-29 20:34:38 +02:00
|
|
|
|
2014-10-20 16:46:53 +02:00
|
|
|
case NETDEV_OPT_NID:
|
|
|
|
if (*value_len < sizeof(uint16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
if (*value_len > sizeof(uint16_t)) {
|
|
|
|
*value_len = sizeof(uint16_t);
|
|
|
|
}
|
|
|
|
*((uint16_t *)value) = at86rf231_get_pan();
|
2014-09-29 20:34:38 +02:00
|
|
|
break;
|
2014-10-20 16:46:53 +02:00
|
|
|
|
|
|
|
case NETDEV_OPT_ADDRESS_LONG:
|
|
|
|
if (*value_len < sizeof(uint64_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
if (*value_len > sizeof(uint64_t)) {
|
|
|
|
*value_len = sizeof(uint64_t);
|
|
|
|
}
|
|
|
|
*((uint64_t *)value) = at86rf231_get_address_long();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETDEV_OPT_MAX_PACKET_SIZE:
|
|
|
|
if (*value_len == 0) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
if (*value_len > sizeof(uint8_t)) {
|
|
|
|
*value_len = sizeof(uint8_t);
|
|
|
|
}
|
|
|
|
*((uint8_t *)value) = AT86RF231_MAX_PKT_LENGTH;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETDEV_OPT_PROTO:
|
|
|
|
if (*value_len < sizeof(netdev_proto_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
if (*value_len > sizeof(netdev_proto_t)) {
|
|
|
|
*value_len = sizeof(netdev_proto_t);
|
|
|
|
}
|
|
|
|
*((netdev_type_t *)value) = NETDEV_PROTO_802154;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETDEV_OPT_SRC_LEN:
|
|
|
|
if (*value_len < sizeof(size_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
if (*value_len > sizeof(size_t)) {
|
|
|
|
*value_len = sizeof(size_t);
|
|
|
|
}
|
|
|
|
*((size_t *)value) = _default_src_addr_len;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _type_pun_up_unsigned(void *value_out, size_t desired_len,
|
|
|
|
void *value_in, size_t given_len)
|
|
|
|
{
|
|
|
|
if (given_len > desired_len) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* XXX this is ugly, but bear with me */
|
|
|
|
switch (given_len) {
|
|
|
|
case 8:
|
|
|
|
switch (desired_len) {
|
|
|
|
case 8:
|
|
|
|
*((uint64_t *)value_out) = (*((uint64_t *)value_in));
|
|
|
|
return 0;
|
|
|
|
default:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
case 4:
|
|
|
|
switch (desired_len) {
|
|
|
|
case 8:
|
|
|
|
*((uint64_t *)value_out) = (uint64_t)(*((uint32_t *)value_in));
|
|
|
|
return 0;
|
|
|
|
case 4:
|
|
|
|
*((uint32_t *)value_out) = (*((uint32_t *)value_in));
|
|
|
|
return 0;
|
|
|
|
default:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
case 2:
|
|
|
|
switch (desired_len) {
|
|
|
|
case 8:
|
|
|
|
*((uint64_t *)value_out) = (uint64_t)(*((uint16_t *)value_in));
|
|
|
|
return 0;
|
|
|
|
case 4:
|
|
|
|
*((uint32_t *)value_out) = (uint32_t)(*((uint16_t *)value_in));
|
|
|
|
return 0;
|
|
|
|
case 2:
|
|
|
|
*((uint16_t *)value_out) = (*((uint16_t *)value_in));
|
|
|
|
return 0;
|
|
|
|
default:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
case 1:
|
|
|
|
switch (desired_len) {
|
|
|
|
case 8:
|
|
|
|
*((uint64_t *)value_out) = (uint64_t)(*((uint8_t *)value_in));
|
|
|
|
return 0;
|
|
|
|
case 4:
|
|
|
|
*((uint32_t *)value_out) = (uint32_t)(*((uint8_t *)value_in));
|
|
|
|
return 0;
|
|
|
|
case 2:
|
|
|
|
*((uint16_t *)value_out) = (uint16_t)(*((uint8_t *)value_in));
|
|
|
|
return 0;
|
|
|
|
case 1:
|
|
|
|
*((uint8_t *)value_out) = (*((uint8_t *)value_in));
|
|
|
|
return 0;
|
|
|
|
default:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int at86rf231_set_option(netdev_t *dev, netdev_opt_t opt, void *value,
|
|
|
|
size_t value_len)
|
|
|
|
{
|
|
|
|
uint8_t set_value[sizeof(uint64_t)];
|
|
|
|
int res = 0;
|
|
|
|
|
|
|
|
/* XXX: first check only for backwards compatibility with transceiver
|
|
|
|
* (see at86rf231_init) remove when adapter for transceiver exists */
|
|
|
|
if (dev != &at86rf231_netdev) {
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (opt) {
|
|
|
|
case NETDEV_OPT_CHANNEL:
|
|
|
|
if ((res = _type_pun_up_unsigned(set_value, sizeof(unsigned int),
|
|
|
|
value, value_len)) == 0) {
|
|
|
|
unsigned int *v = (unsigned int *)set_value;
|
|
|
|
if (*v > 26) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
at86rf231_set_channel(*v);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETDEV_OPT_ADDRESS:
|
|
|
|
if ((res = _type_pun_up_unsigned(set_value, sizeof(uint16_t),
|
|
|
|
value, value_len)) == 0) {
|
|
|
|
uint16_t *v = (uint16_t *)set_value;
|
|
|
|
if (*v == 0xffff) {
|
|
|
|
/* Do not allow setting to broadcast */
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
at86rf231_set_address(*v);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETDEV_OPT_NID:
|
|
|
|
if ((res = _type_pun_up_unsigned(set_value, sizeof(uint16_t),
|
|
|
|
value, value_len)) == 0) {
|
|
|
|
uint16_t *v = (uint16_t *)set_value;
|
|
|
|
if (*v == 0xffff) {
|
|
|
|
/* Do not allow setting to broadcast */
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
at86rf231_set_pan(*v);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETDEV_OPT_ADDRESS_LONG:
|
|
|
|
if ((res = _type_pun_up_unsigned(set_value, sizeof(uint64_t),
|
|
|
|
value, value_len)) == 0) {
|
|
|
|
uint64_t *v = (uint64_t *)set_value;
|
|
|
|
/* TODO: error checking? */
|
|
|
|
at86rf231_set_address_long(*v);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETDEV_OPT_SRC_LEN:
|
|
|
|
if ((res = _type_pun_up_unsigned(set_value, sizeof(size_t),
|
|
|
|
value, value_len)) == 0) {
|
|
|
|
size_t *v = (size_t *)set_value;
|
|
|
|
|
2014-10-31 13:48:51 +01:00
|
|
|
if (*v != 2 && *v != 8) {
|
2014-10-20 16:46:53 +02:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
_default_src_addr_len = *v;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
2014-09-29 20:34:38 +02:00
|
|
|
}
|
2014-10-20 16:46:53 +02:00
|
|
|
|
|
|
|
int at86rf231_get_state(netdev_t *dev, netdev_state_t *state)
|
|
|
|
{
|
|
|
|
/* XXX: first check only for backwards compatibility with transceiver
|
|
|
|
* (see at86rf231_init) remove when adapter for transceiver exists */
|
|
|
|
if (dev != &at86rf231_netdev) {
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!at86rf231_is_on()) {
|
|
|
|
*state = NETDEV_STATE_POWER_OFF;
|
|
|
|
}
|
|
|
|
else if (at86rf231_get_monitor()) {
|
|
|
|
*state = NETDEV_STATE_PROMISCUOUS_MODE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*state = NETDEV_STATE_RX_MODE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int at86rf231_set_state(netdev_t *dev, netdev_state_t state)
|
|
|
|
{
|
|
|
|
/* XXX: first check only for backwards compatibility with transceiver
|
|
|
|
* (see at86rf231_init) remove when adapter for transceiver exists */
|
|
|
|
if (dev != &at86rf231_netdev) {
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (state != NETDEV_STATE_PROMISCUOUS_MODE && at86rf231_get_monitor()) {
|
|
|
|
at86rf231_set_monitor(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (state) {
|
|
|
|
case NETDEV_STATE_POWER_OFF:
|
|
|
|
at86rf231_off();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETDEV_STATE_RX_MODE:
|
|
|
|
at86rf231_switch_to_rx();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETDEV_STATE_PROMISCUOUS_MODE:
|
|
|
|
at86rf231_set_monitor(1);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int at86rf231_channel_is_clear(netdev_t *dev)
|
|
|
|
{
|
|
|
|
(void)dev;
|
|
|
|
/* channel is checked by hardware automatically before transmission */
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void at86rf231_event(netdev_t *dev, uint32_t event_type)
|
|
|
|
{
|
|
|
|
(void)dev;
|
|
|
|
(void)event_type;
|
|
|
|
}
|
|
|
|
|
|
|
|
const netdev_802154_driver_t at86rf231_driver = {
|
|
|
|
.init = at86rf231_initialize,
|
|
|
|
.send_data = netdev_802154_send_data,
|
|
|
|
.add_receive_data_callback = at86rf231_add_data_recv_callback,
|
|
|
|
.rem_receive_data_callback = at86rf231_rem_data_recv_callback,
|
|
|
|
.get_option = at86rf231_get_option,
|
|
|
|
.set_option = at86rf231_set_option,
|
|
|
|
.get_state = at86rf231_get_state,
|
|
|
|
.set_state = at86rf231_set_state,
|
|
|
|
.event = at86rf231_event,
|
|
|
|
.load_tx = at86rf231_load_tx_buf,
|
|
|
|
.transmit = at86rf231_transmit_tx_buf,
|
|
|
|
.send = netdev_802154_send,
|
|
|
|
.add_receive_raw_callback = at86rf231_add_raw_recv_callback,
|
|
|
|
.rem_receive_raw_callback = at86rf231_rem_raw_recv_callback,
|
|
|
|
.channel_is_clear = at86rf231_channel_is_clear,
|
|
|
|
};
|
|
|
|
|
|
|
|
netdev_t at86rf231_netdev = { NETDEV_TYPE_802154, (netdev_driver_t *) &at86rf231_driver, NULL };
|