nes-proj/arch/cpu/cc13xx-cc26xx/dev/rf-prop-mode.c

882 lines
27 KiB
C
Raw Normal View History

2018-02-08 10:45:47 +00:00
/*
2018-02-15 11:48:02 +00:00
* Copyright (c) 2018, Texas Instruments Incorporated - http://www.ti.com/
2018-02-08 10:45:47 +00:00
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of the copyright holder nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
* COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
* OF THE POSSIBILITY OF SUCH DAMAGE.
*/
/*---------------------------------------------------------------------------*/
/**
* \addtogroup rf-core-prop
* @{
*
* \file
* Implementation of the CC13xx prop mode NETSTACK_RADIO driver
*/
/*---------------------------------------------------------------------------*/
#include "contiki.h"
#include "net/packetbuf.h"
#include "net/netstack.h"
#include "sys/energest.h"
#include "sys/clock.h"
#include "sys/rtimer.h"
#include "sys/cc.h"
2018-02-09 11:50:55 +00:00
#include "dev/watchdog.h"
2018-02-08 10:45:47 +00:00
/*---------------------------------------------------------------------------*/
/* RF Core Mailbox API */
#include <ti/devices/DeviceFamily.h>
#include DeviceFamily_constructPath(driverlib/rf_mailbox.h)
#include DeviceFamily_constructPath(driverlib/rf_common_cmd.h)
#include DeviceFamily_constructPath(driverlib/rf_data_entry.h)
#include DeviceFamily_constructPath(driverlib/rf_prop_cmd.h)
#include DeviceFamily_constructPath(driverlib/rf_prop_mailbox.h)
2018-02-08 10:45:47 +00:00
#include <ti/drivers/rf/RF.h>
2018-02-15 11:48:02 +00:00
/*---------------------------------------------------------------------------*/
/* RF settings */
#ifdef PROP_MODE_CONF_RF_SETTINGS
2018-06-15 14:37:51 +00:00
# define PROP_MODE_RF_SETTINGS PROP_MODE_CONF_RF_SETTINGS
2018-02-15 11:48:02 +00:00
#else
2018-06-15 14:37:51 +00:00
# define PROP_MODE_RF_SETTINGS "prop-settings.h"
2018-02-15 11:48:02 +00:00
#endif
#include PROP_MODE_RF_SETTINGS
/*---------------------------------------------------------------------------*/
/* Platform RF dev */
2018-02-15 11:48:02 +00:00
#include "rf-common.h"
#include "dot-15-4g.h"
2018-02-08 10:45:47 +00:00
/*---------------------------------------------------------------------------*/
#include <stdint.h>
#include <string.h>
#include <stdio.h>
#include <stdbool.h>
2018-02-09 11:50:55 +00:00
#include <assert.h>
2018-02-08 10:45:47 +00:00
/*---------------------------------------------------------------------------*/
#ifdef NDEBUG
2018-06-15 14:37:51 +00:00
# define PRINTF(...)
2018-02-08 10:45:47 +00:00
#else
2018-06-15 14:37:51 +00:00
# define PRINTF(...) printf(__VA_ARGS__)
#endif
/*---------------------------------------------------------------------------*/
/* Configuration for default Prop channel */
#ifdef PROP_MODE_CONF_CHANNEL
# define PROP_MODE_CHANNEL PROP_MODE_CONF_CHANNEL
#else
# define PROP_MODE_CHANNEL RF_CHANNEL
2018-02-08 10:45:47 +00:00
#endif
/*---------------------------------------------------------------------------*/
/* Data whitener. 1: Whitener, 0: No whitener */
#ifdef PROP_MODE_CONF_DW
2018-06-15 14:37:51 +00:00
# define PROP_MODE_DW PROP_MODE_CONF_DW
2018-02-08 10:45:47 +00:00
#else
2018-06-15 14:37:51 +00:00
# define PROP_MODE_DW 0
2018-02-08 10:45:47 +00:00
#endif
#ifdef PROP_MODE_CONF_USE_CRC16
2018-06-15 14:37:51 +00:00
# define PROP_MODE_USE_CRC16 PROP_MODE_CONF_USE_CRC16
2018-02-08 10:45:47 +00:00
#else
2018-06-15 14:37:51 +00:00
# define PROP_MODE_USE_CRC16 0
2018-02-08 10:45:47 +00:00
#endif
/*---------------------------------------------------------------------------*/
/* Used for the return value of channel_clear */
#define RF_CCA_CLEAR 1
#define RF_CCA_BUSY 0
/* Used as an error return value for get_cca_info */
#define RF_GET_CCA_INFO_ERROR 0xFF
/*
* Values of the individual bits of the ccaInfo field in CMD_IEEE_CCA_REQ's
* status struct
*/
#define RF_CMD_CCA_REQ_CCA_STATE_IDLE 0 /* 00 */
#define RF_CMD_CCA_REQ_CCA_STATE_BUSY 1 /* 01 */
#define RF_CMD_CCA_REQ_CCA_STATE_INVALID 2 /* 10 */
#ifdef PROP_MODE_CONF_RSSI_THRESHOLD
2018-06-15 14:37:51 +00:00
# define PROP_MODE_RSSI_THRESHOLD PROP_MODE_CONF_RSSI_THRESHOLD
2018-02-08 10:45:47 +00:00
#else
2018-06-15 14:37:51 +00:00
# define PROP_MODE_RSSI_THRESHOLD 0xA6
2018-02-08 10:45:47 +00:00
#endif
/*---------------------------------------------------------------------------*/
/* Defines and variables related to the .15.4g PHY HDR */
#define DOT_4G_MAX_FRAME_LEN 2047
#define DOT_4G_PHR_LEN 2
/* PHY HDR bits */
#define DOT_4G_PHR_CRC16 0x10
#define DOT_4G_PHR_DW 0x08
#if PROP_MODE_USE_CRC16
/* CRC16 */
2018-06-15 14:37:51 +00:00
# define DOT_4G_PHR_CRC_BIT DOT_4G_PHR_CRC16
# define CRC_LEN 2
2018-02-08 10:45:47 +00:00
#else
/* CRC32 */
2018-06-15 14:37:51 +00:00
# define DOT_4G_PHR_CRC_BIT 0
# define CRC_LEN 4
#endif /* PROP_MODE_USE_CRC16 */
2018-02-08 10:45:47 +00:00
#if PROP_MODE_DW
2018-06-15 14:37:51 +00:00
# define DOT_4G_PHR_DW_BIT DOT_4G_PHR_DW
2018-02-08 10:45:47 +00:00
#else
2018-06-15 14:37:51 +00:00
#define DOT_4G_PHR_DW_BIT 0
2018-02-08 10:45:47 +00:00
#endif
/*---------------------------------------------------------------------------*/
/* How long to wait for the RF to enter RX in rf_cmd_ieee_rx */
#define ENTER_RX_WAIT_TIMEOUT (RTIMER_SECOND >> 10)
/*---------------------------------------------------------------------------*/
/* Configuration for TX power table */
#ifdef PROP_MODE_CONF_TX_POWER_TABLE
2018-06-15 14:37:51 +00:00
# define TX_POWER_TABLE PROP_MODE_CONF_TX_POWER_TABLE
2018-02-08 10:45:47 +00:00
#else
2018-06-15 14:37:51 +00:00
# define TX_POWER_TABLE propTxPowerTable
2018-02-08 10:45:47 +00:00
#endif
/*---------------------------------------------------------------------------*/
/* TX power table convenience macros */
#define TX_POWER_TABLE_SIZE ((sizeof(TX_POWER_TABLE) / sizeof(TX_POWER_TABLE[0])) - 1)
2018-02-08 10:45:47 +00:00
#define TX_POWER_MIN (TX_POWER_TABLE[0].power)
#define TX_POWER_MAX (TX_POWER_TABLE[TX_POWER_TABLE_SIZE - 1].power)
2018-02-08 10:45:47 +00:00
#define TX_POWER_IN_RANGE(dbm) (((dbm) >= TX_POWER_MIN) && ((dbm) <= TX_POWER_MAX))
2018-02-08 10:45:47 +00:00
/*---------------------------------------------------------------------------*/
2018-06-15 14:37:51 +00:00
/* TX buf configuration */
#define TX_BUF_HDR_LEN 2
#define TX_BUF_PAYLOAD_LEN 180
#define TX_BUF_SIZE (TX_BUF_HDR_LEN + TX_BUF_PAYLOAD_LEN)
/* RX buf configuration */
2018-02-08 10:45:47 +00:00
#ifdef PROP_MODE_CONF_RX_BUF_CNT
2018-06-15 14:37:51 +00:00
# define RX_BUF_CNT PROP_MODE_CONF_RX_BUF_CNT
2018-02-08 10:45:47 +00:00
#else
2018-06-15 14:37:51 +00:00
# define RX_BUF_CNT 4
2018-02-08 10:45:47 +00:00
#endif
2018-06-15 14:37:51 +00:00
#define RX_BUF_SIZE 140
2018-02-08 10:45:47 +00:00
/*---------------------------------------------------------------------------*/
#define DATA_ENTRY_LENSZ_NONE 0
#define DATA_ENTRY_LENSZ_BYTE 1
#define DATA_ENTRY_LENSZ_WORD 2 /* 2 bytes */
2018-06-15 14:37:51 +00:00
/*---------------------------------------------------------------------------*/
#define MAC_RADIO_RECEIVER_SENSITIVITY_DBM -110
#define MAC_RADIO_RECEIVER_SATURATION_DBM 10
#define MAC_SPEC_ED_MIN_DBM_ABOVE_RECEIVER_SENSITIVITY 10
#define MAC_SPEC_ED_MAX 0xFF
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
#define ED_RF_POWER_MIN_DBM (MAC_RADIO_RECEIVER_SENSITIVITY_DBM + MAC_SPEC_ED_MIN_DBM_ABOVE_RECEIVER_SENSITIVITY)
#define ED_RF_POWER_MAX_DBM MAC_RADIO_RECEIVER_SATURATION_DBM
/*---------------------------------------------------------------------------*/
typedef struct {
/* Outgoing frame buffer */
uint8_t tx_buf[TX_BUF_SIZE] CC_ALIGN(4);
/* Incoming frame buffer */
uint8_t rx_buf[RX_BUF_CNT][RX_BUF_SIZE] CC_ALIGN(4);
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
/* RX Data Queue */
dataQueue_t rx_data_queue;
/* RX Statistics struct */
rfc_propRxOutput_t rx_stats;
/* Receive entry pointer to keep track of read items */
volatile uint8_t* rx_read_entry;
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
/* RSSI Threshold */
int8_t rssi_threshold;
/* Indicates RF is supposed to be on or off */
uint8_t rf_is_on;
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
/* RF driver */
RF_Object rf_object;
RF_Handle rf_handle;
} prop_radio_t;
static prop_radio_t prop_radio;
2018-02-08 10:45:47 +00:00
/*---------------------------------------------------------------------------*/
2018-06-15 14:37:51 +00:00
#define cmd_radio_setup (*(volatile rfc_CMD_PROP_RADIO_DIV_SETUP_t *)&rf_cmd_prop_radio_div_setup)
#define cmd_fs (*(volatile rfc_CMD_FS_t *) &rf_cmd_prop_fs)
#define cmd_tx (*(volatile rfc_CMD_PROP_TX_ADV_t *) &rf_cmd_prop_tx_adv)
#define cmd_rx (*(volatile rfc_CMD_PROP_RX_ADV_t *) &rf_cmd_prop_rx_adv)
/*---------------------------------------------------------------------------*/
2018-06-15 14:37:51 +00:00
static CC_INLINE bool tx_active(void) { return cmd_tx.status == ACTIVE; }
static CC_INLINE bool rx_active(void) { return cmd_rx.status == ACTIVE; }
2018-02-09 08:13:12 +00:00
/*---------------------------------------------------------------------------*/
2018-06-15 14:37:51 +00:00
static int on(void);
static int off(void);
2018-02-09 11:50:55 +00:00
/*---------------------------------------------------------------------------*/
static void
2018-06-15 14:37:51 +00:00
cmd_rx_cb(RF_Handle client, RF_CmdHandle command, RF_EventMask events)
2018-02-09 11:50:55 +00:00
{
2018-06-15 14:37:51 +00:00
/* Unused arguments */
(void)client;
(void)command;
if (events & RF_EventRxEntryDone) {
process_poll(&rf_process);
}
2018-02-09 11:50:55 +00:00
}
/*---------------------------------------------------------------------------*/
2018-06-15 14:37:51 +00:00
static cmd_result_t
start_rx(void)
2018-02-09 11:50:55 +00:00
{
2018-06-15 14:37:51 +00:00
cmd_rx.status = IDLE;
RF_CmdHandle rx_handle = RF_postCmd(prop_radio.rf_handle, (RF_Op*)&cmd_rx, RF_PriorityNormal,
&cmd_rx_cb, RF_EventRxEntryDone);
if (rx_handle == RF_ALLOC_ERROR) {
PRINTF("start_rx: RF_ALLOC_ERROR for cmd_rx\n");
return CMD_RESULT_ERROR;
}
2018-02-09 11:50:55 +00:00
2018-06-15 14:37:51 +00:00
/* Wait to enter RX */
const rtimer_clock_t t0 = RTIMER_NOW();
while (!rx_active() &&
(RTIMER_CLOCK_LT(RTIMER_NOW(), t0 + ENTER_RX_WAIT_TIMEOUT)));
2018-02-09 11:50:55 +00:00
2018-06-15 14:37:51 +00:00
if (!rx_active()) {
PRINTF("cmd_rx: status=0x%04x\n", cmd_rx.status);
return CMD_RESULT_ERROR;
}
2018-02-09 11:50:55 +00:00
2018-06-15 14:37:51 +00:00
return CMD_RESULT_OK;
2018-02-09 11:50:55 +00:00
}
/*---------------------------------------------------------------------------*/
2018-06-15 14:37:51 +00:00
static cmd_result_t
stop_rx(void)
2018-02-09 11:50:55 +00:00
{
2018-06-15 14:37:51 +00:00
/* Abort any ongoing operation. Don't care about the result. */
RF_cancelCmd(prop_radio.rf_handle, RF_CMDHANDLE_FLUSH_ALL, 1);
2018-02-09 11:50:55 +00:00
2018-06-15 14:37:51 +00:00
/* Todo: maybe do a RF_pendCmd() to synchronize with command execution. */
2018-02-09 11:50:55 +00:00
2018-06-15 14:37:51 +00:00
if(cmd_rx.status != PROP_DONE_STOPPED &&
cmd_rx.status != PROP_DONE_ABORT) {
PRINTF("RF_cmdPropRxAdv cancel: status=0x%04x\n",
cmd_rx.status);
return CMD_RESULT_ERROR;
}
2018-02-09 11:50:55 +00:00
2018-06-15 14:37:51 +00:00
/* Stopped gracefully */
ENERGEST_OFF(ENERGEST_TYPE_LISTEN);
return CMD_RESULT_OK;
}
/*---------------------------------------------------------------------------*/
2018-06-15 14:37:51 +00:00
static cmd_result_t
2018-02-09 11:50:55 +00:00
rf_run_setup()
{
2018-06-15 14:37:51 +00:00
RF_runCmd(prop_radio.rf_handle, (RF_Op*)&cmd_radio_setup, RF_PriorityNormal, NULL, 0);
if (cmd_radio_setup.status != PROP_DONE_OK) {
2018-02-15 11:48:02 +00:00
return CMD_RESULT_ERROR;
2018-02-09 11:50:55 +00:00
}
2018-02-15 11:48:02 +00:00
return CMD_RESULT_OK;
2018-02-09 11:50:55 +00:00
}
/*---------------------------------------------------------------------------*/
2018-02-08 10:45:47 +00:00
static radio_value_t
get_rssi(void)
{
2018-06-15 14:37:51 +00:00
if (tx_active()) {
PRINTF("get_rssi: called while in TX\n");
return RF_GET_RSSI_ERROR_VAL;
}
2018-02-09 08:13:12 +00:00
2018-06-15 14:37:51 +00:00
const bool was_off = !rx_active();
if (was_off && start_rx() == CMD_RESULT_ERROR) {
PRINTF("get_rssi: unable to start RX\n");
return RF_GET_RSSI_ERROR_VAL;
}
2018-02-09 08:13:12 +00:00
2018-06-15 14:37:51 +00:00
int8_t rssi = RF_GET_RSSI_ERROR_VAL;
while(rssi == RF_GET_RSSI_ERROR_VAL || rssi == 0) {
rssi = RF_getRssi(prop_radio.rf_handle);
}
2018-02-09 08:13:12 +00:00
2018-06-15 14:37:51 +00:00
if (was_off) {
stop_rx();
}
2018-02-09 08:13:12 +00:00
2018-06-15 14:37:51 +00:00
return rssi;
2018-02-08 10:45:47 +00:00
}
/*---------------------------------------------------------------------------*/
static uint8_t
get_channel(void)
{
2018-06-15 14:37:51 +00:00
uint32_t freq_khz = cmd_fs.frequency * 1000;
2018-02-09 08:13:12 +00:00
/*
* For some channels, fractFreq * 1000 / 65536 will return 324.99xx.
* Casting the result to uint32_t will truncate decimals resulting in the
* function returning channel - 1 instead of channel. Thus, we do a quick
* positive integer round up.
*/
2018-06-15 14:37:51 +00:00
freq_khz += (((cmd_fs.fractFreq * 1000) + 65535) / 65536);
2018-02-09 08:13:12 +00:00
return (freq_khz - DOT_15_4G_CHAN0_FREQUENCY) / DOT_15_4G_CHANNEL_SPACING;
2018-02-08 10:45:47 +00:00
}
/*---------------------------------------------------------------------------*/
2018-06-15 14:37:51 +00:00
static cmd_result_t
2018-02-08 10:45:47 +00:00
set_channel(uint8_t channel)
{
2018-06-15 14:37:51 +00:00
uint32_t new_freq = DOT_15_4G_CHAN0_FREQUENCY + (channel * DOT_15_4G_CHANNEL_SPACING);
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
uint16_t freq = (uint16_t)(new_freq / 1000);
uint16_t frac = (new_freq - (freq * 1000)) * 65536 / 1000;
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
PRINTF("set_channel: %u = 0x%04x.0x%04x (%lu)\n",
channel, freq, frac, new_freq);
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
cmd_radio_setup.centerFreq = freq;
cmd_fs.frequency = freq;
cmd_fs.fractFreq = frac;
2018-02-09 08:13:12 +00:00
2018-06-15 14:37:51 +00:00
// We don't care whether the FS command is successful because subsequent
// TX and RX commands will tell us indirectly.
RF_EventMask rf_events = RF_runCmd(prop_radio.rf_handle, (RF_Op*)&cmd_fs,
RF_PriorityNormal, NULL, 0);
if ((rf_events & (RF_EventCmdDone | RF_EventLastCmdDone)) == 0) {
PRINTF("set_channel: RF_runCmd failed, events=0x%llx\n", rf_events);
return CMD_RESULT_ERROR;
}
if (cmd_fs.status != DONE_OK) {
PRINTF("set_channel: cmd_fs failed, status=0x%04x\n", cmd_fs.status);
return CMD_RESULT_ERROR;
}
2018-02-09 08:13:12 +00:00
2018-06-15 14:37:51 +00:00
return CMD_RESULT_OK;
2018-02-08 10:45:47 +00:00
}
/*---------------------------------------------------------------------------*/
/* Returns the current TX power in dBm */
static radio_value_t
get_tx_power(void)
{
2018-06-15 14:37:51 +00:00
const RF_TxPowerTable_Value value = RF_getTxPower(prop_radio.rf_handle);
return (radio_value_t)RF_TxPowerTable_findPowerLevel(TX_POWER_TABLE, value);
2018-02-08 10:45:47 +00:00
}
/*---------------------------------------------------------------------------*/
/*
* The caller must make sure to send a new CMD_PROP_RADIO_DIV_SETUP to the
* radio after calling this function.
*/
static radio_result_t
set_tx_power(const radio_value_t power)
2018-02-08 10:45:47 +00:00
{
if (!TX_POWER_IN_RANGE(power)) {
return RADIO_RESULT_INVALID_VALUE;
}
const RF_TxPowerTable_Value value = RF_TxPowerTable_findValue(TX_POWER_TABLE, power);
2018-06-15 14:37:51 +00:00
RF_Stat stat = RF_setTxPower(prop_radio.rf_handle, value);
2018-02-08 10:45:47 +00:00
return (stat == RF_StatSuccess)
? RADIO_RESULT_OK
: RADIO_RESULT_ERROR;
2018-02-08 10:45:47 +00:00
}
/*---------------------------------------------------------------------------*/
2018-06-15 14:37:51 +00:00
static uint8_t
calculate_lqi(int8_t rssi)
{
/* Note : Currently the LQI value is simply the energy detect measurement.
* A more accurate value could be derived by using the correlation
* value along with the RSSI value. */
rssi = CLAMP(rssi, ED_RF_POWER_MIN_DBM, ED_RF_POWER_MAX_DBM);
/* Create energy detect measurement by normalizing and scaling RF power level.
* Note : The division operation below is designed for maximum accuracy and
* best granularity. This is done by grouping the math operations to
* compute the entire numerator before doing any division. */
return (MAC_SPEC_ED_MAX * (rssi - ED_RF_POWER_MIN_DBM)) / (ED_RF_POWER_MAX_DBM - ED_RF_POWER_MIN_DBM);
}
/*---------------------------------------------------------------------------*/
2018-02-08 10:45:47 +00:00
static void
init_rx_buffers(void)
{
2018-06-15 14:37:51 +00:00
size_t i = 0;
for (i = 0; i < RX_BUF_CNT; i++) {
const rfc_dataEntry_t data_entry = {
.status = DATA_ENTRY_PENDING,
.config.type = DATA_ENTRY_TYPE_GEN,
.config.lenSz = DATA_ENTRY_LENSZ_WORD,
.length = RX_BUF_SIZE - sizeof(rfc_dataEntry_t), /* TODO is this sizeof sound? */
/* Point to fist entry if this is last entry, else point to next entry */
.pNextEntry = (i == (RX_BUF_CNT - 1))
? prop_radio.rx_buf[0]
: prop_radio.rx_buf[i]
};
/* Write back data entry struct */
*(rfc_dataEntry_t *)prop_radio.rx_buf[i] = data_entry;
2018-02-08 10:45:47 +00:00
}
}
/*---------------------------------------------------------------------------*/
static int
prepare(const void *payload, unsigned short payload_len)
{
2018-06-15 14:37:51 +00:00
int len = MIN(payload_len, TX_BUF_PAYLOAD_LEN);
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
memcpy(prop_radio.tx_buf + TX_BUF_HDR_LEN, payload, len);
return 0;
2018-02-08 10:45:47 +00:00
}
/*---------------------------------------------------------------------------*/
static int
transmit(unsigned short transmit_len)
{
2018-06-15 14:37:51 +00:00
int ret;
uint8_t was_off = 0;
if (tx_active()) {
PRINTF("transmit: not allowed while transmitting\n");
return RADIO_TX_ERR;
} else if (rx_active()) {
stop_rx();
} else {
was_off = 1;
}
/* Length in .15.4g PHY HDR. Includes the CRC but not the HDR itself */
uint16_t total_length;
/*
* Prepare the .15.4g PHY header
* MS=0, Length MSBits=0, DW and CRC configurable
* Total length = transmit_len (payload) + CRC length
*
* The Radio will flip the bits around, so tx_buf[0] must have the length
* LSBs (PHR[15:8] and tx_buf[1] will have PHR[7:0]
*/
total_length = transmit_len + CRC_LEN;
prop_radio.tx_buf[0] = total_length & 0xFF;
prop_radio.tx_buf[1] = (total_length >> 8) + DOT_4G_PHR_DW_BIT + DOT_4G_PHR_CRC_BIT;
2018-02-09 11:50:55 +00:00
2018-06-15 14:37:51 +00:00
/*
* pktLen: Total number of bytes in the TX buffer, including the header if
* one exists, but not including the CRC (which is not present in the buffer)
*/
cmd_tx.pktLen = transmit_len + DOT_4G_PHR_LEN;
cmd_tx.pPkt = prop_radio.tx_buf;
// TODO: Register callback
RF_runCmd(prop_radio.rf_handle, (RF_Op*)&cmd_tx, RF_PriorityNormal, NULL, 0);
2018-02-09 13:39:40 +00:00
// if (txHandle == RF_ALLOC_ERROR)
// {
// /* Failure sending the CMD_PROP_TX command */
// PRINTF("transmit: PROP_TX_ERR ret=%d, CMDSTA=0x%08lx, status=0x%04x\n",
// ret, cmd_status, cmd_tx_adv->status);
// return RADIO_TX_ERR;
// }
//
// ENERGEST_ON(ENERGEST_TYPE_TRANSMIT);
//
// // watchdog_periodic();
//
// /* Idle away while the command is running */
2018-06-15 14:37:51 +00:00
// RF_pendCmd(rf_handle, txHandle, RF_EventLastCmdDone);
if(cmd_tx.status == PROP_DONE_OK) {
/* Sent OK */
ret = RADIO_TX_OK;
} else {
/* Operation completed, but frame was not sent */
PRINTF("transmit: Not Sent OK status=0x%04x\n",
cmd_tx.status);
ret = RADIO_TX_ERR;
}
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
ENERGEST_OFF(ENERGEST_TYPE_TRANSMIT);
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
/* Workaround. Set status to IDLE */
cmd_tx.status = IDLE;
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
if (was_off) {
RF_yield(prop_radio.rf_handle);
} else {
start_rx();
}
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
return ret;
2018-02-08 10:45:47 +00:00
}
/*---------------------------------------------------------------------------*/
static int
send(const void *payload, unsigned short payload_len)
{
2018-06-15 14:37:51 +00:00
prepare(payload, payload_len);
return transmit(payload_len);
2018-02-08 10:45:47 +00:00
}
/*---------------------------------------------------------------------------*/
static int
2018-06-15 14:37:51 +00:00
read(void *buf, unsigned short buf_len)
2018-02-08 10:45:47 +00:00
{
2018-06-15 14:37:51 +00:00
rfc_dataEntryGeneral_t *entry = (rfc_dataEntryGeneral_t *)prop_radio.rx_read_entry;
uint8_t *data_ptr = &entry->data;
uint16_t len = 0;
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
if (entry->status != DATA_ENTRY_FINISHED) {
return 0;
}
/* First 2 bytes in the data entry are the length.
* Our data entry consists of: Payload + RSSI (1 byte) + Status (1 byte)
* This length includes all of those. */
len = *(uint16_t *)data_ptr;
data_ptr += 2;
len -= 2;
const bool len_ok = (0 < len) && (len <= (uint16_t)buf_len);
if (len_ok) {
memcpy(buf, data_ptr, len);
int8_t rssi = (int8_t)data_ptr[len];
uint8_t lqi = calculate_lqi(rssi);
packetbuf_set_attr(PACKETBUF_ATTR_RSSI, (packetbuf_attr_t)rssi);
packetbuf_set_attr(PACKETBUF_ATTR_LINK_QUALITY, (packetbuf_attr_t)lqi);
}
/* Move read entry pointer to next entry */
prop_radio.rx_read_entry = entry->pNextEntry;
entry->status = DATA_ENTRY_PENDING;
return (len_ok)
? (int)len
: 0;
2018-02-08 10:45:47 +00:00
}
/*---------------------------------------------------------------------------*/
static int
channel_clear(void)
{
2018-06-15 14:37:51 +00:00
if (tx_active()) {
PRINTF("channel_clear: called while in TX\n");
return RF_CCA_CLEAR;
}
2018-02-08 15:41:27 +00:00
2018-06-15 14:37:51 +00:00
const bool rx_was_off = !rx_active();
if (rx_was_off) {
start_rx();
}
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
int8_t rssi = RF_GET_RSSI_ERROR_VAL;
while (rssi == RF_GET_RSSI_ERROR_VAL || rssi == 0) {
rssi = RF_getRssi(prop_radio.rf_handle);
}
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
if (rx_was_off) {
stop_rx();
}
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
if(rssi >= prop_radio.rssi_threshold) {
return RF_CCA_BUSY;
}
return RF_CCA_CLEAR;
2018-02-08 10:45:47 +00:00
}
/*---------------------------------------------------------------------------*/
static int
receiving_packet(void)
{
2018-06-15 14:37:51 +00:00
if (!rx_active()) {
return 0;
}
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
if (channel_clear() == RF_CCA_CLEAR) {
return 0;
}
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
return 1;
2018-02-08 10:45:47 +00:00
}
/*---------------------------------------------------------------------------*/
static int
pending_packet(void)
{
2018-06-15 14:37:51 +00:00
const rfc_dataEntry_t *const first_entry = (rfc_dataEntry_t *)prop_radio.rx_data_queue.pCurrEntry;
volatile const rfc_dataEntry_t *entry = first_entry;
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
int rv = 0;
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
/* Go through RX Circular buffer and check their status */
do {
const uint8_t status = entry->status;
if ((status == DATA_ENTRY_FINISHED) ||
(status == DATA_ENTRY_BUSY)) {
rv += 1;
}
entry = (rfc_dataEntry_t *)entry->pNextEntry;
} while (entry != first_entry);
if (rv > 0) {
process_poll(&rf_process);
}
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
/* If we didn't find an entry at status finished, no frames are pending */
return rv;
2018-02-08 10:45:47 +00:00
}
/*---------------------------------------------------------------------------*/
static int
2018-06-15 14:37:51 +00:00
on(void)
2018-02-08 10:45:47 +00:00
{
2018-06-15 14:37:51 +00:00
if (prop_radio.rf_is_on) {
PRINTF("RF on: Radio already in RX\n");
return CMD_RESULT_OK;
}
/* Reset all RF command statuses */
cmd_radio_setup.status = IDLE;
cmd_fs.status = IDLE;
cmd_tx.status = IDLE;
cmd_rx.status = IDLE;
init_rx_buffers();
const int rx_ok = start_rx();
if (!rx_ok) {
off();
return CMD_RESULT_ERROR;
}
prop_radio.rf_is_on = true;
return CMD_RESULT_OK;
2018-02-08 10:45:47 +00:00
}
/*---------------------------------------------------------------------------*/
static int
2018-06-15 14:37:51 +00:00
off(void)
2018-02-08 10:45:47 +00:00
{
2018-06-15 14:37:51 +00:00
if (!prop_radio.rf_is_on) {
return CMD_RESULT_OK;
}
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
// Force abort of any ongoing RF operation.
RF_flushCmd(prop_radio.rf_handle, RF_CMDHANDLE_FLUSH_ALL, RF_ABORT_GRACEFULLY);
2018-02-09 08:13:12 +00:00
2018-06-15 14:37:51 +00:00
// Trigger a manual power-down
RF_yield(prop_radio.rf_handle);
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
/* We pulled the plug, so we need to restore the status manually */
cmd_radio_setup.status = IDLE;
cmd_fs.status = IDLE;
cmd_tx.status = IDLE;
cmd_rx.status = IDLE;
2018-02-08 10:45:47 +00:00
2018-06-15 14:37:51 +00:00
prop_radio.rf_is_on = false;
return CMD_RESULT_OK;
2018-02-08 10:45:47 +00:00
}
/*---------------------------------------------------------------------------*/
static radio_result_t
get_value(radio_param_t param, radio_value_t *value)
{
if (!value) {
2018-02-08 10:45:47 +00:00
return RADIO_RESULT_INVALID_VALUE;
}
switch (param) {
2018-02-08 10:45:47 +00:00
case RADIO_PARAM_POWER_MODE:
/* On / off */
2018-06-15 14:37:51 +00:00
*value = (prop_radio.rf_is_on)
? RADIO_POWER_MODE_ON
: RADIO_POWER_MODE_OFF;
2018-02-08 10:45:47 +00:00
return RADIO_RESULT_OK;
2018-02-08 10:45:47 +00:00
case RADIO_PARAM_CHANNEL:
*value = (radio_value_t)get_channel();
return RADIO_RESULT_OK;
2018-02-08 10:45:47 +00:00
case RADIO_PARAM_TXPOWER:
*value = get_tx_power();
return RADIO_RESULT_OK;
2018-02-08 10:45:47 +00:00
case RADIO_PARAM_CCA_THRESHOLD:
2018-06-15 14:37:51 +00:00
*value = prop_radio.rssi_threshold;
2018-02-08 10:45:47 +00:00
return RADIO_RESULT_OK;
2018-02-08 10:45:47 +00:00
case RADIO_PARAM_RSSI:
*value = get_rssi();
2018-06-15 14:37:51 +00:00
return (*value == RF_GET_RSSI_ERROR_VAL)
? RADIO_RESULT_ERROR
: RADIO_RESULT_OK;
2018-02-08 10:45:47 +00:00
case RADIO_CONST_CHANNEL_MIN:
*value = 0;
return RADIO_RESULT_OK;
2018-02-08 10:45:47 +00:00
case RADIO_CONST_CHANNEL_MAX:
*value = DOT_15_4G_CHANNEL_MAX;
return RADIO_RESULT_OK;
2018-02-08 10:45:47 +00:00
case RADIO_CONST_TXPOWER_MIN:
*value = (radio_value_t)TX_POWER_MIN;
2018-02-08 10:45:47 +00:00
return RADIO_RESULT_OK;
2018-02-08 10:45:47 +00:00
case RADIO_CONST_TXPOWER_MAX:
*value = (radio_value_t)TX_POWER_MAX;
2018-02-08 10:45:47 +00:00
return RADIO_RESULT_OK;
2018-02-08 10:45:47 +00:00
default:
return RADIO_RESULT_NOT_SUPPORTED;
}
}
/*---------------------------------------------------------------------------*/
static radio_result_t
set_value(radio_param_t param, radio_value_t value)
{
switch(param) {
case RADIO_PARAM_POWER_MODE:
if(value == RADIO_POWER_MODE_ON) {
2018-02-09 11:50:55 +00:00
// Powering on happens implicitly
2018-02-08 10:45:47 +00:00
return RADIO_RESULT_OK;
}
if(value == RADIO_POWER_MODE_OFF) {
2018-06-15 14:37:51 +00:00
off();
2018-02-08 10:45:47 +00:00
return RADIO_RESULT_OK;
}
return RADIO_RESULT_INVALID_VALUE;
2018-02-08 10:45:47 +00:00
case RADIO_PARAM_CHANNEL:
if(value < 0 ||
value > DOT_15_4G_CHANNEL_MAX) {
return RADIO_RESULT_INVALID_VALUE;
}
if(get_channel() == (uint8_t)value) {
/* We already have that very same channel configured.
* Nothing to do here. */
return RADIO_RESULT_OK;
}
set_channel((uint8_t)value);
break;
2018-02-08 10:45:47 +00:00
case RADIO_PARAM_TXPOWER:
return set_tx_power(value);
2018-02-08 10:45:47 +00:00
case RADIO_PARAM_RX_MODE:
return RADIO_RESULT_OK;
2018-02-08 10:45:47 +00:00
case RADIO_PARAM_CCA_THRESHOLD:
2018-06-15 14:37:51 +00:00
prop_radio.rssi_threshold = (int8_t)value;
2018-02-09 11:50:55 +00:00
return RADIO_RESULT_OK;
2018-02-08 10:45:47 +00:00
default:
return RADIO_RESULT_NOT_SUPPORTED;
}
/* If we reach here we had no errors. Apply new settings */
2018-06-15 14:37:51 +00:00
if (rx_active()) {
stop_rx();
if (rf_run_setup() != CMD_RESULT_OK) {
2018-02-08 10:45:47 +00:00
return RADIO_RESULT_ERROR;
2018-06-15 14:37:51 +00:00
}
start_rx();
} else if (tx_active()) {
// Should not happen. TX is always synchronous and blocking.
// Todo: maybe remove completely here.
PRINTF("set_value: cannot apply new value while transmitting. \n");
return RADIO_RESULT_ERROR;
2018-02-09 11:50:55 +00:00
} else {
2018-06-15 14:37:51 +00:00
// was powered off. Nothing to do. New values will be
// applied automatically on next power-up.
2018-02-08 10:45:47 +00:00
}
2018-02-09 11:50:55 +00:00
return RADIO_RESULT_OK;
2018-02-08 10:45:47 +00:00
}
/*---------------------------------------------------------------------------*/
static radio_result_t
get_object(radio_param_t param, void *dest, size_t size)
{
2018-02-09 08:13:12 +00:00
return RADIO_RESULT_NOT_SUPPORTED;
2018-02-08 10:45:47 +00:00
}
/*---------------------------------------------------------------------------*/
static radio_result_t
set_object(radio_param_t param, const void *src, size_t size)
{
2018-02-09 08:13:12 +00:00
return RADIO_RESULT_NOT_SUPPORTED;
2018-02-08 10:45:47 +00:00
}
/*---------------------------------------------------------------------------*/
2018-02-09 11:50:55 +00:00
static int
2018-06-15 14:37:51 +00:00
init(void)
2018-02-09 11:50:55 +00:00
{
2018-06-15 14:37:51 +00:00
/* Zero initalize TX and RX buffers */
memset(prop_radio.tx_buf, 0x0, sizeof(prop_radio.tx_buf));
memset(prop_radio.rx_buf, 0x0, sizeof(prop_radio.rx_buf));
/* Circular buffer, no last entry */
prop_radio.rx_data_queue.pCurrEntry = prop_radio.rx_buf[0];
prop_radio.rx_data_queue.pLastEntry = NULL;
/* Initialize current read pointer to first element (used in ISR) */
prop_radio.rx_read_entry = prop_radio.rx_buf[0];
/* Set configured RSSI threshold */
prop_radio.rssi_threshold = PROP_MODE_RSSI_THRESHOLD;
/* RX is off */
prop_radio.rf_is_on = false;
/* Configure RX command */
cmd_rx.maxPktLen = DOT_4G_MAX_FRAME_LEN - cmd_rx.lenOffset;
cmd_rx.pQueue = &prop_radio.rx_data_queue;
cmd_rx.pOutput = (uint8_t *)&prop_radio.rx_stats;
/* Init RF params and specify non-default params */
RF_Params rf_params;
RF_Params_init(&rf_params);
rf_params.nInactivityTimeout = 2000; /* 2 ms */
/* Open RF Driver */
prop_radio.rf_handle = RF_open(&prop_radio.rf_object, &rf_prop_mode,
(RF_RadioSetup*)&cmd_radio_setup, &rf_params);
if (prop_radio.rf_handle == NULL) {
return CMD_RESULT_ERROR;
}
2018-02-09 11:50:55 +00:00
2018-06-15 14:37:51 +00:00
set_channel(PROP_MODE_CHANNEL);
2018-02-09 11:50:55 +00:00
2018-06-15 14:37:51 +00:00
ENERGEST_ON(ENERGEST_TYPE_LISTEN);
2018-02-09 11:50:55 +00:00
2018-06-15 14:37:51 +00:00
/* Start RF process */
process_start(&rf_process, NULL);
2018-02-09 11:50:55 +00:00
2018-06-15 14:37:51 +00:00
return CMD_RESULT_OK;
2018-02-09 11:50:55 +00:00
}
/*---------------------------------------------------------------------------*/
2018-02-08 10:45:47 +00:00
const struct radio_driver prop_mode_driver = {
2018-06-15 14:37:51 +00:00
init,
2018-02-08 10:45:47 +00:00
prepare,
transmit,
send,
2018-06-15 14:37:51 +00:00
read,
2018-02-08 10:45:47 +00:00
channel_clear,
receiving_packet,
pending_packet,
2018-06-15 14:37:51 +00:00
on,
off,
2018-02-08 10:45:47 +00:00
get_value,
set_value,
get_object,
set_object,
};
/*---------------------------------------------------------------------------*/
/**
* @}
*/