1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/cpu/cc2538/radio/cc2538_rf_getset.c
2016-07-25 18:35:05 -07:00

256 lines
6.3 KiB
C

/*
* Copyright (C) 2016 MUTEX NZ Ltd.
* Copyright (C) 2015 Loci Controls Inc.
*
* 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 cpu_cc2538
* @{
*
* @file
* @brief Getter and setter functions for the cc2538_rf driver
*
* @author Aaron Sowry <aaron@mutex.nz>
* @author Ian Martin <ian@locicontrols.com>
*
* @}
*/
#include "cc2538_rf.h"
#define ENABLE_DEBUG (0)
#include "debug.h"
/* static const __flash uint8_t? */
static const uint8_t power_lut[NUM_POWER_LEVELS] = {
0, /**< -24 dBm */
7, /**< -23 dBm */
15, /**< -22 dBm */
22, /**< -21 dBm */
29, /**< -20 dBm */
37, /**< -19 dBm */
44, /**< -18 dBm */
51, /**< -17 dBm */
59, /**< -16 dBm */
66, /**< -15 dBm */
77, /**< -14 dBm */
88, /**< -13 dBm */
93, /**< -12 dBm */
98, /**< -11 dBm */
106, /**< -10 dBm */
114, /**< -9 dBm */
125, /**< -8 dBm */
136, /**< -7 dBm */
141, /**< -6 dBm */
145, /**< -5 dBm */
153, /**< -4 dBm */
161, /**< -3 dBm */
169, /**< -2 dBm */
176, /**< -1 dBm */
182, /**< 0 dBm */
197, /**< 1 dBm */
205, /**< 2 dBm */
213, /**< 3 dBm */
225, /**< 4 dBm */
237, /**< 5 dBm */
246, /**< 6 dBm */
255, /**< 7 dBm */
};
uint64_t cc2538_get_addr_long(void)
{
uint64_t addr = RFCORE_FFSM_EXT_ADDR0;
addr <<= 8;
addr |= RFCORE_FFSM_EXT_ADDR1;
addr <<= 8;
addr |= RFCORE_FFSM_EXT_ADDR2;
addr <<= 8;
addr |= RFCORE_FFSM_EXT_ADDR3;
addr <<= 8;
addr |= RFCORE_FFSM_EXT_ADDR4;
addr <<= 8;
addr |= RFCORE_FFSM_EXT_ADDR5;
addr <<= 8;
addr |= RFCORE_FFSM_EXT_ADDR6;
addr <<= 8;
addr |= RFCORE_FFSM_EXT_ADDR7;
return addr;
}
uint16_t cc2538_get_addr_short(void)
{
return (RFCORE_FFSM_SHORT_ADDR0 << 8) | RFCORE_FFSM_SHORT_ADDR1;
}
unsigned int cc2538_get_chan(void)
{
return IEEE802154_FREQ2CHAN(CC2538_MIN_FREQ + RFCORE_XREG_FREQCTRL);
}
uint64_t cc2538_get_eui64_primary(void)
{
/*
* The primary EUI-64 seems to be written to memory in a non-sequential
* byte order, with both 4-byte halves of the address flipped.
*/
uint64_t eui64 = ((uint8_t*)CC2538_EUI64_LOCATION_PRI)[4];
eui64 <<= 8;
eui64 |= ((uint8_t*)CC2538_EUI64_LOCATION_PRI)[5];
eui64 <<= 8;
eui64 |= ((uint8_t*)CC2538_EUI64_LOCATION_PRI)[6];
eui64 <<= 8;
eui64 |= ((uint8_t*)CC2538_EUI64_LOCATION_PRI)[7];
eui64 <<= 8;
eui64 |= ((uint8_t*)CC2538_EUI64_LOCATION_PRI)[0];
eui64 <<= 8;
eui64 |= ((uint8_t*)CC2538_EUI64_LOCATION_PRI)[1];
eui64 <<= 8;
eui64 |= ((uint8_t*)CC2538_EUI64_LOCATION_PRI)[2];
eui64 <<= 8;
eui64 |= ((uint8_t*)CC2538_EUI64_LOCATION_PRI)[3];
return eui64;
}
bool cc2538_get_monitor(void)
{
return NOT(RFCORE->XREG_FRMFILT0bits.FRAME_FILTER_EN);
}
uint16_t cc2538_get_pan(void)
{
return (RFCORE_FFSM_PAN_ID1 << 8) | RFCORE_FFSM_PAN_ID0;
}
int cc2538_get_tx_power(void)
{
int index;
int best_index = 0;
int best_delta = INT_MAX;
int txpower;
txpower = RFCORE_XREG_TXPOWER & 0xff;
for (index = 0; index < NUM_POWER_LEVELS; index++) {
int delta = ABS_DIFF(power_lut[index], txpower);
if (delta < best_delta) {
best_delta = delta;
best_index = index;
}
}
return OUTPUT_POWER_MIN + best_index;
}
void cc2538_set_addr_long(uint64_t addr)
{
RFCORE_FFSM_EXT_ADDR0 = addr >> (7 * 8);
RFCORE_FFSM_EXT_ADDR1 = addr >> (6 * 8);
RFCORE_FFSM_EXT_ADDR2 = addr >> (5 * 8);
RFCORE_FFSM_EXT_ADDR3 = addr >> (4 * 8);
RFCORE_FFSM_EXT_ADDR4 = addr >> (3 * 8);
RFCORE_FFSM_EXT_ADDR5 = addr >> (2 * 8);
RFCORE_FFSM_EXT_ADDR6 = addr >> (1 * 8);
RFCORE_FFSM_EXT_ADDR7 = addr >> (0 * 8);
}
void cc2538_set_addr_short(uint16_t addr)
{
RFCORE_FFSM_SHORT_ADDR1 = addr;
RFCORE_FFSM_SHORT_ADDR0 = addr >> 8;
}
void cc2538_set_chan(unsigned int chan)
{
DEBUG("%s(%u): Setting channel to ", __FUNCTION__, chan);
if (chan < IEEE802154_MIN_CHANNEL) {
chan = IEEE802154_MIN_CHANNEL;
}
else if (chan > IEEE802154_MAX_CHANNEL) {
chan = IEEE802154_MAX_CHANNEL;
}
DEBUG("%i (range %i-%i)\n", chan, IEEE802154_MIN_CHANNEL,
IEEE802154_MAX_CHANNEL);
cc2538_set_freq(IEEE802154_CHAN2FREQ(chan));
}
void cc2538_set_freq(unsigned int MHz)
{
DEBUG("%s(%u): Setting frequency to ", __FUNCTION__, MHz);
if (MHz < IEEE802154_MIN_FREQ) {
MHz = IEEE802154_MIN_FREQ;
}
else if (MHz > IEEE802154_MAX_FREQ) {
MHz = IEEE802154_MAX_FREQ;
}
DEBUG("%i (range %i-%i)\n", MHz, IEEE802154_MIN_FREQ, IEEE802154_MAX_FREQ);
RFCORE_XREG_FREQCTRL = MHz - CC2538_MIN_FREQ;
}
void cc2538_set_monitor(bool mode)
{
RFCORE->XREG_FRMFILT0bits.FRAME_FILTER_EN = NOT(mode);
}
void cc2538_set_state(cc2538_rf_t *dev, netopt_state_t state)
{
switch (state) {
case NETOPT_STATE_OFF:
case NETOPT_STATE_SLEEP:
cc2538_off();
dev->state = state;
break;
case NETOPT_STATE_RX:
case NETOPT_STATE_IDLE:
if (!cc2538_is_on()) {
cc2538_on();
RFCORE_WAIT_UNTIL(RFCORE->XREG_FSMSTAT0bits.FSM_FFCTRL_STATE > FSM_STATE_RX_CALIBRATION);
}
dev->state = state;
break;
case NETOPT_STATE_TX:
dev->state = NETOPT_STATE_IDLE;
break;
case NETOPT_STATE_RESET:
cc2538_off();
cc2538_on();
RFCORE_WAIT_UNTIL(RFCORE->XREG_FSMSTAT0bits.FSM_FFCTRL_STATE > FSM_STATE_RX_CALIBRATION);
dev->state = NETOPT_STATE_IDLE;
break;
}
}
void cc2538_set_pan(uint16_t pan)
{
RFCORE_FFSM_PAN_ID0 = pan;
RFCORE_FFSM_PAN_ID1 = pan >> 8;
}
void cc2538_set_tx_power(int dBm)
{
DEBUG("%s(%i): Setting TX power to ", __FUNCTION__, dBm);
if (dBm < OUTPUT_POWER_MIN) {
dBm = OUTPUT_POWER_MIN;
}
else if (dBm > OUTPUT_POWER_MAX) {
dBm = OUTPUT_POWER_MAX;
}
DEBUG("%idBm (range %i-%i dBm)\n", dBm, OUTPUT_POWER_MIN, OUTPUT_POWER_MAX);
RFCORE_XREG_TXPOWER = power_lut[dBm - OUTPUT_POWER_MIN];
}