1
0
mirror of git://projects.qi-hardware.com/ben-wpan.git synced 2024-11-29 10:01:53 +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:
Werner Almesberger 2011-04-13 18:40:39 -03:00
parent a7d9dfd258
commit a1fc867bf2
3 changed files with 167 additions and 26 deletions

View File

@ -15,6 +15,7 @@
#include <stdio.h> #include <stdio.h>
#include <unistd.h> #include <unistd.h>
#include <string.h> #include <string.h>
#include <ctype.h>
#include "at86rf230.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) struct sample *res)
{ {
int chan; int i, chan;
int fail = 0;
if (sweep->cont_tx && sweep->cont_tx != cont_tx) if (sweep->cont_tx && sweep->cont_tx != cont_tx)
return; return 0;
for (chan = 11; chan <= 26; chan++) { chan = 11;
for (i = 0; i != N_CHAN; i++) {
set_channel(sweep->rx, chan); set_channel(sweep->rx, chan);
set_channel(sweep->tx, chan); set_channel(sweep->tx, chan);
usleep(155); /* table 7-2, tTR19 */ usleep(155); /* table 7-2, tTR19 */
sample(sweep, cont_tx, res, chan == 11 && !sweep->cont_tx); 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; 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); int fail1, fail2;
do_half_sweep(sweep, CONT_TX_P500K, res+1);
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) 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; int i;
for (i = 0; i != sweeps; 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) static void usage(const char *name)
{ {
fprintf(stderr, fprintf(stderr,
@ -175,15 +271,16 @@ static void usage(const char *name)
"%6s %s -g common_args [[sweeps] samples]\n" "%6s %s -g common_args [[sweeps] samples]\n"
#endif #endif
"\n" "\n"
" common args: [-p power] [-t trim_tx [-t trim_rx]] [-T offset]\n" " common args: [-p power] [-P profile] [-t trim_tx [-t trim_rx]]\n"
" driver_tx[:arg] driver_rx[:arg]\n\n" " [-T offset] driver_tx[:arg] driver_rx[:arg]\n\n"
#ifdef HAVE_GFX #ifdef HAVE_GFX
" -g display results graphically\n" " -g display results graphically\n"
#endif #endif
" -p power transmit power, 0 to 15 (default %d)\n" " -p power transmit power, 0 to 15 (default %d)\n"
" -t trim trim capacitor, 0 to 15 (default %d)\n" " -P profile load profile for pass/fail decisions\n"
" -T offset constant wave offset in MHz, -0.5 or +0.5 (default: scan both)\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, , name,
#ifdef HAVE_GFX #ifdef HAVE_GFX
@ -206,12 +303,13 @@ int main(int argc, char **argv)
}; };
int graphical = 0; int graphical = 0;
int power = DEFAULT_POWER; int power = DEFAULT_POWER;
const char *profile = NULL;
int sweeps = 1; int sweeps = 1;
unsigned long tmp; unsigned long tmp;
char *end; char *end;
int c; int c;
while ((c = getopt(argc, argv, "gp:t:T:")) != EOF) while ((c = getopt(argc, argv, "gp:P:t:T:")) != EOF)
switch (c) { switch (c) {
case'g': case'g':
graphical = 1; graphical = 1;
@ -223,6 +321,9 @@ int main(int argc, char **argv)
usage(*argv); usage(*argv);
power = tmp; power = tmp;
break; break;
case 'P':
profile = optarg;
break;
case 't': case 't':
tmp = strtoul(optarg, &end, 0); tmp = strtoul(optarg, &end, 0);
if (*end || tmp > 15) if (*end || tmp > 15)
@ -270,6 +371,8 @@ int main(int argc, char **argv)
usage(*argv); usage(*argv);
} }
read_profile(profile, &sweep);
sweep.tx = atrf_open(tx_drv); sweep.tx = atrf_open(tx_drv);
if (!sweep.tx) if (!sweep.tx)
return 1; return 1;

View File

@ -29,11 +29,10 @@
#define XRES 320 #define XRES 320
#define YRES 240 #define YRES 240
#define N_CHAN 16
#define FG_RGBA 0xffffffff /* measurement color */ #define FG_RGBA 0xffffffff /* measurement color */
#define OK_RGBA 0x00ff00ff #define OK_RGBA 0x00ff00ff
#define OVER_RGBA 0xffff00ff
#define UNDER_RGBA 0xff0000ff
#define CHAN_STEP 20 /* 4 pixels/MHz */ #define CHAN_STEP 20 /* 4 pixels/MHz */
#define SIDE_STEP 2 #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) static void clear(SDL_Surface *s)
{ {
SDL_FillRect(s, NULL, SDL_MapRGB(s->format, 0, 0, 0)); 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_Surface *surf;
SDL_Event event; SDL_Event event;
int cycle = 0; int cycle = 0;
int fail;
if (SDL_Init(SDL_INIT_VIDEO) < 0) { if (SDL_Init(SDL_INIT_VIDEO) < 0) {
fprintf(stderr, "SDL_init: %s\n", SDL_GetError()); fprintf(stderr, "SDL_init: %s\n", SDL_GetError());
@ -145,19 +172,14 @@ void gui(const struct sweep *sweep, int sweeps)
event.type == SDL_QUIT) event.type == SDL_QUIT)
return; return;
tstart(); tstart();
do_sweep(sweep, res); fail = do_sweep(sweep, res);
tstop(); tstop();
SDL_LockSurface(surf); SDL_LockSurface(surf);
clear(surf); clear(surf);
if (cycle++ & 1) { indicate(surf, fail);
filledCircleColor(surf, STATUS_X, STATUS_Y, STATUS_R,
OK_RGBA);
aacircleColor(surf, STATUS_X, STATUS_Y, STATUS_R,
OK_RGBA);
}
draw(surf, res, sweep->cont_tx); draw(surf, res, sweep->cont_tx);
SDL_UnlockSurface(surf); SDL_UnlockSurface(surf);

View File

@ -19,6 +19,12 @@
#include "atrf.h" #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 sweep {
struct atrf_dsc *tx; struct atrf_dsc *tx;
struct atrf_dsc *rx; struct atrf_dsc *rx;
@ -27,6 +33,8 @@ struct sweep {
int power; int power;
uint8_t cont_tx; uint8_t cont_tx;
int samples; int samples;
double min[N_CHAN];
double max[N_CHAN];
}; };
struct sample { 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 */ #endif /* !SWEEP_H */