297 lines
5.9 KiB
C

/*
* Copyright © 2014 Keith Packard <keithp@keithp.com>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
*/
#include <err.h>
#include <fcntl.h>
#include <gelf.h>
#include <stdio.h>
#include <stdint.h>
#include <stdlib.h>
#include <sysexits.h>
#include <unistd.h>
#include <getopt.h>
#include <string.h>
#include <stdbool.h>
#include <termios.h>
#include <math.h>
#include "ao-elf.h"
#include "ccdbg.h"
#include "cc-usb.h"
#include "cc.h"
#include "ao-verbose.h"
static const struct option options[] = {
{ .name = "tty", .has_arg = 1, .val = 'T' },
{ .name = "device", .has_arg = 1, .val = 'D' },
{ .name = "verbose", .has_arg = 0, .val = 'v' },
{ .name = "output", .has_arg = 1, .val = 'o' },
{ .name = "nosave", .has_arg = 0, .val = 'n' },
{ 0, 0, 0, 0},
};
static void usage(char *program)
{
fprintf(stderr, "usage: %s [--verbose] [--nosave] [--device=<device>] [-tty=<tty>] [--output=<cal-value-file>]\n", program);
exit(1);
}
static char **
tok(char *line) {
char **strs = malloc (sizeof (char *)), *str;
int n = 0;
while ((str = strtok(line, " \t"))) {
line = NULL;
strs = realloc(strs, (n + 2) * sizeof (char *));
strs[n] = strdup(str);
n++;
}
strs[n] = '\0';
return strs;
}
struct flash {
struct flash *next;
char line[512];
char **strs;
};
static struct flash *
flash(struct cc_usb *usb)
{
struct flash *head = NULL, **tail = &head;
cc_usb_printf(usb, "c s\nv\n");
for (;;) {
char line[512];
struct flash *b;
cc_usb_getline(usb, line, sizeof (line));
b = malloc (sizeof (struct flash));
strcpy(b->line, line);
b->strs = tok(line);
b->next = NULL;
*tail = b;
tail = &b->next;
if (strstr(line, "software-version"))
break;
}
return head;
}
static char **
find_flash(struct flash *b, char *word0) {
for (;b; b = b->next) {
if (strstr(b->line, word0))
return b->strs;
}
return NULL;
}
static int
do_save(struct cc_usb *usb)
{
int ret = 0;
printf("Saving calibration to device\n");
cc_usb_printf(usb, "c w\nv\n");
for (;;) {
char line[512];
cc_usb_getline(usb, line, sizeof (line));
if (strstr(line, "Nothing to save"))
ret = 1;
if (strstr(line, "Saved"))
ret = 1;
if (strstr(line, "software-version"))
break;
}
if (!ret) {
printf("Calibration save failed\n");
}
return ret;
}
static int
do_output(char *output, int cur_cal)
{
printf ("Saving calibration value to file \"%s\"\n", output);
FILE *out = fopen(output, "w");
int ret = 1;
if (!out) {
perror(output);
return 0;
}
if (fprintf(out, "%d\n", cur_cal) < 0) {
perror("fprintf");
ret = 0;
}
if (fflush(out) != 0) {
perror("fflush");
ret = 0;
}
if (fclose(out) != 0) {
perror("fclose");
ret = 0;
}
return ret;
}
static int
do_cal(char *tty, int save, char *output)
{
struct cc_usb *usb = NULL;
struct flash *b;
char line[1024];
double measured_freq;
char **cur_freq_words;
char **cur_cal_words;
char *line_end;
int cur_freq;
int cur_cal;
int new_cal;
int ret = 1;
int changed = 0;
for(;;) {
usb = cc_usb_open(tty);
if (!usb) {
fprintf(stderr, "failed to open device\n");
ret = 0;
break;
}
cc_usb_printf(usb, "E 0\n");
b = flash(usb);
cur_cal_words = find_flash(b, "Radio cal:");
cur_freq_words = find_flash(b, "Frequency:");
if (!cur_cal_words || !cur_freq_words) {
fprintf(stderr, "no response\n");
ret = 0;
break;
}
cur_cal = atoi(cur_cal_words[2]);
cur_freq = atoi(cur_freq_words[1]);
printf ("Current radio calibration %d\n", cur_cal);
printf ("Current radio frequency: %7.3f\n", cur_freq / 1000.0);
cc_usb_sync(usb);
cc_usb_printf(usb, "C 1\n");
cc_usb_sync(usb);
printf("Generating RF carrier. Please enter measured frequency [enter for done]: ");
fflush(stdout);
fgets(line, sizeof (line) - 1, stdin);
cc_usb_printf(usb, "C 0\n");
cc_usb_sync(usb);
measured_freq = strtod(line, &line_end);
if (line_end == line)
break;
new_cal = floor ((((double) cur_freq / 1000.0) / measured_freq) * cur_cal + 0.5);
if (new_cal == cur_cal) {
printf("Calibration value %d unchanged\n", cur_cal);
} else {
printf ("Setting cal value %d\n", new_cal);
cc_usb_printf (usb, "c f %d\n", new_cal);
changed = 1;
cc_usb_sync(usb);
}
cc_usb_close(usb);
}
if (usb) {
if (ret && save) {
if (changed) {
if (!do_save(usb))
ret = 0;
} else {
printf("Calibration unchanged, not saving\n");
}
}
if (ret && output) {
if (!do_output(output, cur_cal))
ret = 0;
}
cc_usb_close(usb);
}
return ret;
}
int
main (int argc, char **argv)
{
char *device = NULL;
int c;
char *tty = NULL;
int verbose = 0;
int save = 1;
int ret = 0;
char *output = NULL;
while ((c = getopt_long(argc, argv, "vnT:D:o:", options, NULL)) != -1) {
switch (c) {
case 'T':
tty = optarg;
break;
case 'D':
device = optarg;
break;
case 'v':
verbose++;
break;
case 'n':
save = 0;
break;
case 'o':
output = optarg;
break;
default:
usage(argv[0]);
break;
}
}
ao_verbose = verbose;
if (verbose)
ccdbg_add_debug(CC_DEBUG_BITBANG);
if (!tty)
tty = cc_usbdevs_find_by_arg(device, "AltosFlash");
if (!tty)
tty = cc_usbdevs_find_by_arg(device, "TeleMega");
if (!tty)
tty = getenv("ALTOS_TTY");
if (!tty)
tty="/dev/ttyACM0";
if (!do_cal(tty, save, output))
ret = 1;
return ret;
}