1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/spectrum-scanner/main.c
2017-10-23 10:06:19 +02:00

147 lines
5.3 KiB
C

/*
* Copyright (C) 2017 Eistec AB
*
* 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.
*/
/**
* @defgroup app_spectrum_scanner
* @brief Scanner application to find free channels
* @{
*
* @file
* @brief Spectrum scanner application for RIOT
*
* @author Joakim Nohlgård <joakim.nohlgard@eistec.se>
*
* @}
*/
#include <stdint.h>
#include <math.h>
#include "fmt.h"
#include "thread.h"
#include "xtimer.h"
#include "net/ieee802154.h"
#include "net/gnrc.h"
/* Scanning interval */
#define INTERVAL (500U * US_PER_MS)
/**
* @brief Measure the radio energy spectrum and print on stdout
*
* Algorithm description:
*
* The process will repeat as many measurements as possible during the
* measurement interval, before the average power is computed. This reduces the
* noise in the measurement and will yield a better image of what the radio
* environment contains.
*
* Still, 122 measurements per second (frdm-kw41z) and 128 us per measurement
* will only give a time coverage of about 1.5%, but because the measurements are
* spread out over time they should still give a good representation of which
* channels are free.
*
* Note that because the ED values are given in decibels, the average radio
* power is not the same as the arithmetic mean of the ED measurements. To
* compute the average of the dB measurements this algorithm requires both
* logarithm and exponentiation, quite heavy operations on the kinds of CPUs
* that RIOT targets. Increasing the CPU clock frequency may therefore reduce
* the noise in the output, because of the more frequent energy measurements
* possible.
*/
void spectrum_scanner(unsigned long interval_us)
{
kernel_pid_t ifs[GNRC_NETIF_NUMOF];
size_t netif_numof = gnrc_netif_get(ifs);
/* Using expf(x) (natural exponent) gives quicker computations on Cortex-M0+,
* compared to using powf(10, x). */
/*
* This was optimized by testing different combinations of expf, powf, logf, log10f:
*
* functions used | measurement iterations per 0.5 s on reference system (frdm-kw41z)
* ------------------------------------------------------------------
* expf, logf | 64
* powf, log10f | 46
* expf, log10f | 61
* no-op (baseline) | 83 (but the measurements are useless)
*/
while(1) {
/* Stack optimization, statically allocate this buffer */
static float ed_average[GNRC_NETIF_NUMOF][IEEE802154_CHANNEL_MAX+1];
memset(ed_average, 0, sizeof(ed_average));
uint64_t last_wakeup = xtimer_now_usec64();
uint64_t target = last_wakeup + interval_us;
/* We spin and try to do as many measurements as possible in the
* interval time */
unsigned int count = 0;
do {
for (unsigned int k = 0; k < netif_numof; ++k) {
kernel_pid_t dev = ifs[k];
for (unsigned int ch = IEEE802154_CHANNEL_MIN; ch <= IEEE802154_CHANNEL_MAX; ++ch) {
uint16_t tmp_ch = ch;
int res;
res = gnrc_netapi_set(dev, NETOPT_CHANNEL, 0, &tmp_ch, sizeof(uint16_t));
if (res < 0) {
continue;
}
netopt_enable_t tmp;
/* Perform CCA to update ED level */
res = gnrc_netapi_get(dev, NETOPT_IS_CHANNEL_CLR, 0, &tmp, sizeof(netopt_enable_t));
if (res < 0) {
continue;
}
int8_t level = 0;
res = gnrc_netapi_get(dev, NETOPT_LAST_ED_LEVEL, 0, &level, sizeof(int8_t));
if (res < 0) {
continue;
}
/* Convert dB to pseudo-energy before summing together the
* measurements. "Pseudo" because we use the natural
* exponential function e^x instead of computing 10^x which
* would be required if we needed the real measured energy.
* There is no need to know the real energy level because we
* will be converting back to dB again before printing. */
ed_average[k][ch] += expf((float)level / 128.f);
}
}
++count;
thread_yield();
} while(xtimer_now_usec64() < target);
for (unsigned int k = 0; k < netif_numof; ++k) {
print("[", 1);
print_u32_dec(k);
print(", ", 2);
print_u64_dec(target);
print(", ", 2);
print_u32_dec(count);
print("] ", 2);
for (unsigned int ch = IEEE802154_CHANNEL_MIN; ch <= IEEE802154_CHANNEL_MAX; ++ch) {
/* Compute the average pseudo-energy and convert back to dB */
ed_average[k][ch] = logf(ed_average[k][ch] / count) * 128.f;
print_u32_dec(ch);
print(": ", 2);
print_float(ed_average[k][ch], 4);
print(", ", 2);
}
print("\n", 1);
}
}
}
int main(void)
{
puts("RIOT scanner application");
spectrum_scanner(INTERVAL);
return 0;
}