Skip to content

Commit

Permalink
Wait for a valid RSSI reading in CC13xx/CC26xx RF drivers
Browse files Browse the repository at this point in the history
As discussed in #1341, the current CC13xx/CC26xx IEEE mode driver sends `CMD_GET_RSSI` once and returns the RSSI reading uncondtionally. This happens within the `get_rssi()` function.

This logic is broken if `get_rssi()` is called with the radio off. The function will make sure to turn on the radio first, but it does not make sure the RSSI reading is valid, which only happens a number of symbol periods after the radio enters RX. The outcome is that `NETSTACK_RADIO.get_value(RADIO_PARAM_RSSI, ...)` will always return -128 (meaning that RSSI is unavailable) if the radio was off at the time of calling the function.

The same condition affects the prop mode driver.

This commit changes the logic of `get_rssi()`:
* For PROP mode, if `CMD_GET_RSSI` returns an invalid RSSI, we send it again. For unknown reasons, `CMD_GET_RSSI` on occasion returns 0, so we ignore that value too.
* For IEEE mode, we use `CMD_IEEE_CCA_REQ` and we inspect the value of `ccaInfo.ccaEnergy` of the return structure. If the value is 0x02 (Invalid), we send the command again.

Fixes #1341
  • Loading branch information
g-oikonomou committed Jul 17, 2016
1 parent c05665a commit 16f56ab
Show file tree
Hide file tree
Showing 2 changed files with 25 additions and 14 deletions.
21 changes: 13 additions & 8 deletions cpu/cc26xx-cc13xx/rf-core/ieee-mode.c
Original file line number Diff line number Diff line change
Expand Up @@ -384,9 +384,8 @@ static radio_value_t
get_rssi(void)
{
uint32_t cmd_status;
int8_t rssi;
uint8_t was_off = 0;
rfc_CMD_GET_RSSI_t cmd;
rfc_CMD_IEEE_CCA_REQ_t cmd;

/* If we are off, turn on first */
if(!rf_is_on()) {
Expand All @@ -398,21 +397,27 @@ get_rssi(void)
}

memset(&cmd, 0x00, sizeof(cmd));
cmd.commandNo = CMD_GET_RSSI;
cmd.ccaInfo.ccaEnergy = RF_CMD_CCA_REQ_CCA_STATE_INVALID;

rssi = RF_CMD_CCA_REQ_RSSI_UNKNOWN;
while(cmd.ccaInfo.ccaEnergy == RF_CMD_CCA_REQ_CCA_STATE_INVALID) {
memset(&cmd, 0x00, sizeof(cmd));
cmd.commandNo = CMD_IEEE_CCA_REQ;

if(rf_core_send_cmd((uint32_t)&cmd, &cmd_status) == RF_CORE_CMD_OK) {
/* Current RSSI in bits 23:16 of cmd_status */
rssi = (cmd_status >> 16) & 0xFF;
if(rf_core_send_cmd((uint32_t)&cmd, &cmd_status) == RF_CORE_CMD_ERROR) {
PRINTF("get_rssi: CMDSTA=0x%08lx\n", cmd_status);

/* Make sure to return RSSI unknown */
cmd.currentRssi = RF_CMD_CCA_REQ_RSSI_UNKNOWN;
break;
}
}

/* If we were off, turn back off */
if(was_off) {
off();
}

return rssi;
return cmd.currentRssi;
}
/*---------------------------------------------------------------------------*/
/* Returns the current TX power in dBm */
Expand Down
18 changes: 12 additions & 6 deletions cpu/cc26xx-cc13xx/rf-core/prop-mode.c
Original file line number Diff line number Diff line change
Expand Up @@ -271,6 +271,7 @@ get_rssi(void)
{
uint32_t cmd_status;
int8_t rssi;
uint8_t attempts = 0;
uint8_t was_off = 0;
rfc_CMD_GET_RSSI_t cmd;

Expand All @@ -283,14 +284,19 @@ get_rssi(void)
}
}

memset(&cmd, 0x00, sizeof(cmd));
cmd.commandNo = CMD_GET_RSSI;

rssi = RF_CMD_CCA_REQ_RSSI_UNKNOWN;

if(rf_core_send_cmd((uint32_t)&cmd, &cmd_status) == RF_CORE_CMD_OK) {
/* Current RSSI in bits 23:16 of cmd_status */
rssi = (cmd_status >> 16) & 0xFF;
while((rssi == RF_CMD_CCA_REQ_RSSI_UNKNOWN || rssi == 0) && ++attempts < 10) {
memset(&cmd, 0x00, sizeof(cmd));
cmd.commandNo = CMD_GET_RSSI;

if(rf_core_send_cmd((uint32_t)&cmd, &cmd_status) == RF_CORE_CMD_ERROR) {
PRINTF("get_rssi: CMDSTA=0x%08lx\n", cmd_status);
break;
} else {
/* Current RSSI in bits 23:16 of cmd_status */
rssi = (cmd_status >> 16) & 0xFF;
}
}

/* If we were off, turn back off */
Expand Down

0 comments on commit 16f56ab

Please sign in to comment.