2014-07-03 10:36:46 +02:00
|
|
|
/*
|
2014-07-31 19:59:02 +02:00
|
|
|
* at86rf231.c - Implementation of at86rf231 functions.
|
|
|
|
* 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-08-12 17:21:42 +02:00
|
|
|
#include "kernel_types.h"
|
|
|
|
#include "transceiver.h"
|
|
|
|
|
2014-07-03 10:36:46 +02:00
|
|
|
/**
|
|
|
|
* @ingroup drivers_at86rf231
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file at86rf231.c
|
|
|
|
* @brief Driver implementation for at86rf231 chip
|
|
|
|
*
|
|
|
|
* @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_arch.h"
|
|
|
|
#include "at86rf231_spi.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
|
|
|
|
|
|
|
static uint16_t radio_pan;
|
|
|
|
static uint8_t radio_channel;
|
|
|
|
static uint16_t radio_address;
|
|
|
|
static uint64_t radio_address_long;
|
|
|
|
|
2014-07-17 15:42:30 +02:00
|
|
|
uint8_t driver_state;
|
|
|
|
|
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;
|
2013-07-12 12:31:16 +02:00
|
|
|
|
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
|
|
|
|
2013-08-15 19:13:00 +02:00
|
|
|
// TODO : Enable addr decode, auto ack, auto crc
|
|
|
|
// and configure security, power, channel, pan
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2014-01-21 14:39:13 +01:00
|
|
|
radio_pan = 0;
|
|
|
|
radio_pan = 0x00FF & (uint16_t)at86rf231_reg_read(AT86RF231_REG__PAN_ID_0);
|
|
|
|
radio_pan |= (uint16_t)at86rf231_reg_read(AT86RF231_REG__PAN_ID_1) << 8;
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
2013-08-15 19:13:00 +02:00
|
|
|
at86rf231_switch_to_rx();
|
2013-07-12 12:31:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void at86rf231_switch_to_rx(void)
|
|
|
|
{
|
2013-08-15 19:13:00 +02:00
|
|
|
at86rf231_disable_interrupts();
|
|
|
|
// Send a FORCE TRX OFF command
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__TRX_STATE, AT86RF231_TRX_STATE__FORCE_TRX_OFF);
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2013-08-15 19:13:00 +02:00
|
|
|
// Reset IRQ to TRX END only
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__IRQ_MASK, AT86RF231_IRQ_STATUS_MASK__TRX_END);
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2013-08-15 19:13:00 +02:00
|
|
|
// Read IRQ to clear it
|
|
|
|
at86rf231_reg_read(AT86RF231_REG__IRQ_STATUS);
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2013-08-15 19:13:00 +02:00
|
|
|
// Enable IRQ interrupt
|
|
|
|
at86rf231_enable_interrupts();
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2013-08-15 19:13:00 +02:00
|
|
|
// Start RX
|
|
|
|
at86rf231_reg_write(AT86RF231_REG__TRX_STATE, AT86RF231_TRX_STATE__RX_ON);
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2013-08-15 19:13:00 +02:00
|
|
|
// wait until it is on RX_ON state
|
|
|
|
uint8_t status;
|
|
|
|
uint8_t max_wait = 100; // TODO : move elsewhere, this is in 10us
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2013-08-15 19:13:00 +02:00
|
|
|
do {
|
|
|
|
status = at86rf231_get_status();
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2013-08-15 19:13:00 +02:00
|
|
|
vtimer_usleep(10);
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2013-08-15 19:13:00 +02:00
|
|
|
if (!--max_wait) {
|
2014-07-03 10:36:46 +02:00
|
|
|
printf("at86rf231 : ERROR : could not enter RX_ON mode\n");
|
2013-08-15 19:13:00 +02:00
|
|
|
break;
|
|
|
|
}
|
2013-07-12 12:31:16 +02:00
|
|
|
}
|
2013-08-15 19:13:00 +02:00
|
|
|
while ((status & AT86RF231_TRX_STATUS_MASK__TRX_STATUS) != AT86RF231_TRX_STATUS__RX_ON);
|
2013-07-12 12:31:16 +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);
|
|
|
|
/* clear internal state */
|
|
|
|
driver_state = AT_DRIVER_STATE_DEFAULT;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* handle receive */
|
|
|
|
at86rf231_rx_handler();
|
|
|
|
}
|
2013-07-12 12:31:16 +02:00
|
|
|
}
|
|
|
|
|
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-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
|
|
|
|
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-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
|
|
|
|
|
|
|
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
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t at86rf231_set_channel(uint8_t channel)
|
|
|
|
{
|
2014-01-24 12:38:37 +01:00
|
|
|
uint8_t cca_state;
|
2013-08-15 19:13:00 +02:00
|
|
|
radio_channel = channel;
|
|
|
|
|
|
|
|
if (channel < RF86RF231_MIN_CHANNEL ||
|
|
|
|
channel > RF86RF231_MAX_CHANNEL) {
|
|
|
|
radio_channel = RF86RF231_MAX_CHANNEL;
|
|
|
|
}
|
2013-07-12 12:31:16 +02:00
|
|
|
|
2014-03-08 17:12:55 +01:00
|
|
|
cca_state = at86rf231_reg_read(AT86RF231_REG__PHY_CC_CCA) & ~AT86RF231_PHY_CC_CCA_MASK__CHANNEL;
|
2014-01-24 12:38:37 +01:00
|
|
|
at86rf231_reg_write(AT86RF231_REG__PHY_CC_CCA, cca_state | (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
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t at86rf231_get_channel(void)
|
2013-08-15 19:13:00 +02:00
|
|
|
{
|
|
|
|
return radio_channel;
|
2013-07-12 12:31:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void at86rf231_set_monitor(uint8_t mode)
|
|
|
|
{
|
2014-01-24 12:38:37 +01:00
|
|
|
(void) mode;
|
2013-08-15 19:13:00 +02:00
|
|
|
// TODO
|
2013-07-12 12:31:16 +02:00
|
|
|
}
|