mirror of
git://projects.qi-hardware.com/ben-wpan.git
synced 2024-11-25 18:57:11 +02:00
atrf-path: option -P to load a min/max profile; pass/fail indication in GUI
- sweep.h (MIN_DIFF, MAX_DIFF, struct sweep): added min/max profile - sweep.h (do_sweep), atrf-path.c (do_half_sweep, do_sweep): compare result against limits and return pass/fail decision - sweep.h (N_CHAN), atrf-path.c (do_half_sweep, do_sweeps), gui.c (N_CHAN): declare number of channels in one central place instead of scattering it all around the program - atrf-path.c (do_read_profile, read_profile, usage, main): new option -P to read a min/max profile - gui.c (indicate, main): moved indicator to separate function and improved blink logic - gui.c (OVER_RGBA, UNDER_RGBA, indicate, main): change color to indicate pass/fail
This commit is contained in:
parent
a7d9dfd258
commit
a1fc867bf2
@ -15,6 +15,7 @@
|
||||
#include <stdio.h>
|
||||
#include <unistd.h>
|
||||
#include <string.h>
|
||||
#include <ctype.h>
|
||||
|
||||
#include "at86rf230.h"
|
||||
|
||||
@ -110,28 +111,44 @@ static void sample(const struct sweep *sweep, int cont_tx,
|
||||
}
|
||||
|
||||
|
||||
static void do_half_sweep(const struct sweep *sweep, int cont_tx,
|
||||
static int do_half_sweep(const struct sweep *sweep, int cont_tx,
|
||||
struct sample *res)
|
||||
{
|
||||
int chan;
|
||||
int i, chan;
|
||||
int fail = 0;
|
||||
|
||||
if (sweep->cont_tx && sweep->cont_tx != cont_tx)
|
||||
return;
|
||||
for (chan = 11; chan <= 26; chan++) {
|
||||
return 0;
|
||||
chan = 11;
|
||||
for (i = 0; i != N_CHAN; i++) {
|
||||
set_channel(sweep->rx, chan);
|
||||
set_channel(sweep->tx, chan);
|
||||
usleep(155); /* table 7-2, tTR19 */
|
||||
|
||||
sample(sweep, cont_tx, res, chan == 11 && !sweep->cont_tx);
|
||||
if (res->avg > sweep->max[i])
|
||||
fail = 1;
|
||||
if (!fail && res->avg < sweep->min[i])
|
||||
fail = -1;
|
||||
|
||||
res += 2;
|
||||
chan++;
|
||||
}
|
||||
return fail;
|
||||
}
|
||||
|
||||
|
||||
void do_sweep(const struct sweep *sweep, struct sample *res)
|
||||
int do_sweep(const struct sweep *sweep, struct sample *res)
|
||||
{
|
||||
do_half_sweep(sweep, CONT_TX_M500K, res);
|
||||
do_half_sweep(sweep, CONT_TX_P500K, res+1);
|
||||
int fail1, fail2;
|
||||
|
||||
fail1 = do_half_sweep(sweep, CONT_TX_M500K, res);
|
||||
fail2 = do_half_sweep(sweep, CONT_TX_P500K, res+1);
|
||||
if (fail1 > 0 || fail2 > 0)
|
||||
return 1;
|
||||
if (fail1 < 0 || fail2 < 0)
|
||||
return -1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
@ -154,7 +171,7 @@ static void print_sweep(const struct sweep *sweep, const struct sample *res)
|
||||
|
||||
static void do_sweeps(const struct sweep *sweep, int sweeps)
|
||||
{
|
||||
struct sample res[16*2]; /* 16 channels, 2 offsets */
|
||||
struct sample res[N_CHAN*2]; /* 2 offsets per channel */
|
||||
int i;
|
||||
|
||||
for (i = 0; i != sweeps; i++) {
|
||||
@ -166,6 +183,85 @@ static void do_sweeps(const struct sweep *sweep, int sweeps)
|
||||
}
|
||||
|
||||
|
||||
static int do_read_profile(const char *name, struct sweep *sweep)
|
||||
{
|
||||
FILE *file;
|
||||
char buf[300];
|
||||
int got;
|
||||
char *p;
|
||||
double min = MIN_DIFF, max = MAX_DIFF;
|
||||
int n = 0;
|
||||
|
||||
file = fopen(name, "r");
|
||||
if (!file) {
|
||||
perror(name);
|
||||
exit(1);
|
||||
}
|
||||
while (fgets(buf, sizeof(buf), file)) {
|
||||
p = strchr(buf, '\n');
|
||||
if (p)
|
||||
*p = 0;
|
||||
p = strchr(buf, '#');
|
||||
if (p)
|
||||
*p = 0;
|
||||
for (p = buf; *p && isspace(*p); p++);
|
||||
if (!*p)
|
||||
continue;
|
||||
got = sscanf(buf, "%lf %lf", &min, &max);
|
||||
switch (got) {
|
||||
case 0:
|
||||
fprintf(stderr, "can't parse \"%s\"\n", buf);
|
||||
exit(1);
|
||||
case 1:
|
||||
max = MAX_DIFF;
|
||||
/* fall through */
|
||||
case 2:
|
||||
if (min < MIN_DIFF) {
|
||||
fprintf(stderr, "minimum is %g dBm\n",
|
||||
MIN_DIFF);
|
||||
exit(1);
|
||||
}
|
||||
if (max > MAX_DIFF) {
|
||||
fprintf(stderr, "maximum is %g dBm\n",
|
||||
MAX_DIFF);
|
||||
exit(1);
|
||||
}
|
||||
if (min > max) {
|
||||
fprintf(stderr, "lower bound > upper bound\n");
|
||||
exit(1);
|
||||
}
|
||||
if (n == N_CHAN) {
|
||||
fprintf(stderr, "too many channels\n");
|
||||
exit(1);
|
||||
}
|
||||
sweep->min[n] = min;
|
||||
sweep->max[n] = max;
|
||||
n++;
|
||||
break;
|
||||
default:
|
||||
abort();
|
||||
}
|
||||
}
|
||||
fclose(file);
|
||||
return n;
|
||||
}
|
||||
|
||||
|
||||
static void read_profile(const char *name, struct sweep *sweep)
|
||||
{
|
||||
int n = 0;
|
||||
|
||||
if (name)
|
||||
n = do_read_profile(name, sweep);
|
||||
|
||||
while (n != N_CHAN) {
|
||||
sweep->min[n] = MIN_DIFF;
|
||||
sweep->max[n] = MAX_DIFF;
|
||||
n++;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
static void usage(const char *name)
|
||||
{
|
||||
fprintf(stderr,
|
||||
@ -175,15 +271,16 @@ static void usage(const char *name)
|
||||
"%6s %s -g common_args [[sweeps] samples]\n"
|
||||
#endif
|
||||
"\n"
|
||||
" common args: [-p power] [-t trim_tx [-t trim_rx]] [-T offset]\n"
|
||||
" driver_tx[:arg] driver_rx[:arg]\n\n"
|
||||
" common args: [-p power] [-P profile] [-t trim_tx [-t trim_rx]]\n"
|
||||
" [-T offset] driver_tx[:arg] driver_rx[:arg]\n\n"
|
||||
|
||||
#ifdef HAVE_GFX
|
||||
" -g display results graphically\n"
|
||||
" -g display results graphically\n"
|
||||
#endif
|
||||
" -p power transmit power, 0 to 15 (default %d)\n"
|
||||
" -t trim trim capacitor, 0 to 15 (default %d)\n"
|
||||
" -T offset constant wave offset in MHz, -0.5 or +0.5 (default: scan both)\n"
|
||||
" -p power transmit power, 0 to 15 (default %d)\n"
|
||||
" -P profile load profile for pass/fail decisions\n"
|
||||
" -t trim trim capacitor, 0 to 15 (default %d)\n"
|
||||
" -T offset constant wave offset in MHz, -0.5 or +0.5 (default: scan both)\n"
|
||||
|
||||
, name,
|
||||
#ifdef HAVE_GFX
|
||||
@ -206,12 +303,13 @@ int main(int argc, char **argv)
|
||||
};
|
||||
int graphical = 0;
|
||||
int power = DEFAULT_POWER;
|
||||
const char *profile = NULL;
|
||||
int sweeps = 1;
|
||||
unsigned long tmp;
|
||||
char *end;
|
||||
int c;
|
||||
|
||||
while ((c = getopt(argc, argv, "gp:t:T:")) != EOF)
|
||||
while ((c = getopt(argc, argv, "gp:P:t:T:")) != EOF)
|
||||
switch (c) {
|
||||
case'g':
|
||||
graphical = 1;
|
||||
@ -223,6 +321,9 @@ int main(int argc, char **argv)
|
||||
usage(*argv);
|
||||
power = tmp;
|
||||
break;
|
||||
case 'P':
|
||||
profile = optarg;
|
||||
break;
|
||||
case 't':
|
||||
tmp = strtoul(optarg, &end, 0);
|
||||
if (*end || tmp > 15)
|
||||
@ -270,6 +371,8 @@ int main(int argc, char **argv)
|
||||
usage(*argv);
|
||||
}
|
||||
|
||||
read_profile(profile, &sweep);
|
||||
|
||||
sweep.tx = atrf_open(tx_drv);
|
||||
if (!sweep.tx)
|
||||
return 1;
|
||||
|
@ -29,11 +29,10 @@
|
||||
#define XRES 320
|
||||
#define YRES 240
|
||||
|
||||
#define N_CHAN 16
|
||||
|
||||
|
||||
#define FG_RGBA 0xffffffff /* measurement color */
|
||||
#define OK_RGBA 0x00ff00ff
|
||||
#define OVER_RGBA 0xffff00ff
|
||||
#define UNDER_RGBA 0xff0000ff
|
||||
|
||||
#define CHAN_STEP 20 /* 4 pixels/MHz */
|
||||
#define SIDE_STEP 2
|
||||
@ -84,6 +83,33 @@ static void draw(SDL_Surface *s, const struct sample *res, int cont_tx)
|
||||
}
|
||||
|
||||
|
||||
static void indicate(SDL_Surface *s, int fail)
|
||||
{
|
||||
static uint32_t last = 0;
|
||||
uint32_t color;
|
||||
|
||||
switch (fail) {
|
||||
case 0:
|
||||
color = OK_RGBA;
|
||||
break;
|
||||
case 1:
|
||||
color = OVER_RGBA;
|
||||
break;
|
||||
case -1:
|
||||
color = UNDER_RGBA;
|
||||
break;
|
||||
default:
|
||||
abort();
|
||||
}
|
||||
if (color == last)
|
||||
color = 0;
|
||||
last = color;
|
||||
|
||||
filledCircleColor(s, STATUS_X, STATUS_Y, STATUS_R, color);
|
||||
aacircleColor(s, STATUS_X, STATUS_Y, STATUS_R, color);
|
||||
}
|
||||
|
||||
|
||||
static void clear(SDL_Surface *s)
|
||||
{
|
||||
SDL_FillRect(s, NULL, SDL_MapRGB(s->format, 0, 0, 0));
|
||||
@ -124,6 +150,7 @@ void gui(const struct sweep *sweep, int sweeps)
|
||||
SDL_Surface *surf;
|
||||
SDL_Event event;
|
||||
int cycle = 0;
|
||||
int fail;
|
||||
|
||||
if (SDL_Init(SDL_INIT_VIDEO) < 0) {
|
||||
fprintf(stderr, "SDL_init: %s\n", SDL_GetError());
|
||||
@ -145,19 +172,14 @@ void gui(const struct sweep *sweep, int sweeps)
|
||||
event.type == SDL_QUIT)
|
||||
return;
|
||||
tstart();
|
||||
do_sweep(sweep, res);
|
||||
fail = do_sweep(sweep, res);
|
||||
tstop();
|
||||
|
||||
SDL_LockSurface(surf);
|
||||
|
||||
clear(surf);
|
||||
|
||||
if (cycle++ & 1) {
|
||||
filledCircleColor(surf, STATUS_X, STATUS_Y, STATUS_R,
|
||||
OK_RGBA);
|
||||
aacircleColor(surf, STATUS_X, STATUS_Y, STATUS_R,
|
||||
OK_RGBA);
|
||||
}
|
||||
indicate(surf, fail);
|
||||
draw(surf, res, sweep->cont_tx);
|
||||
|
||||
SDL_UnlockSurface(surf);
|
||||
|
@ -19,6 +19,12 @@
|
||||
#include "atrf.h"
|
||||
|
||||
|
||||
#define N_CHAN 16
|
||||
|
||||
#define MIN_DIFF -97.0 /* RSSI(min)-TX(max) = -94 - 3 */
|
||||
#define MAX_DIFF 7.0 /* RSSI(max)-TX(min) = -10 - (-17) */
|
||||
|
||||
|
||||
struct sweep {
|
||||
struct atrf_dsc *tx;
|
||||
struct atrf_dsc *rx;
|
||||
@ -27,6 +33,8 @@ struct sweep {
|
||||
int power;
|
||||
uint8_t cont_tx;
|
||||
int samples;
|
||||
double min[N_CHAN];
|
||||
double max[N_CHAN];
|
||||
};
|
||||
|
||||
struct sample {
|
||||
@ -35,6 +43,14 @@ struct sample {
|
||||
};
|
||||
|
||||
|
||||
void do_sweep(const struct sweep *sweep, struct sample *res);
|
||||
/*
|
||||
* do_sweep returns whether the signal is within the limits:
|
||||
*
|
||||
* 1: at least one sample is above the maximum
|
||||
* 0: all samples are between minimum and maximum
|
||||
* -1: at least one sample below the minimum, and none above the maximum
|
||||
*/
|
||||
|
||||
int do_sweep(const struct sweep *sweep, struct sample *res);
|
||||
|
||||
#endif /* !SWEEP_H */
|
||||
|
Loading…
Reference in New Issue
Block a user